From 3d02d70eab669b72429cb6065e65b817ba34c1dd Mon Sep 17 00:00:00 2001 From: Ryan Pavlik Date: Mon, 10 Jul 2023 17:26:14 -0500 Subject: [PATCH 1/4] Changes from OpenXR 1.0.28 --- .azure-pipelines/shared/build_jobs.yml | 5 +- .../shared/check_clang_format.yml | 3 +- .azure-pipelines/shared/check_file_format.yml | 3 +- .azure-pipelines/shared/codespell.yml | 3 +- .gitattributes | 5 +- .github/workflows/android-cts-build.yml | 2 +- .github/workflows/android-cts-pr.yml | 2 +- .github/workflows/msvc-build-preset.yml | 2 +- .gitignore | 1 + .reuse/dep5 | 25 +- BUILDING.md | 15 + CMakeLists.txt | 2 + LICENSES/Zlib.txt | 11 + changes/conformance/mr.2510.gl.md | 5 + changes/conformance/mr.2669.gl.md | 1 + changes/conformance/mr.2689.gl.md | 1 + changes/conformance/mr.2704.gl.md | 6 + changes/conformance/mr.2717.gl.md | 1 + changes/conformance/mr.2725.gl.md | 1 + changes/conformance/mr.2729.gl.md | 1 + changes/conformance/mr.2730.gl.md | 1 + changes/conformance/mr.2735.gl.md | 5 + changes/conformance/mr.2736.gl.md | 4 + changes/conformance/mr.2737.gl.md | 1 + changes/conformance/mr.2746.gl.md | 1 + changes/conformance/mr.2756.gl.md | 1 + changes/conformance/mr.2766.gl.md | 1 + include/openxr/openxr_platform_defines.h | 4 + maintainer-scripts/common.sh | 43 +- runClangFormat.sh | 10 +- specification/Makefile | 30 +- specification/registry/xr.xml | 1137 +- specification/scripts/cgenerator.py | 4 +- specification/scripts/docgenerator.py | 2 + specification/scripts/findBareNormatives.sh | 82 + specification/scripts/genRef.py | 4 + specification/scripts/reg.py | 4 + specification/scripts/reserve_extensions.py | 32 + .../{openxr-macros.rb => spec-macros.rb} | 3 +- .../extension.rb | 132 +- .../scripts/spec_tools/conventions.py | 2 +- .../scripts/spec_tools/macro_checker_file.py | 6 +- .../scripts/test_check_spec_links.py | 4 +- specification/scripts/xml_consistency.py | 20 +- specification/scripts/xrconventions.py | 2 +- src/CMakeLists.txt | 34 +- src/common/filesystem_utils.cpp | 2 +- src/common/filesystem_utils.hpp | 2 +- src/common/gfxwrapper_opengl.c | 16 +- src/common/gfxwrapper_opengl.h | 6 +- src/common/object_info.cpp | 4 +- src/common/platform_utils.hpp | 5 +- src/conformance/CMakeLists.txt | 2 + src/conformance/build.gradle | 7 +- src/conformance/checkSchema.sh | 2 +- .../conformance_cli/CMakeLists.txt | 2 + src/conformance/conformance_cli/main.cpp | 2 +- .../conformance_layer/CMakeLists.txt | 8 + .../conformance_layer/HandleState.cpp | 15 +- .../conformance_layer/HandleState.h | 1 - .../conformance_layer/Negotiate.cpp | 8 +- .../XrApiLayer_conformance_layer.expsym | 5 + .../conformance_test/CMakeLists.txt | 53 +- .../ext_plane_detection_contour.png | Bin 0 -> 143122 bytes .../ext_plane_detection_contour.png.license | 3 + .../palm_pose.png.license | 2 +- .../conformance_test/conformance_test.cpp | 47 +- .../conformance_test/conformance_test.h | 2 +- .../conformance_test/test_ActionPoses.cpp | 28 +- .../conformance_test/test_FrameSubmission.cpp | 20 +- .../conformance_test/test_HapticInterrupt.cpp | 25 +- .../test_InteractiveThrow.cpp | 16 +- .../test_LayerComposition.cpp | 19 +- .../conformance_test/test_SessionState.cpp | 19 +- .../conformance_test/test_Swapchains.cpp | 103 +- .../test_ViewConfigurations.cpp | 13 +- .../test_XR_EXT_debug_utils.cpp | 20 +- .../test_XR_EXT_dpad_binding.cpp | 22 +- .../test_XR_EXT_eye_gaze_interaction.cpp | 17 +- .../test_XR_EXT_hand_tracking.cpp | 216 +- .../test_XR_EXT_local_floor.cpp | 11 +- .../test_XR_EXT_palm_pose.cpp | 31 +- .../test_XR_EXT_performance_settings.cpp | 9 - .../test_XR_EXT_plane_detection.cpp | 860 + .../test_XR_EXT_thermal_query.cpp | 8 +- .../test_XR_KHR_D3D11_enable.cpp | 18 +- .../test_XR_KHR_D3D12_enable.cpp | 17 +- .../test_XR_KHR_OpenGL_ES_enable.cpp | 20 +- .../test_XR_KHR_OpenGL_enable.cpp | 19 +- .../test_XR_KHR_composition_layer_cube.cpp | 22 +- ...test_XR_KHR_composition_layer_cylinder.cpp | 20 +- .../test_XR_KHR_composition_layer_depth.cpp | 16 +- ...test_XR_KHR_composition_layer_equirect.cpp | 22 +- .../test_XR_KHR_convert_timespec_time.cpp | 2 +- .../conformance_test/test_XR_KHR_headless.cpp | 2 +- .../test_XR_KHR_visibility_mask.cpp | 33 +- .../test_XR_KHR_vulkan_enable.cpp | 21 +- .../test_XR_KHR_vulkan_enable2.cpp | 21 +- ...win32_convert_performance_counter_time.cpp | 9 +- .../test_XR_META_headset_id.cpp | 30 +- .../test_XR_META_performance_metrics.cpp | 14 +- .../conformance_test/test_XR_MND_headless.cpp | 9 +- .../test_XrCompositionLayerProjection.cpp | 23 +- .../test_XrCompositionLayerQuad.cpp | 20 +- .../conformance_test/test_actions.cpp | 37 +- .../conformance_test/test_multithreading.cpp | 24 +- .../test_xrCreateInstance.cpp | 17 +- .../test_xrCreateReferenceSpace.cpp | 8 +- .../conformance_test/test_xrCreateSession.cpp | 16 +- .../test_xrCreateSwapchain.cpp | 15 +- .../test_xrEnumerateApiLayerProperties.cpp | 13 +- .../test_xrEnumerateEnvironmentBlendModes.cpp | 14 +- ...xrEnumerateInstanceExtensionProperties.cpp | 13 +- .../test_xrEnumerateReferenceSpaces.cpp | 10 +- .../test_xrEnumerateSwapchainFormats.cpp | 3 + .../test_xrGetInstanceProcAddr.cpp | 17 +- .../test_xrGetInstanceProperties.cpp | 13 +- .../test_xrGetReferenceSpaceBoundsRect.cpp | 10 +- .../conformance_test/test_xrGetSystem.cpp | 12 +- .../test_xrGetSystemProperties.cpp | 11 +- .../conformance_test/test_xrLinear.cpp | 5 +- .../conformance_test/test_xrLocateSpace.cpp | 14 +- .../conformance_test/test_xrPathToString.cpp | 9 +- .../conformance_test/test_xrPollEvent.cpp | 14 +- .../test_xrRequestExitSession.cpp | 3 +- .../test_xrResultToString.cpp | 19 +- .../conformance_test/test_xrStringToPath.cpp | 13 +- .../test_xrStructureTypeToString.cpp | 18 +- src/conformance/framework/CMakeLists.txt | 91 + src/conformance/framework/RGBAImage.cpp | 20 +- src/conformance/framework/RGBAImage.h | 4 +- src/conformance/framework/action_utils.h | 14 +- .../framework/catch_reporter_cts.h | 2 +- .../framework/composition_utils.cpp | 28 +- src/conformance/framework/composition_utils.h | 87 +- .../framework/conformance_framework.cpp | 126 +- .../framework/conformance_framework.h | 48 +- .../framework/conformance_utils.cpp | 59 +- src/conformance/framework/conformance_utils.h | 86 +- src/conformance/framework/graphics_plugin.h | 2 +- .../framework/graphics_plugin_d3d11.cpp | 35 +- .../framework/graphics_plugin_d3d12.cpp | 58 +- .../framework/graphics_plugin_factory.cpp | 2 +- .../framework/graphics_plugin_opengl.cpp | 36 +- .../framework/graphics_plugin_opengles.cpp | 46 +- .../framework/graphics_plugin_vulkan.cpp | 51 +- .../framework/input_testinputdevice.cpp | 28 +- .../framework/input_testinputdevice.h | 8 +- .../framework/mesh_projection_layer.cpp | 10 + .../framework/mesh_projection_layer.h | 12 +- .../framework/platform_plugin_android.cpp | 22 +- .../framework/platform_plugin_posix.cpp | 4 +- .../framework/platform_plugin_win32.cpp | 4 +- .../framework/swapchain_image_data.cpp | 9 +- .../framework/swapchain_image_data.h | 13 +- src/conformance/framework/two_call_struct.h | 2 +- .../framework/two_call_struct_tests.h | 2 +- .../framework/types_and_constants.cpp | 88 - .../vulkan_shaders/frag.glsl | 0 .../vulkan_shaders/frag.spv | 0 .../vulkan_shaders/frag.spv.license | 0 .../vulkan_shaders/vert.glsl | 0 .../vulkan_shaders/vert.spv | 0 .../vulkan_shaders/vert.spv.license | 0 .../framework/xml_test_environment.cpp | 16 +- .../gradle/wrapper/gradle-wrapper.jar | Bin 58910 -> 60756 bytes .../gradle/wrapper/gradle-wrapper.properties | 2 +- src/conformance/gradlew | 265 +- src/conformance/gradlew.bat | 35 +- src/conformance/gradlew.bat.license | 2 + src/conformance/gradlew.license | 2 + .../platform_specific/AndroidManifest.xml | 119 +- .../platform_specific/android_main.cpp | 2 +- src/conformance/utilities/CMakeLists.txt | 34 + .../{framework => utilities}/Geometry.cpp | 2 + .../{framework => utilities}/Geometry.h | 0 src/conformance/utilities/align_to.h | 20 + src/conformance/utilities/array_size.h | 18 + .../bitmask_generator.cpp | 7 +- .../bitmask_generator.h | 2 +- .../bitmask_to_string.cpp | 9 +- .../bitmask_to_string.h | 5 +- .../{framework => utilities}/d3d_common.cpp | 19 +- .../{framework => utilities}/d3d_common.h | 3 +- .../{framework => utilities}/event_reader.cpp | 4 +- .../{framework => utilities}/event_reader.h | 3 +- .../{framework => utilities}/generator.h | 2 +- src/conformance/utilities/stringification.cpp | 35 + src/conformance/utilities/stringification.h | 37 + .../swapchain_parameters.h | 2 +- .../utilities/system_properties_helper.h | 129 + .../{framework => utilities}/throw_helpers.h | 11 +- .../utilities/types_and_constants.cpp | 49 + .../types_and_constants.h | 36 +- .../{framework => utilities}/utils.cpp | 9 +- .../{framework => utilities}/utils.h | 21 +- .../{framework => utilities}/vulkan_utils.h | 10 +- .../utilities/xrduration_literals.h | 46 + src/external/d3dx12/LICENSE | 21 + src/external/d3dx12/README.md | 59 + src/external/d3dx12/d3dx12.h | 4049 +++ src/external/mikktspace/README.md | 4 + src/external/mikktspace/mikktspace.c | 1899 ++ src/external/mikktspace/mikktspace.h | 145 + src/external/tinygltf/LICENSE | 21 + src/external/tinygltf/README.md | 267 + src/external/tinygltf/json.hpp | 26753 ++++++++++++++++ src/external/tinygltf/tiny_gltf.cc | 4 + src/external/tinygltf/tiny_gltf.h | 8249 +++++ src/loader/AndroidManifest.xml | 7 + src/loader/AndroidManifest.xml.in | 14 +- src/loader/CMakeLists.txt | 19 +- src/loader/android_utilities.cpp | 8 +- src/loader/api_layer_interface.cpp | 14 +- src/loader/build.gradle | 204 - src/loader/loader_instance.cpp | 4 +- src/loader/loader_logger_recorders.cpp | 9 +- src/loader/manifest_file.cpp | 9 +- src/loader/manifest_file.hpp | 2 +- src/loader/openxr-loader.expsym | 59 + src/loader/runtime_interface.cpp | 4 +- src/scripts/utility_source_generator.py | 1 + src/scripts/validation_layer_generator.py | 15 +- 223 files changed, 46131 insertions(+), 1623 deletions(-) create mode 100644 LICENSES/Zlib.txt create mode 100644 changes/conformance/mr.2510.gl.md create mode 100644 changes/conformance/mr.2669.gl.md create mode 100644 changes/conformance/mr.2689.gl.md create mode 100644 changes/conformance/mr.2704.gl.md create mode 100644 changes/conformance/mr.2717.gl.md create mode 100644 changes/conformance/mr.2725.gl.md create mode 100644 changes/conformance/mr.2729.gl.md create mode 100644 changes/conformance/mr.2730.gl.md create mode 100644 changes/conformance/mr.2735.gl.md create mode 100644 changes/conformance/mr.2736.gl.md create mode 100644 changes/conformance/mr.2737.gl.md create mode 100644 changes/conformance/mr.2746.gl.md create mode 100644 changes/conformance/mr.2756.gl.md create mode 100644 changes/conformance/mr.2766.gl.md create mode 100755 specification/scripts/findBareNormatives.sh create mode 100755 specification/scripts/reserve_extensions.py rename specification/scripts/{openxr-macros.rb => spec-macros.rb} (89%) rename specification/scripts/{openxr-macros => spec-macros}/extension.rb (52%) create mode 100644 src/conformance/conformance_layer/XrApiLayer_conformance_layer.expsym create mode 100644 src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png create mode 100644 src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png.license create mode 100644 src/conformance/conformance_test/test_XR_EXT_plane_detection.cpp create mode 100644 src/conformance/framework/CMakeLists.txt delete mode 100644 src/conformance/framework/types_and_constants.cpp rename src/conformance/{conformance_test => framework}/vulkan_shaders/frag.glsl (100%) rename src/conformance/{conformance_test => framework}/vulkan_shaders/frag.spv (100%) rename src/conformance/{conformance_test => framework}/vulkan_shaders/frag.spv.license (100%) rename src/conformance/{conformance_test => framework}/vulkan_shaders/vert.glsl (100%) rename src/conformance/{conformance_test => framework}/vulkan_shaders/vert.spv (100%) rename src/conformance/{conformance_test => framework}/vulkan_shaders/vert.spv.license (100%) create mode 100644 src/conformance/gradlew.bat.license create mode 100644 src/conformance/gradlew.license create mode 100644 src/conformance/utilities/CMakeLists.txt rename src/conformance/{framework => utilities}/Geometry.cpp (98%) rename src/conformance/{framework => utilities}/Geometry.h (100%) create mode 100644 src/conformance/utilities/align_to.h create mode 100644 src/conformance/utilities/array_size.h rename src/conformance/{framework => utilities}/bitmask_generator.cpp (94%) rename src/conformance/{framework => utilities}/bitmask_generator.h (98%) rename src/conformance/{framework => utilities}/bitmask_to_string.cpp (96%) rename src/conformance/{framework => utilities}/bitmask_to_string.h (98%) rename src/conformance/{framework => utilities}/d3d_common.cpp (98%) rename src/conformance/{framework => utilities}/d3d_common.h (98%) rename src/conformance/{framework => utilities}/event_reader.cpp (98%) rename src/conformance/{framework => utilities}/event_reader.h (98%) rename src/conformance/{framework => utilities}/generator.h (97%) create mode 100644 src/conformance/utilities/stringification.cpp create mode 100644 src/conformance/utilities/stringification.h rename src/conformance/{framework => utilities}/swapchain_parameters.h (99%) create mode 100644 src/conformance/utilities/system_properties_helper.h rename src/conformance/{framework => utilities}/throw_helpers.h (93%) create mode 100644 src/conformance/utilities/types_and_constants.cpp rename src/conformance/{framework => utilities}/types_and_constants.h (93%) rename src/conformance/{framework => utilities}/utils.cpp (98%) rename src/conformance/{framework => utilities}/utils.h (95%) rename src/conformance/{framework => utilities}/vulkan_utils.h (98%) create mode 100644 src/conformance/utilities/xrduration_literals.h create mode 100644 src/external/d3dx12/LICENSE create mode 100644 src/external/d3dx12/README.md create mode 100644 src/external/d3dx12/d3dx12.h create mode 100644 src/external/mikktspace/README.md create mode 100644 src/external/mikktspace/mikktspace.c create mode 100644 src/external/mikktspace/mikktspace.h create mode 100644 src/external/tinygltf/LICENSE create mode 100644 src/external/tinygltf/README.md create mode 100644 src/external/tinygltf/json.hpp create mode 100644 src/external/tinygltf/tiny_gltf.cc create mode 100644 src/external/tinygltf/tiny_gltf.h delete mode 100644 src/loader/build.gradle create mode 100644 src/loader/openxr-loader.expsym diff --git a/.azure-pipelines/shared/build_jobs.yml b/.azure-pipelines/shared/build_jobs.yml index ea5e977f..213a40a8 100644 --- a/.azure-pipelines/shared/build_jobs.yml +++ b/.azure-pipelines/shared/build_jobs.yml @@ -26,7 +26,8 @@ jobs: presentationBackend: wayland pool: vmImage: "ubuntu-latest" - container: khronosgroup/docker-images:openxr-sdk.20230209 + container: khronosgroup/docker-images:openxr-sdk.20230323 + # container: khronosgroup/docker-images@sha256:20edadbaa6cdec4fed7417c24b18dfb4b93eec940fdf1a27b5f97272dec47032 steps: # First build as debug - template: build_linux.yml @@ -130,7 +131,7 @@ jobs: # Use the specified version of Python from the tool cache - task: UsePythonVersion@0 inputs: - versionSpec: "3.9" + versionSpec: "3.10" - task: PythonScript@0 displayName: Move artifact contents inputs: diff --git a/.azure-pipelines/shared/check_clang_format.yml b/.azure-pipelines/shared/check_clang_format.yml index c0574751..d6d1cadb 100644 --- a/.azure-pipelines/shared/check_clang_format.yml +++ b/.azure-pipelines/shared/check_clang_format.yml @@ -6,7 +6,8 @@ jobs: displayName: "clang-format" pool: vmImage: "ubuntu-latest" - container: khronosgroup/docker-images:openxr-sdk.20230209 + container: khronosgroup/docker-images:openxr-sdk.20230323 + # container: khronosgroup/docker-images@sha256:20edadbaa6cdec4fed7417c24b18dfb4b93eec940fdf1a27b5f97272dec47032 steps: - script: ./runClangFormat.sh displayName: Run clang-format diff --git a/.azure-pipelines/shared/check_file_format.yml b/.azure-pipelines/shared/check_file_format.yml index f634aac2..2c14cbc4 100644 --- a/.azure-pipelines/shared/check_file_format.yml +++ b/.azure-pipelines/shared/check_file_format.yml @@ -5,7 +5,8 @@ jobs: displayName: 'Check file formatting' pool: vmImage: 'ubuntu-latest' - container: khronosgroup/docker-images:openxr-sdk.20230209 + container: khronosgroup/docker-images:openxr-sdk.20230323 + # container: khronosgroup/docker-images@sha256:20edadbaa6cdec4fed7417c24b18dfb4b93eec940fdf1a27b5f97272dec47032 steps: - script: ./file_format.sh displayName: File formatting checks (file_format.sh) diff --git a/.azure-pipelines/shared/codespell.yml b/.azure-pipelines/shared/codespell.yml index df0a3c99..d582e521 100644 --- a/.azure-pipelines/shared/codespell.yml +++ b/.azure-pipelines/shared/codespell.yml @@ -6,7 +6,8 @@ jobs: displayName: "codespell" pool: vmImage: "ubuntu-latest" - container: khronosgroup/docker-images:openxr-sdk.20230209 + container: khronosgroup/docker-images:openxr-sdk.20230323 + # container: khronosgroup/docker-images@sha256:20edadbaa6cdec4fed7417c24b18dfb4b93eec940fdf1a27b5f97272dec47032 steps: - script: ./checkCodespell displayName: Run Codespell script diff --git a/.gitattributes b/.gitattributes index 892b854b..dcddd89a 100644 --- a/.gitattributes +++ b/.gitattributes @@ -23,7 +23,10 @@ *.sh eol=lf *.png binary -*.pdf binary + +# git-lfs big files +*.pdf filter=lfs diff=lfs merge=lfs -text +*.glb filter=lfs diff=lfs merge=lfs -text # Shell/python scripts that don't end in .sh specification/makeAllExts eol=lf diff --git a/.github/workflows/android-cts-build.yml b/.github/workflows/android-cts-build.yml index cd54de16..8d8c1796 100644 --- a/.github/workflows/android-cts-build.yml +++ b/.github/workflows/android-cts-build.yml @@ -22,7 +22,7 @@ jobs: - uses: actions/checkout@v3 - name: Get modern CMake and Ninja - uses: lukka/get-cmake@v3.26.3 + uses: lukka/get-cmake@v3.26.4 - name: set up JDK 11 uses: actions/setup-java@v3 diff --git a/.github/workflows/android-cts-pr.yml b/.github/workflows/android-cts-pr.yml index e575c980..42f53207 100644 --- a/.github/workflows/android-cts-pr.yml +++ b/.github/workflows/android-cts-pr.yml @@ -17,7 +17,7 @@ jobs: - uses: actions/checkout@v3 - name: Get modern CMake and Ninja - uses: lukka/get-cmake@v3.26.3 + uses: lukka/get-cmake@v3.26.4 - name: set up JDK 11 uses: actions/setup-java@v3 diff --git a/.github/workflows/msvc-build-preset.yml b/.github/workflows/msvc-build-preset.yml index 9d8cb9fe..c427dca1 100644 --- a/.github/workflows/msvc-build-preset.yml +++ b/.github/workflows/msvc-build-preset.yml @@ -32,7 +32,7 @@ jobs: - uses: actions/checkout@v3 - name: Get modern CMake and Ninja - uses: lukka/get-cmake@v3.26.3 + uses: lukka/get-cmake@v3.26.4 - name: Add msbuild to PATH uses: microsoft/setup-msbuild@v1.3 diff --git a/.gitignore b/.gitignore index 03c1a0d7..13a23f16 100644 --- a/.gitignore +++ b/.gitignore @@ -18,6 +18,7 @@ pregen/ generated-includes cmake_install.cmake cmake_uninstall.cmake +cmake-build-* # Marker file for a snapshot build SNAPSHOT diff --git a/.reuse/dep5 b/.reuse/dep5 index c4426f94..9091bf0b 100644 --- a/.reuse/dep5 +++ b/.reuse/dep5 @@ -38,6 +38,27 @@ Copyright: 2007-2010 Baptiste Lepilleur and The JsonCpp Authors License: MIT OR LicenseRef-jsoncpp-public-domain Comment: Unmodified, vendored copy of jsoncpp 1.9.5 +Files: src/external/tinygltf/* +Copyright: 2017 Syoyo Fujita, Aurélien Chatelain and many contributors +License: MIT +Comment: Unmodified, vendored copy of a subset of the tiny-gltf repo v2.8.9 + +Files: src/external/tinygltf/json.hpp +Copyright: 2013-2017 Niels Lohmann +License: MIT +Comment: Unmodified, included in tiny-gltf repo. + +Files: src/external/d3dx12/* +Copyright: Copyright (c) Microsoft Corporation. +License: MIT +Comment: Unmodified, vendored copy of DirectX-Headers commit da7aedb + of https://github.com/microsoft/DirectX-Headers filtered to just d3dx12 headers + +Files: src/external/mikktspace/* +Copyright: 2011 by Morten S. Mikkelsen +License: Zlib +Comment: Unmodified, vendored copy of MikkTSpace commit 3e895b4 + Files: src/external/jnipp/* Copyright: 2016-2020, Mitchell Dowd 2020, Collabora, Ltd. @@ -67,7 +88,7 @@ Comment: In-line license comments requested, https://gitlab.khronos.org/openxr/o Files: specification/sources/chapters/extensions/ext/ext_performance_settings.adoc specification/sources/chapters/extensions/ext/ext_thermal_query.adoc -Copyright: 2017-2020, The Khronos Group Inc. +Copyright: 2017-2023, The Khronos Group Inc. License: CC-BY-4.0 Comment: In-line license comments requested, https://gitlab.khronos.org/openxr/openxr/-/issues/1419 @@ -76,7 +97,7 @@ Files: src/conformance/platform_specific/android_resources/mipmap-hdpi/* src/conformance/platform_specific/android_resources/mipmap-xhdpi/* src/conformance/platform_specific/android_resources/mipmap-xxhdpi/* src/conformance/platform_specific/android_resources/mipmap-xxxhdpi/* -Copyright: 2020, The Khronos Group Inc. +Copyright: 2020-2023, The Khronos Group Inc. 2020, Google License: Apache-2.0 Comment: Generated .png versions of an icon, created in Android Studio diff --git a/BUILDING.md b/BUILDING.md index c3a61e6e..aa2a87c2 100644 --- a/BUILDING.md +++ b/BUILDING.md @@ -142,6 +142,21 @@ cmake -DCMAKE_BUILD_TYPE=RelWithDebInfo ../.. make ``` +### macOS + +Building the OpenXR components in this tree on macOS is supported using Xcode +14.0 and newer. You may need to install Xcode Command Line Tools and cmake. + +First, generate the Xcode project file using CMake: + +```cmd +mkdir -p build/macos +cd build/macos +cmake -G "Xcode" ../.. +``` + +Finally, open the build/macos/OPENXR.xcodeproj in Xcode to build the samples. + ### Android ```sh diff --git a/CMakeLists.txt b/CMakeLists.txt index 44255b71..43ab332b 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -33,6 +33,8 @@ set(LOADER_FOLDER "Loader") set(HELPER_FOLDER "Helpers") set(CODEGEN_FOLDER "Generated") set(TESTS_FOLDER "Tests") +set(CONFORMANCE_TESTS_FOLDER "Conformance Test Suite") +set(LOADER_TESTS_FOLDER "Loader Tests") set(API_LAYERS_FOLDER "Layers") set(SAMPLES_FOLDER "Samples") diff --git a/LICENSES/Zlib.txt b/LICENSES/Zlib.txt new file mode 100644 index 00000000..e0e3605b --- /dev/null +++ b/LICENSES/Zlib.txt @@ -0,0 +1,11 @@ +zlib License + +This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. + +Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: + + 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. + + 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. + + 3. This notice may not be removed or altered from any source distribution. diff --git a/changes/conformance/mr.2510.gl.md b/changes/conformance/mr.2510.gl.md new file mode 100644 index 00000000..bd57edb1 --- /dev/null +++ b/changes/conformance/mr.2510.gl.md @@ -0,0 +1,5 @@ +--- +- mr.2751.gl +- mr.2676.gl +--- +New test: Added `XR_EXT_plane_detection` extension. diff --git a/changes/conformance/mr.2669.gl.md b/changes/conformance/mr.2669.gl.md new file mode 100644 index 00000000..abcfc994 --- /dev/null +++ b/changes/conformance/mr.2669.gl.md @@ -0,0 +1 @@ +Improvement: Refactor utilities that do not depend on Catch2 into a separate internal library. diff --git a/changes/conformance/mr.2689.gl.md b/changes/conformance/mr.2689.gl.md new file mode 100644 index 00000000..07b3e8bd --- /dev/null +++ b/changes/conformance/mr.2689.gl.md @@ -0,0 +1 @@ +Improvement: Make composition test help/example world locked but based on initial view, for more natural reading. diff --git a/changes/conformance/mr.2704.gl.md b/changes/conformance/mr.2704.gl.md new file mode 100644 index 00000000..71b2bf72 --- /dev/null +++ b/changes/conformance/mr.2704.gl.md @@ -0,0 +1,6 @@ +--- +- mr.2784.gl +- mr.2785.gl +- mr.2808.gl +--- +Improvement: Cleanup and code quality work. diff --git a/changes/conformance/mr.2717.gl.md b/changes/conformance/mr.2717.gl.md new file mode 100644 index 00000000..cb4f77b3 --- /dev/null +++ b/changes/conformance/mr.2717.gl.md @@ -0,0 +1 @@ +Improvement: Fix clang warnings `-Wundef` and `-Wmissing-braces`. diff --git a/changes/conformance/mr.2725.gl.md b/changes/conformance/mr.2725.gl.md new file mode 100644 index 00000000..e33f542f --- /dev/null +++ b/changes/conformance/mr.2725.gl.md @@ -0,0 +1 @@ +gradle: Add license for gradlew and gradlew.bat diff --git a/changes/conformance/mr.2729.gl.md b/changes/conformance/mr.2729.gl.md new file mode 100644 index 00000000..95e4d03c --- /dev/null +++ b/changes/conformance/mr.2729.gl.md @@ -0,0 +1 @@ +Improvement: Add joint query to non-interactive test for `XR_EXT_hand_tracking`. diff --git a/changes/conformance/mr.2730.gl.md b/changes/conformance/mr.2730.gl.md new file mode 100644 index 00000000..a7720525 --- /dev/null +++ b/changes/conformance/mr.2730.gl.md @@ -0,0 +1 @@ +Improvement: Add test for calling `xrAcquireSwapchainImage` multiple times without calling `xrEndFrame`. diff --git a/changes/conformance/mr.2735.gl.md b/changes/conformance/mr.2735.gl.md new file mode 100644 index 00000000..d7f49116 --- /dev/null +++ b/changes/conformance/mr.2735.gl.md @@ -0,0 +1,5 @@ +--- +- issue.53.gh.OpenXR-CTS +- issue.1947.gl +--- +Improvement: Optionally poll `xrGetSystem` before running test cases. diff --git a/changes/conformance/mr.2736.gl.md b/changes/conformance/mr.2736.gl.md new file mode 100644 index 00000000..29a8c04a --- /dev/null +++ b/changes/conformance/mr.2736.gl.md @@ -0,0 +1,4 @@ +--- +- issue.1950.gl +--- +Improvement: Select the first enumerated environment blend mode by default, rather than always using "opaque" diff --git a/changes/conformance/mr.2737.gl.md b/changes/conformance/mr.2737.gl.md new file mode 100644 index 00000000..869cf7aa --- /dev/null +++ b/changes/conformance/mr.2737.gl.md @@ -0,0 +1 @@ +Improvement: Migrate more tests to use the `SKIP` macro when appropriate. diff --git a/changes/conformance/mr.2746.gl.md b/changes/conformance/mr.2746.gl.md new file mode 100644 index 00000000..34bc4818 --- /dev/null +++ b/changes/conformance/mr.2746.gl.md @@ -0,0 +1 @@ +Fix: Use actual acquired image index in swapchain rendering test. diff --git a/changes/conformance/mr.2756.gl.md b/changes/conformance/mr.2756.gl.md new file mode 100644 index 00000000..fef78530 --- /dev/null +++ b/changes/conformance/mr.2756.gl.md @@ -0,0 +1 @@ +Fix: Do not use Catch2 assertion macros in graphics plugin methods that may be called before the first test case execution begins. diff --git a/changes/conformance/mr.2766.gl.md b/changes/conformance/mr.2766.gl.md new file mode 100644 index 00000000..7a0fd1ec --- /dev/null +++ b/changes/conformance/mr.2766.gl.md @@ -0,0 +1 @@ +Fix spelling. diff --git a/include/openxr/openxr_platform_defines.h b/include/openxr/openxr_platform_defines.h index 9573c101..820b7b3e 100644 --- a/include/openxr/openxr_platform_defines.h +++ b/include/openxr/openxr_platform_defines.h @@ -103,6 +103,10 @@ typedef unsigned __int64 uint64_t; #endif #endif +#if !defined(XR_CPP_NULLPTR_SUPPORTED) +#define XR_CPP_NULLPTR_SUPPORTED 0 +#endif // !defined(XR_CPP_NULLPTR_SUPPORTED) + #ifdef __cplusplus } #endif diff --git a/maintainer-scripts/common.sh b/maintainer-scripts/common.sh index 9992c5b3..37043223 100644 --- a/maintainer-scripts/common.sh +++ b/maintainer-scripts/common.sh @@ -193,13 +193,26 @@ getSDKSourceFilenames() { specification/Makefile \ specification/README.md \ specification/requirements.txt \ - src/ \ + src/.clang-format \ + src/.gitignore \ + src/CMakeLists.txt \ + src/api_layers \ + src/cmake \ + src/common \ + src/common_config.h.in \ + src/external/CMakeLists.txt \ + src/external/android-jni-wrappers \ + src/external/jnipp \ + src/external/jsoncpp \ + src/loader \ + src/scripts \ + src/tests \ + src/version.cmake \ + src/version.gradle \ | grep -v "${COMMON_EXCLUDE_PATTERN}" \ | grep -v "conformance" \ | grep -v "template_gen_dispatch" \ - | grep -v "catch2" \ | grep -v "function_info" \ - | grep -v "stb" \ | grep -v "htmldiff" \ | grep -v "katex" } @@ -224,17 +237,17 @@ getSDKFilenames() { .github/workflows/windows-matrix.yml \ specification/registry/*.xml \ include/ \ + src/.clang-format \ src/CMakeLists.txt \ - src/version.cmake \ - src/common_config.h.in \ - src/common \ src/cmake \ - src/loader \ + src/common \ + src/common_config.h.in \ src/external/CMakeLists.txt \ - src/external/jsoncpp \ - src/external/jnipp \ src/external/android-jni-wrappers \ - src/.clang-format \ + src/external/jnipp \ + src/external/jsoncpp \ + src/loader \ + src/version.cmake \ | grep -v "${COMMON_EXCLUDE_PATTERN}" \ | grep -v "gfxwrapper" \ | grep -v "include/.gitignore" \ @@ -287,18 +300,18 @@ getConformanceFilenames() { specification/Makefile \ specification/README.md \ specification/requirements.txt \ - src/version.gradle \ + src/.clang-format \ + src/.gitignore \ + src/CMakeLists.txt \ src/cmake \ src/common \ + src/common_config.h.in \ src/conformance \ src/external \ src/loader \ src/scripts \ - src/.clang-format \ - src/.gitignore \ - src/CMakeLists.txt \ - src/common_config.h.in \ src/version.cmake \ + src/version.gradle \ | grep -v "${COMMON_EXCLUDE_PATTERN}" \ | grep -v "htmldiff" \ | grep -v "katex" diff --git a/runClangFormat.sh b/runClangFormat.sh index 49eb9e65..2fb98a75 100755 --- a/runClangFormat.sh +++ b/runClangFormat.sh @@ -1,4 +1,4 @@ -#!/bin/bash +#!/usr/bin/env bash # Copyright (c) 2017-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 @@ -18,11 +18,11 @@ set -e ( PREFERRED_CLANG_FORMAT=clang-format-10 - ACCEPTABLE_CLANG_FORMATS="${PREFERRED_CLANG_FORMAT} clang-format-11 clang-format-12 clang-format-13 clang-format-14 clang-format-15 clang-format" - cd "$(dirname $0)" + ACCEPTABLE_CLANG_FORMATS="${PREFERRED_CLANG_FORMAT} clang-format-11 clang-format-12 clang-format-13 clang-format-14 clang-format-15 clang-format-16 clang-format" + cd "$(dirname "$0")" if [ ! "${CLANGFORMAT}" ]; then for tool in ${ACCEPTABLE_CLANG_FORMATS}; do - if which $tool > /dev/null 2> /dev/null; then + if which "$tool" > /dev/null 2> /dev/null; then CLANGFORMAT=$tool break fi @@ -42,6 +42,6 @@ set -e -and -not \( -wholename ./src/external/\* \) \ -and -not \( -wholename ./src/scripts/\* \) \ -and \( -name \*.hpp -or -name \*.h -or -name \*.cpp -or -name \*.c \) \ - -exec ${CLANGFORMAT} -i -style=file {} + + -exec "${CLANGFORMAT}" -i -style=file {} + ) diff --git a/specification/Makefile b/specification/Makefile index 16320e88..5b69343d 100644 --- a/specification/Makefile +++ b/specification/Makefile @@ -32,7 +32,7 @@ ifneq (,$(strip $(VERY_STRICT))) ASCIIDOC := $(ASCIIDOC) --failure-level WARN endif -SPECREVISION = 1.0.27 +SPECREVISION = 1.0.28 REVISION_COMPONENTS = $(subst ., ,$(SPECREVISION)) MAJORMINORVER = $(word 1,$(REVISION_COMPONENTS)).$(word 2,$(REVISION_COMPONENTS)) @@ -53,7 +53,8 @@ GENOPTS = SCRIPTS := ./scripts SPECTOOLS := $(SCRIPTS)/spec_tools -XRAPI := $(GENDIR)/apimap.py +PYAPIMAP := $(GENDIR)/apimap.py +RBAPIMAP := $(GENDIR)/apimap.rb METADIR := $(GENDIR)/meta VK_REF_PAGE_ROOT := https://www.khronos.org/registry/vulkan/specs/1.3-extensions/man/html @@ -181,7 +182,7 @@ GENSTAMPS := \ $(METADIR)/extinc \ # The actual generated index -GENDEPENDS := $(XRAPI) $(GENSTAMPS) $(GENDIR)/index.adoc +GENDEPENDS := $(PYAPIMAP) $(RBAPIMAP) $(GENSTAMPS) $(GENDIR)/index.adoc # The rule for every genxr-generated file $(GENDEPENDS) $(GENHEADERS): $(BASIC_GENERATED_DEPENDS) @@ -194,9 +195,10 @@ $(GENDEPENDS) $(GENHEADERS): $(BASIC_GENERATED_DEPENDS) $(GENSTAMPS): STAMP_NOTE = (and additional files in $(@D)) # Extra deps -$(GENDIR)/api/apiinc: $(XRAPI) $(SCRIPTS)/docgenerator.py +$(GENDIR)/api/apiinc: $(PYAPIMAP) $(SCRIPTS)/docgenerator.py $(GENDIR)/validity/validinc: $(SCRIPTS)/validitygenerator.py $(SPECTOOLS)/validity.py $(SPECTOOLS)/attributes.py $(SPECTOOLS)/data_structures.py -$(XRAPI): $(SCRIPTS)/pygenerator.py $(SCRIPTS)/docgenerator.py +$(PYAPIMAP): $(SCRIPTS)/pygenerator.py $(SCRIPTS)/docgenerator.py +$(RBAPIMAP): $(SCRIPTS)/rubygenerator.py $(SCRIPTS)/docgenerator.py $(GENHEADERS): $(SCRIPTS)/cgenerator.py $(REFLECTHEADERS): $(SCRIPTS)/creflectiongenerator.py $(SCRIPTS)/jinja_helpers.py $(wildcard $(SCRIPTS)/template_*) @@ -227,7 +229,7 @@ extinc: $(METADIR)/extinc # The actual generated include files GENINCLUDE = $(GENAPI) $(GENVALIDITY) $(GENSYNC) $(GENMETA) .PHONY: generated -generated: $(GENDEPENDS) $(XRAPI) +generated: $(GENDEPENDS) ################################################ # OpenXR Style Guide @@ -246,7 +248,7 @@ ASCIIDOCTOR_TARGETS += $(STYLEGUIDE) # Target-specific variables and deps customizing the AsciiDoctor rule $(STYLEGUIDE): SPECSRC=$(STYLESRC) $(STYLEGUIDE): LOGFILE=$(OUTDIR)/adoc_styleguide_stderr.txt -$(STYLEGUIDE): $(STYLESRC) $(STYLEFILES) $(GENDIR)/validity/validinc $(GENDIR)/api/apiinc +$(STYLEGUIDE): $(STYLESRC) $(STYLEFILES) $(GENDIR)/validity/validinc $(GENDIR)/api/apiinc $(RBAPIMAP) ################################################ @@ -264,7 +266,7 @@ ASCIIDOCTOR_TARGETS += $(LOADERGUIDE) # Target-specific variables and deps customizing the AsciiDoctor rule $(LOADERGUIDE): SPECSRC=$(LOADERSRC) $(LOADERGUIDE): LOGFILE=$(OUTDIR)/adoc_loader_stderr.txt -$(LOADERGUIDE): $(LOADERSRC) $(LOADERFILES) +$(LOADERGUIDE): $(LOADERSRC) $(LOADERFILES) $(RBAPIMAP) ################################################ @@ -351,7 +353,9 @@ ATTRIBOPTS = -a revnumber="$(SPECREVISION)" \ -a appendices=$(APPENDICES) \ $(EXTATTRIBS) -ADOCOPTS = --doctype book -a data-uri -r $(CURDIR)/scripts/openxr-macros.rb $(ATTRIBOPTS) +# Look in $(GENERATED) for explicitly required non-extension Ruby, such +# as apimap.rb +ADOCOPTS = --doctype book -a data-uri -I$(GENDIR) -r $(CURDIR)/scripts/spec-macros.rb $(ATTRIBOPTS) ifneq (,$(strip $(RELEASE))) # No dates or internal commit hashes in release builds for reproducibility @@ -384,7 +388,7 @@ $(ASCIIDOCTOR_TARGETS): OUTSPEC_DOS=$$(cygpath -w $@) ;\ SPECSRC_DOS=$$(cygpath -w $(SPECSRC)) ;\ ATTRIBOPTS_DOS='$(ATTRIBOPTS)' ;\ - ADOCOPTS_DOS="--doctype book -a data-uri -r $$(cygpath -w $(CURDIR)/scripts/openxr-macros.rb) $$ATTRIBOPTS_DOS" ;\ + ADOCOPTS_DOS="--doctype book -a data-uri -r $$(cygpath -w $(CURDIR)/scripts/spec-macros.rb) $$ATTRIBOPTS_DOS" ;\ BATCH_FILE=$$(cygpath -w $$(mktemp)).bat ;\ echo $(ASCIIDOC) $$ADOCOPTS_DOS $(BACKEND_ARGS) --out-file $$OUTSPEC_DOS $$SPECSRC_DOS > $$BATCH_FILE ;\ CMD /C $$BATCH_FILE ;\ @@ -459,7 +463,7 @@ EXTENSION_SOURCES := $(foreach ext,$(patsubst xr_%,%,$(EXTENSIONS_LOWER)),$(call GENREF = $(SCRIPTS)/genRef.py LOGFILE = $(REFPATH)/refpage.log refpages: $(REFPATH)/apispec.txt -$(REFPATH)/apispec.txt: $(SPECFILES) $(EXTENSION_SOURCES) $(GENREF) $(SCRIPTS)/reflib.py $(XRAPI) +$(REFPATH)/apispec.txt: $(SPECFILES) $(EXTENSION_SOURCES) $(GENREF) $(SCRIPTS)/reflib.py $(PYAPIMAP) $(ECHO) "[genRef.py] $(REGISTRY) and spec -> $@" $(ECHO) " (and additional files in $(@D))" $(QUIET)$(MKDIR) "$(REFPATH)" @@ -487,7 +491,7 @@ buildmanpages: $(MANHTML) $(MANHTMLDIR)/openxr.html # This target does not at present, since OpenXR does not alias refpage # content yet. -manaliases: $(XRAPI) +manaliases: $(PYAPIMAP) # This is the single-page ref page. # 'doctype-manpage' allows use of the "book" style but still enable the @@ -636,7 +640,7 @@ clean_dirt: # Clean intermediate generated files # Don't remove OUTDIR, since it contains the config stamp and final output targets clean_generated: - $(RMRF) $(XRAPI) $(GENDIR)/index.adoc + $(RMRF) $(PYAPIMAP) $(RBAPIMAP) $(GENDIR)/index.adoc $(RMRF) $(GENDIR)/api $(GENDIR)/validity $(GENDIR)/hostsynctable $(METADIR) $(REFPATH) # Clean generated targets as well as intermediates. diff --git a/specification/registry/xr.xml b/specification/registry/xr.xml index 388985de..0fccc519 100644 --- a/specification/registry/xr.xml +++ b/specification/registry/xr.xml @@ -30,6 +30,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + @@ -47,7 +48,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + @@ -130,7 +131,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. updates them automatically by processing a line at a time. --> // OpenXR current version number. -#define XR_CURRENT_API_VERSION XR_MAKE_VERSION(1, 0, 27) +#define XR_CURRENT_API_VERSION XR_MAKE_VERSION(1, 0, 28) + typedef XrFlags64 XrSemanticLabelsSupportFlagsFB; + typedef XrFlags64 XrHandTrackingAimFlagsFB; @@ -358,6 +368,13 @@ maintained in the default branch of the Khronos OpenXR GitHub project. typedef XrFlags64 XrGlobalDimmerFrameEndInfoFlagsML; + + typedef XrFlags64 XrPlaneDetectorFlagsEXT; + typedef XrFlags64 XrPlaneDetectionCapabilityFlagsEXT; + + + typedef XrFlags64 XrVirtualKeyboardInputStateFlagsMETA; + XR_DEFINE_HANDLE(XrInstance) @@ -401,6 +418,15 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XR_DEFINE_HANDLE(XrSpaceUserFB) + + XR_DEFINE_HANDLE(XrPassthroughColorLutMETA) + + + XR_DEFINE_HANDLE(XrPlaneDetectorEXT) + + + XR_DEFINE_HANDLE(XrVirtualKeyboardMETA) + @@ -491,6 +517,9 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + @@ -508,6 +537,9 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + @@ -552,6 +584,18 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + + + + float x @@ -592,6 +636,11 @@ maintained in the default branch of the Khronos OpenXR GitHub project. float width float height + + float width + float height + float depth + XrOffset2Df offset XrExtent2Df extent @@ -1601,7 +1650,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrTime time - + XrStructureType type @@ -1609,6 +1658,22 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrHandJointsMotionRangeEXT handJointsMotionRange + + + + XrStructureType type + const void* next + uint32_t requestedDataSourceCount + XrHandTrackingDataSourceEXT* requestedDataSources + + + + XrStructureType type + void* next + XrBool32 isActive + XrHandTrackingDataSourceEXT dataSource + + @@ -2402,6 +2467,12 @@ maintained in the default branch of the Khronos OpenXR GitHub project. uint32_t vertexCountOutput XrVector2f* vertices + + XrStructureType type + const void* next + XrSemanticLabelsSupportFlagsFB flags + const char* recognizedLabels + @@ -2795,6 +2866,47 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrUuidEXT id + + + uint32_t bufferSize + const uint8_t* buffer + + + + XrStructureType type + const void* next + XrPassthroughColorLutChannelsMETA channels + uint32_t resolution + XrPassthroughColorLutDataMETA data + + + + XrStructureType type + const void* next + XrPassthroughColorLutDataMETA data + + + + XrStructureType type + const void* next + XrPassthroughColorLutMETA colorLut + float weight + + + + XrStructureType type + const void* next + XrPassthroughColorLutMETA sourceColorLut + XrPassthroughColorLutMETA targetColorLut + float weight + + + + XrStructureType type + const void* next + uint32_t maxColorLutResolution + + XrStructureType type @@ -2910,6 +3022,174 @@ maintained in the default branch of the Khronos OpenXR GitHub project. float value + + + XrStructureType type + void* next + XrPlaneDetectionCapabilityFlagsEXT supportedFeatures + + + + XrStructureType type + const void* next + XrPlaneDetectorFlagsEXT flags + + + + XrStructureType type + const void* next + XrSpace baseSpace + XrTime time + uint32_t orientationCount + const XrPlaneDetectorOrientationEXT* orientations + uint32_t semanticTypeCount + const XrPlaneDetectorSemanticTypeEXT* semanticTypes + uint32_t maxPlanes + float minArea + XrPosef boundingBoxPose + XrExtent3DfEXT boundingBoxExtent + + + + XrStructureType type + const void* next + XrSpace baseSpace + XrTime time + + + + XrStructureType type + void* next + uint32_t planeLocationCapacityInput + uint32_t planeLocationCountOutput + XrPlaneDetectorLocationEXT* planeLocations + + + + XrStructureType type + void* next + uint64_t planeId + XrSpaceLocationFlags locationFlags + XrPosef pose + XrExtent2Df extents + XrPlaneDetectorOrientationEXT orientation + XrPlaneDetectorSemanticTypeEXT semanticType + uint32_t polygonBufferCount + + + + XrStructureType type + void* next + uint32_t vertexCapacityInput + uint32_t vertexCountOutput + XrVector2f* vertices + + + + + XrStructureType type + void* next + XrBool32 supportsVirtualKeyboard + + + + XrStructureType type + const void* next + + + + XrStructureType type + const void* next + XrVirtualKeyboardLocationTypeMETA locationType + XrSpace space + XrPosef poseInSpace + + + + XrStructureType type + const void* next + XrVirtualKeyboardLocationTypeMETA locationType + XrSpace space + XrPosef poseInSpace + float scale + + + + XrStructureType type + const void* next + XrBool32 visible + + + + XrStructureType type + void* next + int32_t animationIndex + float fraction + + + + XrStructureType type + void* next + uint32_t stateCapacityInput + uint32_t stateCountOutput + XrVirtualKeyboardAnimationStateMETA* states + + + + XrStructureType type + void* next + uint32_t textureWidth + uint32_t textureHeight + uint32_t bufferCapacityInput + uint32_t bufferCountOutput + uint8_t* buffer + + + + XrStructureType type + const void* next + XrVirtualKeyboardInputSourceMETA inputSource + XrSpace inputSpace + XrPosef inputPoseInSpace + XrVirtualKeyboardInputStateFlagsMETA inputState + + + + XrStructureType type + const void* next + const char* textContext + + + + XrStructureType type + const void* next + XrVirtualKeyboardMETA keyboard + char text[XR_MAX_VIRTUAL_KEYBOARD_COMMIT_TEXT_SIZE_META] + + + + XrStructureType type + const void* next + XrVirtualKeyboardMETA keyboard + + + + XrStructureType type + const void* next + XrVirtualKeyboardMETA keyboard + + + + XrStructureType type + const void* next + XrVirtualKeyboardMETA keyboard + + + + XrStructureType type + const void* next + XrVirtualKeyboardMETA keyboard + @@ -3183,6 +3463,12 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + @@ -3492,13 +3778,13 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + - + @@ -3649,6 +3935,12 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + @@ -3791,6 +4083,12 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + @@ -3870,6 +4168,66 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -4343,23 +4701,23 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrResult xrCreateVulkanInstanceKHR - XrInstance instance + XrInstance instance const XrVulkanInstanceCreateInfoKHR* createInfo - VkInstance* vulkanInstance - VkResult* vulkanResult + VkInstance* vulkanInstance + VkResult* vulkanResult XrResult xrCreateVulkanDeviceKHR - XrInstance instance - const XrVulkanDeviceCreateInfoKHR* createInfo - VkDevice* vulkanDevice - VkResult* vulkanResult + XrInstance instance + const XrVulkanDeviceCreateInfoKHR* createInfo + VkDevice* vulkanDevice + VkResult* vulkanResult XrResult xrGetVulkanGraphicsDevice2KHR - XrInstance instance + XrInstance instance const XrVulkanGraphicsDeviceGetInfoKHR* getInfo - VkPhysicalDevice* vulkanPhysicalDevice + VkPhysicalDevice* vulkanPhysicalDevice @@ -4579,7 +4937,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. uint32_t* bufferCountOutput uint8_t* buffer - + XrResult xrGetControllerModelPropertiesMSFT XrSession session XrControllerModelKeyMSFT modelKey @@ -4829,7 +5187,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrSpace space XrUuidEXT* uuid - + XrResult xrEnumerateSpaceSupportedComponentsFB XrSpace space uint32_t componentTypeCapacityInput @@ -4958,7 +5316,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. const XrSpaceQueryInfoBaseHeaderFB* info XrAsyncRequestIdFB* requestId - + XrResult xrRetrieveSpaceQueryResultsFB XrSession session XrAsyncRequestIdFB requestId @@ -4996,7 +5354,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + XrResult xrGetSpaceContainerFB XrSession session XrSpace space @@ -5016,19 +5374,19 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrSpace space XrRect3DfFB* boundingBox3DOutput - + XrResult xrGetSpaceSemanticLabelsFB XrSession session XrSpace space XrSemanticLabelsFB* semanticLabelsOutput - + XrResult xrGetSpaceBoundary2DFB XrSession session XrSpace space XrBoundary2DFB* boundary2DOutput - + XrResult xrGetSpaceRoomLayoutFB XrSession session XrSpace space @@ -5184,6 +5542,25 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrExternalCameraOCULUS* cameras + + + XrResult xrCreatePassthroughColorLutMETA + XrPassthroughFB passthrough + const XrPassthroughColorLutCreateInfoMETA* createInfo + XrPassthroughColorLutMETA* colorLut + + + + XrResult xrDestroyPassthroughColorLutMETA + XrPassthroughColorLutMETA colorLut + + + + XrResult xrUpdatePassthroughColorLutMETA + XrPassthroughColorLutMETA colorLut + const XrPassthroughColorLutUpdateInfoMETA* updateInfo + + XrResult xrEnumeratePerformanceMetricsCounterPathsMETA @@ -5267,6 +5644,120 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrHandTrackerEXT handTracker const XrForceFeedbackCurlApplyLocationsMNDX* locations + + + + XrResult xrCreatePlaneDetectorEXT + XrSession session + const XrPlaneDetectorCreateInfoEXT* createInfo + XrPlaneDetectorEXT* planeDetector + + + + XrResult xrDestroyPlaneDetectorEXT + XrPlaneDetectorEXT planeDetector + + + + XrResult xrBeginPlaneDetectionEXT + XrPlaneDetectorEXT planeDetector + const XrPlaneDetectorBeginInfoEXT* beginInfo + + + + XrResult xrGetPlaneDetectionStateEXT + XrPlaneDetectorEXT planeDetector + XrPlaneDetectionStateEXT* state + + + + XrResult xrGetPlaneDetectionsEXT + XrPlaneDetectorEXT planeDetector + const XrPlaneDetectorGetInfoEXT* info + XrPlaneDetectorLocationsEXT* locations + + + + XrResult xrGetPlanePolygonBufferEXT + XrPlaneDetectorEXT planeDetector + uint64_t planeId + uint32_t polygonBufferIndex + XrPlaneDetectorPolygonBufferEXT* polygonBuffer + + + + + + XrResult xrCreateVirtualKeyboardMETA + XrSession session + const XrVirtualKeyboardCreateInfoMETA* createInfo + XrVirtualKeyboardMETA* keyboard + + + + XrResult xrDestroyVirtualKeyboardMETA + XrVirtualKeyboardMETA keyboard + + + + XrResult xrCreateVirtualKeyboardSpaceMETA + XrSession session + XrVirtualKeyboardMETA keyboard + const XrVirtualKeyboardSpaceCreateInfoMETA* createInfo + XrSpace* keyboardSpace + + + + XrResult xrSuggestVirtualKeyboardLocationMETA + XrVirtualKeyboardMETA keyboard + const XrVirtualKeyboardLocationInfoMETA* locationInfo + + + + XrResult xrGetVirtualKeyboardScaleMETA + XrVirtualKeyboardMETA keyboard + float* scale + + + + XrResult xrSetVirtualKeyboardModelVisibilityMETA + XrVirtualKeyboardMETA keyboard + const XrVirtualKeyboardModelVisibilitySetInfoMETA* modelVisibility + + + + XrResult xrGetVirtualKeyboardModelAnimationStatesMETA + XrVirtualKeyboardMETA keyboard + XrVirtualKeyboardModelAnimationStatesMETA* animationStates + + + + XrResult xrGetVirtualKeyboardDirtyTexturesMETA + XrVirtualKeyboardMETA keyboard + uint32_t textureIdCapacityInput + uint32_t* textureIdCountOutput + uint64_t* textureIds + + + + XrResult xrGetVirtualKeyboardTextureDataMETA + XrVirtualKeyboardMETA keyboard + uint64_t textureId + XrVirtualKeyboardTextureDataMETA* textureData + + + + XrResult xrSendVirtualKeyboardInputMETA + XrVirtualKeyboardMETA keyboard + const XrVirtualKeyboardInputInfoMETA* info + XrPosef* interactorRootPose + + + + XrResult xrChangeVirtualKeyboardTextContextMETA + XrVirtualKeyboardMETA keyboard + const XrVirtualKeyboardTextContextChangeInfoMETA* changeInfo + @@ -5748,6 +6239,50 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -5795,6 +6330,8 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + @@ -6402,11 +6939,20 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + @@ -6642,6 +7188,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -6960,6 +7514,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -6982,6 +7544,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -7138,6 +7708,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -7213,6 +7791,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -7226,6 +7812,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -7481,7 +8075,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + @@ -7662,6 +8256,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + @@ -7981,6 +8583,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + @@ -8056,12 +8659,13 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - - + + - - - + + + + @@ -8069,6 +8673,10 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + @@ -8318,6 +8926,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + @@ -8521,10 +9130,61 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -8900,10 +9560,31 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + - - + + + + + + + + + + + + + + + + + + + + + + + @@ -9154,10 +9835,17 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + + + + + + + + @@ -9682,10 +10370,10 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + - - + + @@ -9810,6 +10498,20 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + + + + + + @@ -10114,17 +10816,60 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + - - + + + + + + + + + - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + @@ -10289,10 +11034,312 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + - - + + diff --git a/specification/scripts/cgenerator.py b/specification/scripts/cgenerator.py index 64300ea5..790579c8 100644 --- a/specification/scripts/cgenerator.py +++ b/specification/scripts/cgenerator.py @@ -513,7 +513,7 @@ def genCmd(self, cmdinfo, name, alias): self.appendSection('commandPointer', decls[1]) def misracstyle(self): - return self.genOpts.misracstyle; + return self.genOpts.misracstyle def misracppstyle(self): - return self.genOpts.misracppstyle; + return self.genOpts.misracppstyle diff --git a/specification/scripts/docgenerator.py b/specification/scripts/docgenerator.py index 5b5e414a..832498e1 100644 --- a/specification/scripts/docgenerator.py +++ b/specification/scripts/docgenerator.py @@ -354,6 +354,8 @@ def genType(self, typeinfo, name, alias): name, category)) else: body = self.genRequirements(name) + if category in ('define',): + body = body.strip() if alias: # If the type is an alias, just emit a typedef declaration body += 'typedef ' + alias + ' ' + name + ';\n' diff --git a/specification/scripts/findBareNormatives.sh b/specification/scripts/findBareNormatives.sh new file mode 100755 index 00000000..a5477b52 --- /dev/null +++ b/specification/scripts/findBareNormatives.sh @@ -0,0 +1,82 @@ +#!/usr/bin/env bash +# +# Copyright (c) 2018-2023, The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 + +# findBareNormatives.sh - search usage of words that are or seem like normatives, +# but lack normative markup/highlighting. +# +# Usage: scripts/findBareNormatives.sh [sources/chapters/...]... +# +# If one or more adoc file paths (relative to specification dir) are provided, +# only those paths are checked; otherwise all chapters are checked. + +set -e + +list_sources() { + ( + cd "$(dirname "$0")/.." + find sources/chapters -name "*.adoc" | grep -v appendix + ) +} + +SOURCES=() +if [ $# -gt 0 ]; then + echo "Only checking $*" + echo "" + SOURCES+=("$@") +else + while IFS='' read -r line; do SOURCES+=("$line"); done < <(list_sources) +fi + +# Base function for grepping through sources for errors +# Params: +# 1. Description of issue being checked for +# All remaining parameters: passed to grep, typically a regex or something like -e regex -e regex +# May set GREPCMD before call to do something other than egrep +grep_sources() { + DESC="$1" + shift + if ${GREPCMD:-grep -P} -n "$@" "${SOURCES[@]}"; then + echo "" + echo "*** Warning: Above lines matched grep with $*" + echo "*** $DESC" + echo "" + echo "" + FAILED="yes" + fi +} + + +# Common Params for the various check markup functions: +# 1. macro first letter (so s in slink) +# 2. description of category +# 3. expectation, completing the sentence "Things like this should..." +# 4. partial regex, which will be appended to the end of markup macro (eg slink:yourregex) + +### +# Actual Checks +### +( + export FAILED= + + cd "$(dirname "$0")/.." + grep_sources \ + "found normative without trailing colon - add colon or rephrase" \ + "\b(cannot|can|may|must|optional|should)\b(?!:)" + + # not looking for "required" because there are too many places we don't want the markup. + + grep_sources \ + "these suggest a normative, but are not normatives used in the OpenXR spec - rephrase" \ + "\b(shall|might|could|would)\b(?!:)" + + + if [ "$FAILED" != "" ]; then + echo "Failures detected" + exit 1 + fi + +) + diff --git a/specification/scripts/genRef.py b/specification/scripts/genRef.py index 0ddbc2a1..b5dfa189 100644 --- a/specification/scripts/genRef.py +++ b/specification/scripts/genRef.py @@ -835,6 +835,10 @@ def genExtension(baseDir, extpath, name, info): # Do not link to spec version or extension name - those ref pages are not created. continue + if req_name.startswith('/interaction_profiles'): + # We do not yet make ref pages for interaction profiles, though we may in the future + continue + if required.get('extends'): # These are either extensions of enumerated types, or const enum # values: neither of which get a ref page - although we could diff --git a/specification/scripts/reg.py b/specification/scripts/reg.py index 91902b0c..ebe7840c 100755 --- a/specification/scripts/reg.py +++ b/specification/scripts/reg.py @@ -1220,6 +1220,10 @@ def generateFeature(self, fname, ftype, dictionary): extname = elem.get('extname') version = elem.get('version') if extname is not None: + # if this enum is not defined in an extension that is being emitted + # then it should not be included in the emitted header file + if re.match(self.genOpts.emitExtensions, extname) is None: + continue; # 'supported' attribute was injected when the element was # moved into the group in Registry.parseTree() supported_list = elem.get('supported').split(",") diff --git a/specification/scripts/reserve_extensions.py b/specification/scripts/reserve_extensions.py new file mode 100755 index 00000000..cb3bf8da --- /dev/null +++ b/specification/scripts/reserve_extensions.py @@ -0,0 +1,32 @@ +#!/usr/bin/python3 +# +# Copyright 2020-2023 The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 +""" +Simple script to generate XML to reserve extensions. + +The generated XML still needs to be added to xr.xml and have +PrettyRegistryXML.OpenXR run on it. +""" + +import click + + +@click.command() +@click.option('--extensions', default=1, help='number of extensions to reserve') +@click.argument('vendor') +@click.argument('first_extension', type=int) +def generate_reserved_extensions(vendor: str, first_extension: int, extensions: int): + """Prints extension reservation XML.""" + vendor = vendor.upper() + for num in range(first_extension, first_extension + extensions): + print(" " + f""" + + + + + + + """.strip()) + print() diff --git a/specification/scripts/openxr-macros.rb b/specification/scripts/spec-macros.rb similarity index 89% rename from specification/scripts/openxr-macros.rb rename to specification/scripts/spec-macros.rb index bd042b2e..e1ab5808 100644 --- a/specification/scripts/openxr-macros.rb +++ b/specification/scripts/spec-macros.rb @@ -3,7 +3,7 @@ # SPDX-License-Identifier: Apache-2.0 #require 'asciidoctor/extensions' unless RUBY_ENGINE == 'opal' -RUBY_ENGINE == 'opal' ? (require 'openxr-macros/extension') : (require_relative 'openxr-macros/extension') +RUBY_ENGINE == 'opal' ? (require 'spec-macros/extension') : (require_relative 'spec-macros/extension') # All the inline macros we need Asciidoctor::Extensions.register do @@ -15,6 +15,7 @@ inline_macro RequiredInlineMacro inline_macro ShouldInlineMacro inline_macro ReflinkInlineMacro + inline_macro ApiextInlineMacro inline_macro FlinkInlineMacro inline_macro FnameInlineMacro inline_macro FtextInlineMacro diff --git a/specification/scripts/openxr-macros/extension.rb b/specification/scripts/spec-macros/extension.rb similarity index 52% rename from specification/scripts/openxr-macros/extension.rb rename to specification/scripts/spec-macros/extension.rb index c28154a8..08f953ca 100644 --- a/specification/scripts/openxr-macros/extension.rb +++ b/specification/scripts/spec-macros/extension.rb @@ -6,12 +6,16 @@ include ::Asciidoctor -class OpenXRInlineMacroBase < Extensions::InlineMacroProcessor +# This is the generated map of API interfaces in this spec build +require 'apimap.rb' +$apiNames = APInames.new + +class SpecInlineMacroBase < Extensions::InlineMacroProcessor use_dsl using_format :short end -class NormativeInlineMacroBase < OpenXRInlineMacroBase +class NormativeInlineMacroBase < SpecInlineMacroBase def text 'normative' end @@ -21,8 +25,34 @@ def process parent, target, attributes end end -class LinkInlineMacroBase < OpenXRInlineMacroBase +class LinkInlineMacroBase < SpecInlineMacroBase + # Check if a link macro target exists - overridden by specific macros + # Default assumption is that it does exist + def exists? target + return true + end + def process parent, target, attributes + if not exists? target + # If the macro target is not in this build, but has an alias, + # substitute that alias as the argument. + # Otherwise, turn the (attempted) link into text, and complain. + if $apiNames.nonexistent.has_key? target + oldtarget = target + target = $apiNames.nonexistent[oldtarget] + msg = 'Rewriting nonexistent link macro target: ' + @name.to_s + ':' + oldtarget + ' to ' + target + Asciidoctor::LoggerManager.logger.info msg + # Fall through + else + # Suppress warnings for apiext: macros as this is such a common case + if @name.to_s != 'apiext' + msg = 'Textifying unknown link macro target: ' + @name.to_s + ':' + target + Asciidoctor::LoggerManager.logger.warn msg + end + return create_inline parent, :quoted, '' + target + '' + end + end + if parent.document.attributes['cross-file-links'] return Inline.new(parent, :anchor, target, :type => :link, :target => (target + '.html')) else @@ -31,24 +61,21 @@ def process parent, target, attributes end end -class CodeInlineMacroBase < OpenXRInlineMacroBase +class CodeInlineMacroBase < SpecInlineMacroBase def process parent, target, attributes - - create_inline parent, :quoted, '' + target + '' + create_inline parent, :quoted, '' + target.gsub('→', '->') + '' end end -class StrongInlineMacroBase < OpenXRInlineMacroBase +class StrongInlineMacroBase < SpecInlineMacroBase def process parent, target, attributes - - create_inline parent, :quoted, '' + target + '' + create_inline parent, :quoted, '' + target.gsub('→', '->') + '' end end -class ParamInlineMacroBase < OpenXRInlineMacroBase +class ParamInlineMacroBase < SpecInlineMacroBase def process parent, target, attributes - - create_inline parent, :quoted, '' + target + '' + create_inline parent, :quoted, '' + target.gsub('→', '->') + '' end end @@ -115,14 +142,34 @@ def text end end +# Generic reference page link to any entity with an anchor/refpage class ReflinkInlineMacro < LinkInlineMacroBase named :reflink - match /reflink:(\w+)/ + match /reflink:([-\w]+)/ +end + +# Link to an extension appendix/refpage +class ApiextInlineMacro < LinkInlineMacroBase + named :apiext + match /apiext:(\w+)/ + + def exists? target + $apiNames.features.has_key? target + end + + def process parent, target, attributes + node = super parent, target, attributes + create_inline parent, :quoted, %(#{node.convert}) + end end class FlinkInlineMacro < LinkInlineMacroBase named :flink match /flink:(\w+)/ + + def exists? target + $apiNames.protos.has_key? target + end end class FnameInlineMacro < StrongInlineMacroBase @@ -143,6 +190,10 @@ class SnameInlineMacro < CodeInlineMacroBase class SlinkInlineMacro < LinkInlineMacroBase named :slink match /slink:(\w+)/ + + def exists? target + $apiNames.structs.has_key? target or $apiNames.handles.has_key? target + end end class StextInlineMacro < CodeInlineMacroBase @@ -153,11 +204,19 @@ class StextInlineMacro < CodeInlineMacroBase class EnameInlineMacro < CodeInlineMacroBase named :ename match /ename:(\w+)/ + + def exists? target + $apiNames.consts.has_key? target + end end class ElinkInlineMacro < LinkInlineMacroBase named :elink match /elink:(\w+)/ + + def exists? target + $apiNames.enums.has_key? target or $apiNames.flags.has_key? target + end end class EtextInlineMacro < CodeInlineMacroBase @@ -165,14 +224,16 @@ class EtextInlineMacro < CodeInlineMacroBase match /etext:([\w\*]+)/ end +# this does not handle any [] at the moment + class PnameInlineMacro < ParamInlineMacroBase named :pname - match /pname:((\w[\w.]*)*\w+)/ + match /pname:(\w+((\.|→)\w+)*)/ end class PtextInlineMacro < ParamInlineMacroBase named :ptext - match /ptext:((\w[\w.]*)*\w+)/ + match /ptext:([\w\*]+((\.|→)[\w\*]+)*)/ end class DnameInlineMacro < CodeInlineMacroBase @@ -183,6 +244,10 @@ class DnameInlineMacro < CodeInlineMacroBase class DlinkInlineMacro < LinkInlineMacroBase named :dlink match /dlink:(\w+)/ + + def exists? target + $apiNames.defines.has_key? target or target.start_with? 'XR_USE' or target.start_with? 'XRAPI' or target == 'XR_NO_STDINT_H' + end end class TnameInlineMacro < CodeInlineMacroBase @@ -193,26 +258,41 @@ class TnameInlineMacro < CodeInlineMacroBase class TlinkInlineMacro < LinkInlineMacroBase named :tlink match /tlink:(\w+)/ + + def exists? target + $apiNames.flags.has_key? target or + $apiNames.funcpointers.has_key? target or + $apiNames.defines.has_key? target + end end class BasetypeInlineMacro < LinkInlineMacroBase named :basetype match /basetype:(\w+)/ - def process parent, target, attributes - if parent.document.attributes['cross-file-links'] - return Inline.new(parent, :anchor, target, :type => :link, :target => (target + '.html')) - else - return Inline.new(parent, :anchor, '' + target + '', :type => :xref, :target => ('#' + target), :attributes => {'fragment' => target, 'refid' => target}) - end + + # def process parent, target, attributes + # if parent.document.attributes['cross-file-links'] + # return Inline.new(parent, :anchor, target, :type => :link, :target => (target + '.html')) + # else + # return Inline.new(parent, :anchor, '' + target + '', :type => :xref, :target => ('#' + target), :attributes => {'fragment' => target, 'refid' => target}) + # end + # end + + def exists? target + $apiNames.basetypes.has_key? target end end -class CodeInlineMacro < StrongInlineMacroBase +# This does not include the full range of code: use +# It allows embedded periods (field separators) and wildcards if followed by +# another word, and an ending wildcard. + +class CodeInlineMacro < CodeInlineMacroBase named :code - match /code:([*\w]+)/ + match /code:(\w+([.*]\w+)*\**)/ end -class SemanticSubpathMacroBase < OpenXRInlineMacroBase +class SemanticSubpathMacroBase < SpecInlineMacroBase def process parent, target, attributes create_inline parent, :quoted, '…' + target + '' end @@ -223,7 +303,7 @@ class SemanticSubpathInlineMacro < SemanticSubpathMacroBase match /subpathname:([\w\/\-\.]+)/ end -class SemanticPathMacroBase < OpenXRInlineMacroBase +class SemanticPathMacroBase < SpecInlineMacroBase def process parent, target, attributes create_inline parent, :quoted, '' + target + '' end @@ -234,7 +314,7 @@ class SemanticPathInlineMacro < SemanticPathMacroBase match /pathname:([\w\/\-\.]+)/ end -class ActionRelatedNameMacroBase < OpenXRInlineMacroBase +class ActionRelatedNameMacroBase < SpecInlineMacroBase def process parent, target, attributes create_inline parent, :quoted, '' + target + '' end diff --git a/specification/scripts/spec_tools/conventions.py b/specification/scripts/spec_tools/conventions.py index 2086d878..8d0f0726 100644 --- a/specification/scripts/spec_tools/conventions.py +++ b/specification/scripts/spec_tools/conventions.py @@ -77,7 +77,7 @@ def __init__(self): def formatExtension(self, name): """Mark up an extension name as a link the spec.""" - return '`<<{}>>`'.format(name) + return 'apiext:{}'.format(name) @property @abc.abstractmethod diff --git a/specification/scripts/spec_tools/macro_checker_file.py b/specification/scripts/spec_tools/macro_checker_file.py index ed5e5d37..46c9c2ad 100644 --- a/specification/scripts/spec_tools/macro_checker_file.py +++ b/specification/scripts/spec_tools/macro_checker_file.py @@ -1588,7 +1588,7 @@ def makeSearch(self): def makeMacroMarkup(self, newMacro=None, newEntity=None, data=None): """Construct appropriate markup for referring to an entity. - Typically constructs macro:entity, but can construct `<>` if the supplied + Typically constructs macro:entity, but can construct apiext:EXTENSION_NAME if the supplied entity is identified as an extension. Arguments: @@ -1615,14 +1615,14 @@ def makeMacroMarkup(self, newMacro=None, newEntity=None, data=None): def makeExtensionLink(self, newEntity=None): """Create a correctly-formatted link to an extension. - Result takes the form `<>`. + Result takes the form apiext:EXTENSION_NAME. Argument: newEntity -- The extension name to link to. Defaults to self.entity. """ if not newEntity: newEntity = self.entity - return '`<<{}>>`'.format(newEntity) + return 'apiext:{}'.format(newEntity) def computeExpectedRefPageFromInclude(self, entity): """Compute the expected ref page entity based on an include entity name.""" diff --git a/specification/scripts/test_check_spec_links.py b/specification/scripts/test_check_spec_links.py index 1eac7d72..b5426470 100644 --- a/specification/scripts/test_check_spec_links.py +++ b/specification/scripts/test_check_spec_links.py @@ -162,7 +162,7 @@ def test_extension(ckr): # Check formatting of extension names: # the following is the canonical way to refer to an extension # (link wrapped in backticks) - expected_replacement = '`<<%s>>`' % EXT + expected_replacement = 'apiext:%s' % EXT # Extension name mentioned without any markup, should be added assert (loneMsgReplacement(ckr.check('asdf %s asdf' % EXT)) @@ -187,7 +187,7 @@ def test_extension(ckr): == expected_replacement) # This shouldn't cause errors because this is how we want it to look. - assert not ckr.check('asdf `<<%s>>` asdf' % EXT).messages + assert not ckr.check('asdf apiext:%s asdf' % EXT).messages # This doesn't (shouldn't?) cause errors because just backticks on their own # "escape" names from the "missing markup" tests. diff --git a/specification/scripts/xml_consistency.py b/specification/scripts/xml_consistency.py index df31c6d7..dda22c74 100755 --- a/specification/scripts/xml_consistency.py +++ b/specification/scripts/xml_consistency.py @@ -120,10 +120,7 @@ def makeRegistry(self): # and it provides file line info which is useful in messages. try: import lxml.etree as etree - HAS_LXML = True except ImportError: - HAS_LXML = False - if not HAS_LXML: return super().makeRegistry() registryFile = str(SPECIFICATION_DIR / 'registry/xr.xml') @@ -240,7 +237,7 @@ def check_enum_naming(self, enum_type): if bare_end: # If bare_end is set, end is always non-empty because it means it's a bitmask. - assert(end) + assert end if not name.endswith(end) and not stripped_name.endswith(bare_end): self.record_error('Got an enum value whose name does not match the pattern: got', name, 'but expected something that ended with', end, ', or', bare_end, @@ -275,7 +272,7 @@ def get_codes_for_command_and_type(self, cmd_name, type_name): codes = super().get_codes_for_command_and_type(cmd_name, type_name) # Filter out any based on the specific command - if cmd_name.startswith(_DESTROY_PREFIX): + if codes is not None and cmd_name.startswith(_DESTROY_PREFIX): # xrDestroyX should not return XR_ERROR_anything_LOST or XR_anything_LOSS_PENDING codes = {x for x in codes if not x.endswith("_LOST")} codes = {x for x in codes if not x.endswith("_LOSS_PENDING")} @@ -346,7 +343,7 @@ def check_two_call_count_output(self, param_name, param_elem, match): self.record_error('Two-call-idiom call has count parameter', param_name, 'with type', param_type, 'instead of uint32_t') type_elem = param_elem.find('type') - assert(type_elem is not None) + assert type_elem is not None tail = type_elem.tail.strip() if '*' != tail: @@ -370,7 +367,7 @@ def check_two_call_array(self, param_name, param_elem, match): optional, '- expected "true"') type_elem = param_elem.find('type') - assert(type_elem is not None) + assert type_elem is not None tail = type_elem.tail.strip() if '*' not in tail: @@ -378,6 +375,7 @@ def check_two_call_array(self, param_name, param_elem, match): 'that is not a pointer:', type_elem.text, type_elem.tail) length = LengthEntry.parse_len_from_param(param_elem) + assert length is not None if not length[0].other_param_name: self.record_error('Two-call-idiom call has array parameter', param_name, 'whose first length is not another parameter:', length[0]) @@ -444,9 +442,11 @@ def check_two_call_command(self, name, info, params): if not array_param_name: self.record_error('Apparent two-call-idiom call missing an array output parameter') - if not capacity_input_param_name or \ - not count_output_param_name or \ - not array_param_name: + if (capacity_input_param_match is None or + count_output_param_match is None or + capacity_input_param_name is None or + count_output_param_name is None or + array_param_name is None): # If we're missing at least one, stop checking two-call stuff here. return diff --git a/specification/scripts/xrconventions.py b/specification/scripts/xrconventions.py index 50efc310..6fe3d307 100644 --- a/specification/scripts/xrconventions.py +++ b/specification/scripts/xrconventions.py @@ -249,7 +249,7 @@ def spec_no_reflow_dirs(self): """Return a set of directories not to automatically descend into when reflowing spec text """ - return ('styleguide',) + return tuple() @property def zero(self): diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 124acb20..f8af1a0d 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1,4 +1,4 @@ -# Copyright (c) 2017 The Khronos Group Inc. +# Copyright (c) 2017-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 # @@ -14,8 +14,6 @@ # See the License for the specific language governing permissions and # limitations under the License. # -# Author: -# if(POLICY CMP0075) cmake_policy(SET CMP0075 NEW) @@ -39,6 +37,9 @@ if(CMAKE_SYSTEM_NAME STREQUAL "WindowsStore" OR (WIN32 AND CMAKE_GENERATOR_PLATF set(OPENGL_INCOMPATIBLE TRUE) set(VULKAN_INCOMPATIBLE TRUE) message(STATUS "OpenGL/Vulkan disabled due to incompatibility") +elseif(CMAKE_SYSTEM_NAME STREQUAL "Darwin") + set(OPENGL_INCOMPATIBLE TRUE) + message(STATUS "OpenGL disabled as no bindings exist for OpenGL on Darwin") elseif(ANDROID) set(OPENGL_INCOMPATIBLE TRUE) find_path(ANDROID_NATIVE_APP_GLUE android_native_app_glue.h PATHS ${ANDROID_NDK}/sources/android/native_app_glue) @@ -179,7 +180,20 @@ if((OPENGL_FOUND OR OpenGLES_FOUND) AND EXISTS "${CMAKE_CURRENT_SOURCE_DIR}/comm add_library(android_native_app_glue OBJECT "${ANDROID_NATIVE_APP_GLUE}/android_native_app_glue.c") target_include_directories(android_native_app_glue PUBLIC ${ANDROID_NATIVE_APP_GLUE}) target_compile_options(android_native_app_glue PRIVATE -Wno-unused-parameter) + elseif(APPLE) + set(apple_c_compile_options + -x objective-c++ + ) + target_compile_options(openxr-gfxwrapper PRIVATE + ${apple_c_compile_options} + ) + target_link_libraries(openxr-gfxwrapper PUBLIC + "-framework AppKit" + "-framework Foundation" + "-framework CoreGraphics" + ) endif() + set_target_properties(openxr-gfxwrapper PROPERTIES FOLDER ${HELPER_FOLDER}) message(STATUS "Enabling OpenGL support in hello_xr, loader_test, and conformance, if configured") endif() @@ -192,6 +206,14 @@ endif() # Several files use these compile time platform switches if(WIN32) add_definitions(-DXR_USE_PLATFORM_WIN32) +elseif(APPLE) + if(CMAKE_SYSTEM_NAME STREQUAL "Darwin") + if(DARWIN_TARGET_OS_NAME STREQUAL "ios") + add_definitions(-DXR_USE_PLATFORM_IOS) + else() + add_definitions(-DXR_USE_PLATFORM_MACOS) + endif() + endif() elseif(ANDROID) add_definitions(-DXR_USE_PLATFORM_ANDROID) set(OPENXR_ANDROID_VERSION_SUFFIX "" CACHE STRING "Suffix for generated Android artifacts.") @@ -410,7 +432,11 @@ if(NOT MSVC) set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} ${FLAG}") endif() endforeach() - set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,--no-undefined") + if(APPLE) + set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,-undefined,error") + else() + set(CMAKE_SHARED_LINKER_FLAGS "${CMAKE_SHARED_LINKER_FLAGS} -Wl,--no-undefined") + endif() endif() if(ANDROID) diff --git a/src/common/filesystem_utils.cpp b/src/common/filesystem_utils.cpp index d3d4182f..16e6ff32 100644 --- a/src/common/filesystem_utils.cpp +++ b/src/common/filesystem_utils.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2017 The Khronos Group Inc. +// Copyright (c) 2017-2023, The Khronos Group Inc. // Copyright (c) 2017 Valve Corporation // Copyright (c) 2017 LunarG, Inc. // diff --git a/src/common/filesystem_utils.hpp b/src/common/filesystem_utils.hpp index 4a5c987e..3dea1b2c 100644 --- a/src/common/filesystem_utils.hpp +++ b/src/common/filesystem_utils.hpp @@ -1,4 +1,4 @@ -// Copyright (c) 2017 The Khronos Group Inc. +// Copyright (c) 2017-2023, The Khronos Group Inc. // Copyright (c) 2017 Valve Corporation // Copyright (c) 2017 LunarG, Inc. // diff --git a/src/common/gfxwrapper_opengl.c b/src/common/gfxwrapper_opengl.c index cc03a788..8b5c0899 100644 --- a/src/common/gfxwrapper_opengl.c +++ b/src/common/gfxwrapper_opengl.c @@ -223,11 +223,14 @@ Get proc address / extensions #if defined(OS_WINDOWS) PROC GetExtension(const char *functionName) { return wglGetProcAddress(functionName); } #elif defined(OS_APPLE) -void (*GetExtension(const char *functionName))() { return NULL; } +void (*GetExtension(const char *functionName))(void) { + (void)functionName; + return NULL; +} #elif defined(OS_LINUX_XCB) || defined(OS_LINUX_XLIB) || defined(OS_LINUX_XCB_GLX) -void (*GetExtension(const char *functionName))() { return glXGetProcAddress((const GLubyte *)functionName); } +void (*GetExtension(const char *functionName))(void) { return glXGetProcAddress((const GLubyte *)functionName); } #elif defined(OS_ANDROID) || defined(OS_LINUX_WAYLAND) -void (*GetExtension(const char *functionName))() { return eglGetProcAddress(functionName); } +void (*GetExtension(const char *functionName))(void) { return eglGetProcAddress(functionName); } #endif GLint glGetInteger(GLenum pname) { @@ -1626,6 +1629,7 @@ void ksGpuContext_UnsetCurrent(ksGpuContext *context) { #elif defined(OS_LINUX_XCB) xcb_glx_make_current(context->connection, 0, 0, 0); #elif defined(OS_APPLE_MACOS) + (void)context; CGLSetCurrentContext(NULL); #elif defined(OS_ANDROID) || defined(OS_LINUX_WAYLAND) EGL(eglMakeCurrent(context->display, EGL_NO_SURFACE, EGL_NO_SURFACE, EGL_NO_CONTEXT)); @@ -2985,6 +2989,7 @@ NSAutoreleasePool *autoReleasePool; return YES; } - (void)keyDown:(NSEvent *)event { + (void)event; } @end @@ -2998,6 +3003,7 @@ NSAutoreleasePool *autoReleasePool; return YES; } - (void)keyDown:(NSEvent *)event { + (void)event; } @end @@ -3083,8 +3089,8 @@ bool ksGpuWindow_Create(ksGpuWindow *window, ksDriverInstance *instance, const k } } } - CGDisplayErr err = CGDisplaySetDisplayMode(window->display, bestDisplayMode, NULL); - if (err != CGDisplayNoErr) { + CGDisplayErr cgderr = CGDisplaySetDisplayMode(window->display, bestDisplayMode, NULL); + if (cgderr != CGDisplayNoErr) { CFRelease(displayModes); return false; } diff --git a/src/common/gfxwrapper_opengl.h b/src/common/gfxwrapper_opengl.h index e800f26f..a4c245d3 100644 --- a/src/common/gfxwrapper_opengl.h +++ b/src/common/gfxwrapper_opengl.h @@ -124,9 +124,9 @@ extern "C" { #elif defined(__APPLE__) #define OS_APPLE #include -#if __IPHONE_OS_VERSION_MAX_ALLOWED +#if defined(__IPHONE_OS_VERSION_MAX_ALLOWED) && __IPHONE_OS_VERSION_MAX_ALLOWED #define OS_APPLE_IOS -#elif __MAC_OS_X_VERSION_MAX_ALLOWED +#elif defined(__MAC_OS_X_VERSION_MAX_ALLOWED) && __MAC_OS_X_VERSION_MAX_ALLOWED #define OS_APPLE_MACOS #endif #elif defined(__linux__) @@ -414,7 +414,7 @@ Common defines #define ES_HIGHP "" // GLSL "430" disallows a precision qualifier on a image2D #endif -void GlInitExtensions(); +void GlInitExtensions(void); /* ================================================================================================================================ diff --git a/src/common/object_info.cpp b/src/common/object_info.cpp index 3f6b39c4..3f8f96bc 100644 --- a/src/common/object_info.cpp +++ b/src/common/object_info.cpp @@ -132,6 +132,8 @@ XrSdkSessionLabel::XrSdkSessionLabel(const XrDebugUtilsLabelEXT& label_info, boo : label_name(label_info.labelName), debug_utils_label(label_info), is_individual_label(individual) { // Update the c string pointer to the one we hold. debug_utils_label.labelName = label_name.c_str(); + // Zero out the next pointer to avoid a dangling pointer + debug_utils_label.next = nullptr; } XrSdkSessionLabelPtr XrSdkSessionLabel::make(const XrDebugUtilsLabelEXT& label_info, bool individual) { @@ -143,7 +145,7 @@ void DebugUtilsData::AddObjectName(uint64_t object_handle, XrObjectType object_t } // We always want to remove the old individual label before we do anything else. -// So, do that in it's own method +// So, do that in its own method void DebugUtilsData::RemoveIndividualLabel(XrSdkSessionLabelList& label_vec) { if (!label_vec.empty() && label_vec.back()->is_individual_label) { label_vec.pop_back(); diff --git a/src/common/platform_utils.hpp b/src/common/platform_utils.hpp index 6ea95f4e..219d1978 100644 --- a/src/common/platform_utils.hpp +++ b/src/common/platform_utils.hpp @@ -11,6 +11,7 @@ #include "xr_dependencies.h" #include +#include #include // OpenXR paths and registry key locations @@ -59,9 +60,11 @@ static inline char* ImplGetSecureEnv(const char* name) { #elif defined(HAVE___SECURE_GETENV) return __secure_getenv(name); #else +// clang-format off #pragma message( \ "Warning: Falling back to non-secure getenv for environmental" \ "lookups! Consider updating to a different libc.") + // clang-format on return ImplGetEnv(name); #endif @@ -313,7 +316,7 @@ static inline bool PlatformUtilsSetEnv(const char* /* name */, const char* /* va // Intended to be only used as a fallback on Android, with a more open, "native" technique used in most cases static inline bool PlatformGetGlobalRuntimeFileName(uint16_t major_version, std::string& file_name) { // Prefix for the runtime JSON file name - static const char* rt_dir_prefixes[] = {"/oem", "/vendor", "/system"}; + static const char* rt_dir_prefixes[] = {"/product", "/odm", "/oem", "/vendor", "/system"}; static const std::string rt_filename = "/active_runtime.json"; static const std::string subdir = "/etc/openxr/"; for (const auto prefix : rt_dir_prefixes) { diff --git a/src/conformance/CMakeLists.txt b/src/conformance/CMakeLists.txt index 3ab70f66..49d2c566 100644 --- a/src/conformance/CMakeLists.txt +++ b/src/conformance/CMakeLists.txt @@ -18,6 +18,8 @@ # add_subdirectory(conformance_layer) +add_subdirectory(utilities) +add_subdirectory(framework) add_subdirectory(conformance_test) if(NOT ANDROID) add_subdirectory(conformance_cli) diff --git a/src/conformance/build.gradle b/src/conformance/build.gradle index 48ccef46..8276908d 100644 --- a/src/conformance/build.gradle +++ b/src/conformance/build.gradle @@ -10,7 +10,7 @@ buildscript { mavenCentral() } dependencies { - classpath 'com.android.tools.build:gradle:7.0.4' + classpath 'com.android.tools.build:gradle:7.4.2' } } @@ -51,15 +51,14 @@ task copyLayerInfo(type: Copy) { } android { - compileSdkVersion 29 - buildToolsVersion "30.0.3" + compileSdk 29 ndkVersion "21.4.7075529" + buildToolsVersion = "30.0.3" defaultConfig { applicationId "org.khronos.openxr.cts" // for Vulkan, need at least 24 minSdkVersion 24 - compileSdkVersion 26 versionName = project.versionOpenXR.toString() + project.versionQualifier versionCode = project.versionOpenXR.getVersionCode() diff --git a/src/conformance/checkSchema.sh b/src/conformance/checkSchema.sh index 88bc2724..ce1d181f 100755 --- a/src/conformance/checkSchema.sh +++ b/src/conformance/checkSchema.sh @@ -1,6 +1,6 @@ #!/bin/sh -e # -# Copyright (c) 2019-2022, The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 # diff --git a/src/conformance/conformance_cli/CMakeLists.txt b/src/conformance/conformance_cli/CMakeLists.txt index 56eb45dc..f9cfde4c 100644 --- a/src/conformance/conformance_cli/CMakeLists.txt +++ b/src/conformance/conformance_cli/CMakeLists.txt @@ -54,6 +54,8 @@ endif() set_property(TARGET conformance_cli PROPERTY INSTALL_RPATH $ORIGIN) +set_target_properties(conformance_cli PROPERTIES FOLDER ${CONFORMANCE_TESTS_FOLDER}) + install( TARGETS conformance_cli RUNTIME DESTINATION conformance diff --git a/src/conformance/conformance_cli/main.cpp b/src/conformance/conformance_cli/main.cpp index bf878f2a..5358c733 100644 --- a/src/conformance/conformance_cli/main.cpp +++ b/src/conformance/conformance_cli/main.cpp @@ -43,7 +43,7 @@ namespace void SetupConsole() { -#if _WIN32 // Enable ANSI style color escape codes on Windows. Not enabled by default :-( +#if defined(_WIN32) // Enable ANSI style color escape codes on Windows. Not enabled by default :-( DWORD consoleMode; if (GetConsoleMode(GetStdHandle(STD_OUTPUT_HANDLE), &consoleMode)) { consoleMode |= ENABLE_VIRTUAL_TERMINAL_PROCESSING; diff --git a/src/conformance/conformance_layer/CMakeLists.txt b/src/conformance/conformance_layer/CMakeLists.txt index cb64625b..01fa7e22 100644 --- a/src/conformance/conformance_layer/CMakeLists.txt +++ b/src/conformance/conformance_layer/CMakeLists.txt @@ -88,6 +88,12 @@ if(ANDROID) target_link_libraries(XrApiLayer_runtime_conformance ${ANDROID_LOG_LIBRARY}) endif() +if(APPLE) + set_target_properties(XrApiLayer_runtime_conformance PROPERTIES LINK_FLAGS "-Wl,-exported_symbols_list,${CMAKE_CURRENT_SOURCE_DIR}/XrApiLayer_conformance_layer.expsym") +elseif(NOT WIN32) + set_target_properties(XrApiLayer_runtime_conformance PROPERTIES LINK_FLAGS "-Wl,-Bsymbolic,--exclude-libs,ALL") +endif() + if(BUILD_CONFORMANCE_CLI) # Copy conformance layer files to conformance_cli binary folder add_custom_command(TARGET XrApiLayer_runtime_conformance POST_BUILD @@ -96,6 +102,8 @@ if(BUILD_CONFORMANCE_CLI) ) endif() +set_target_properties(XrApiLayer_runtime_conformance PROPERTIES FOLDER ${CONFORMANCE_TESTS_FOLDER}) + install( FILES ${CMAKE_CURRENT_BINARY_DIR}/XrApiLayer_runtime_conformance.json DESTINATION conformance diff --git a/src/conformance/conformance_layer/HandleState.cpp b/src/conformance/conformance_layer/HandleState.cpp index 65337509..6e729ff6 100644 --- a/src/conformance/conformance_layer/HandleState.cpp +++ b/src/conformance/conformance_layer/HandleState.cpp @@ -14,10 +14,21 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include "gen_dispatch.h" +#include "HandleState.h" + #include "Common.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + namespace { struct HandleStateKeyHash diff --git a/src/conformance/conformance_layer/HandleState.h b/src/conformance/conformance_layer/HandleState.h index ee4f4de8..2d774fdd 100644 --- a/src/conformance/conformance_layer/HandleState.h +++ b/src/conformance/conformance_layer/HandleState.h @@ -19,7 +19,6 @@ // #pragma once -#include "xr_dependencies.h" #include #include diff --git a/src/conformance/conformance_layer/Negotiate.cpp b/src/conformance/conformance_layer/Negotiate.cpp index a410d261..dfeafdc1 100644 --- a/src/conformance/conformance_layer/Negotiate.cpp +++ b/src/conformance/conformance_layer/Negotiate.cpp @@ -67,9 +67,9 @@ namespace #define LAYER_EXPORT #endif -extern "C" LAYER_EXPORT XrResult XRAPI_CALL xrNegotiateLoaderApiLayerInterface(const XrNegotiateLoaderInfo* loaderInfo, - const char* /*apiLayerName*/, - XrNegotiateApiLayerRequest* apiLayerRequest) +extern "C" LAYER_EXPORT XRAPI_ATTR XrResult XRAPI_CALL xrNegotiateLoaderApiLayerInterface(const XrNegotiateLoaderInfo* loaderInfo, + const char* /*apiLayerName*/, + XrNegotiateApiLayerRequest* apiLayerRequest) { if (loaderInfo->structType != XR_LOADER_INTERFACE_STRUCT_LOADER_INFO || loaderInfo->structVersion != XR_LOADER_INFO_STRUCT_VERSION || loaderInfo->structSize != sizeof(XrNegotiateLoaderInfo)) { @@ -94,7 +94,7 @@ extern "C" LAYER_EXPORT XrResult XRAPI_CALL xrNegotiateLoaderApiLayerInterface(c apiLayerRequest->layerInterfaceVersion = XR_CURRENT_LOADER_API_LAYER_VERSION; apiLayerRequest->layerApiVersion = LayerApiVersion; apiLayerRequest->getInstanceProcAddr = ConformanceLayer_xrGetInstanceProcAddr; - apiLayerRequest->createApiLayerInstance = reinterpret_cast(ConformanceLayer_RegisterInstance); + apiLayerRequest->createApiLayerInstance = ConformanceLayer_RegisterInstance; return XR_SUCCESS; } diff --git a/src/conformance/conformance_layer/XrApiLayer_conformance_layer.expsym b/src/conformance/conformance_layer/XrApiLayer_conformance_layer.expsym new file mode 100644 index 00000000..ec596832 --- /dev/null +++ b/src/conformance/conformance_layer/XrApiLayer_conformance_layer.expsym @@ -0,0 +1,5 @@ +# Copyright (c) 2019-2023, The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 + +_xrNegotiateLoaderApiLayerInterface diff --git a/src/conformance/conformance_test/CMakeLists.txt b/src/conformance/conformance_test/CMakeLists.txt index 0465f8ce..cb78c7d7 100644 --- a/src/conformance/conformance_test/CMakeLists.txt +++ b/src/conformance/conformance_test/CMakeLists.txt @@ -15,16 +15,10 @@ # limitations under the License. # -run_xr_xml_generate(conformance_generator.py function_info.cpp - ${CMAKE_CURRENT_SOURCE_DIR}/../../scripts/template_function_info.cpp) - file(GLOB LOCAL_HEADERS "*.h") file(GLOB LOCAL_SOURCE "*.cpp") -file(GLOB FRAMEWORK_HEADERS "../framework/*.h") -file(GLOB FRAMEWORK_SOURCE "../framework/*.cpp") file(GLOB ASSETS "composition_examples/*.png" "SourceCodePro-Regular.otf") -file(GLOB VULKAN_SHADERS "vulkan_shaders/*.glsl") # For including compiled shaders include_directories(${CMAKE_CURRENT_BINARY_DIR}) @@ -32,9 +26,6 @@ include_directories(${CMAKE_CURRENT_BINARY_DIR}) add_library(conformance_test SHARED ${LOCAL_HEADERS} ${LOCAL_SOURCE} - ${FRAMEWORK_HEADERS} - ${FRAMEWORK_SOURCE} - ${CMAKE_CURRENT_BINARY_DIR}/function_info.cpp ${VULKAN_SHADERS} ${PROJECT_SOURCE_DIR}/src/common/platform_utils.hpp ) @@ -42,11 +33,10 @@ if(ANDROID) target_sources(conformance_test PRIVATE $) endif() -compile_glsl(run_conformance_test_glsl_compiles ${VULKAN_SHADERS}) +target_link_libraries(conformance_test PRIVATE conformance_utilities conformance_framework) add_dependencies(conformance_test generate_openxr_header - run_conformance_test_glsl_compiles XrApiLayer_runtime_conformance ) openxr_add_filesystem_utils(conformance_test) @@ -54,17 +44,11 @@ openxr_add_filesystem_utils(conformance_test) target_include_directories(conformance_test PRIVATE ${CMAKE_CURRENT_LIST_DIR}/../framework - ${CMAKE_CURRENT_LIST_DIR}/../../common + ${PROJECT_SOURCE_DIR}/src/common ${PROJECT_SOURCE_DIR}/src/external - # Strong types for integers, etc. - ${CMAKE_CURRENT_LIST_DIR}/../../external/type-lite/include - - # Backport of std::span functionality to pre-C++17 - ${CMAKE_CURRENT_LIST_DIR}/../../external/span-lite/include - # Earcut algorithm for simple polygon triangulation - ${CMAKE_CURRENT_LIST_DIR}/../../external/earcut/include + ${PROJECT_SOURCE_DIR}/src/external/earcut/include # for openxr.h: ${PROJECT_BINARY_DIR}/include @@ -76,44 +60,16 @@ target_include_directories(conformance_test ${PROJECT_SOURCE_DIR}/external/include ) -if(Vulkan_FOUND) - target_include_directories(conformance_test - PRIVATE ${Vulkan_INCLUDE_DIRS} - ) -endif() - source_group("Header Files" FILES ${LOCAL_HEADERS}) source_group("Source Files" FILES ${LOCAL_SOURCE}) -source_group("Framework Header Files" FILES ${FRAMEWORK_HEADERS}) -source_group("Framework Source Files" FILES ${FRAMEWORK_SOURCE}) -source_group("Vulkan Shaders" FILES ${VULKAN_SHADERS}) - -if(Vulkan_LIBRARY) - target_link_libraries(conformance_test PRIVATE ${Vulkan_LIBRARY}) -endif() - -if(TARGET openxr-gfxwrapper) - target_link_libraries(conformance_test PRIVATE openxr-gfxwrapper) -endif() - -if(GLSLANG_VALIDATOR AND NOT GLSLC_COMMAND) - target_compile_definitions(conformance_test PRIVATE USE_GLSLANGVALIDATOR) -endif() - -target_link_libraries(conformance_test PRIVATE openxr_loader Threads::Threads Catch2) +target_link_libraries(conformance_test PRIVATE conformance_framework) if(WIN32) target_compile_definitions(conformance_test PRIVATE _CRT_SECURE_NO_WARNINGS) if(MSVC) target_compile_options(conformance_test PRIVATE /Zc:wchar_t /Zc:forScope /W4 /WX /wd4996) endif() - if(MSVC) - # Right now can't build this on MinGW because of directxcolors, etc. - target_link_libraries(conformance_test PRIVATE d3d11 d3d12 d3dcompiler dxgi) - else() - target_compile_definitions(conformance_test PRIVATE MISSING_DIRECTX_COLORS) - endif() endif() @@ -159,6 +115,7 @@ if(BUILD_CONFORMANCE_CLI) ) endif() +set_target_properties(conformance_test PROPERTIES FOLDER ${CONFORMANCE_TESTS_FOLDER}) install( TARGETS conformance_test diff --git a/src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png b/src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png new file mode 100644 index 0000000000000000000000000000000000000000..6147309d9bfb6f63e261dedc2c6e0090fea81bcf GIT binary patch literal 143122 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%0D@jl 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 SCd3wiwD)qVvK7*%LH`5DRmJZB literal 0 HcmV?d00001 diff --git a/src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png.license b/src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png.license new file mode 100644 index 00000000..bc07946a --- /dev/null +++ b/src/conformance/conformance_test/composition_examples/ext_plane_detection_contour.png.license @@ -0,0 +1,3 @@ +SPDX-FileCopyrightText: 2019-2023, The Khronos Group Inc. + +SPDX-License-Identifier: Apache-2.0 diff --git a/src/conformance/conformance_test/composition_examples/palm_pose.png.license b/src/conformance/conformance_test/composition_examples/palm_pose.png.license index 31c3e55d..6d36aba0 100644 --- a/src/conformance/conformance_test/composition_examples/palm_pose.png.license +++ b/src/conformance/conformance_test/composition_examples/palm_pose.png.license @@ -1,3 +1,3 @@ -SPDX-FileCopyrightText: 2020-2021, The Khronos Group Inc. +SPDX-FileCopyrightText: 2020-2023, The Khronos Group Inc. SPDX-License-Identifier: Apache-2.0 diff --git a/src/conformance/conformance_test/conformance_test.cpp b/src/conformance/conformance_test/conformance_test.cpp index ea36f082..0e303f83 100644 --- a/src/conformance/conformance_test/conformance_test.cpp +++ b/src/conformance/conformance_test/conformance_test.cpp @@ -16,43 +16,34 @@ #include "openxr/openxr_platform_defines.h" #define CATCH_CONFIG_NOSTDOUT -#ifdef XR_USE_PLATFORM_ANDROID -#define CATCH_CONFIG_NO_CPP11_TO_STRING -#define CATCH_CONFIG_FALLBACK_STRINGIFIER -#endif // XR_USE_PLATFORM_ANDROID + +#include "conformance_framework.h" +#include "conformance_test.h" +#include "conformance_utils.h" +#include "environment.h" +#include "graphics_plugin.h" +#include "platform_utils.hpp" // for OPENXR_API_LAYER_PATH_ENV_VAR +#include "report.h" +#include "utilities/utils.h" + +#include #include "catch_reporter_cts.h" -#include -#include #include +#include +#include +#include // for customizing arg parsing #include #include -#include - -#include "throw_helpers.h" -#include -#include -#include #include "xr_dependencies.h" #include #include -#include #include -#include -#include - -#include "throw_helpers.h" -#include "environment.h" -#include "conformance_test.h" -#include "report.h" -#include "utils.h" -#include "platform_utils.hpp" // for OPENXR_API_LAYER_PATH_ENV_VAR -#include "filesystem_utils.hpp" -#include "two_call_util.h" -#include "conformance_utils.h" +#include +#include using namespace Conformance; @@ -336,6 +327,12 @@ namespace ("Disables logging file/line data.") .optional() + | Opt(options.pollGetSystem) // poll xrGetSystem at startup + ["--pollGetSystem"] // + ("Retry xrGetSystem until success or timeout expires before running tests.") + .optional() + + // | Opt([&](bool enabled) { options.debugMode = enabled; }) // ["-D"]["--debugMode"] // ("Sets debug mode as enabled or disabled.") diff --git a/src/conformance/conformance_test/conformance_test.h b/src/conformance/conformance_test/conformance_test.h index ecbbbce2..7e2adfeb 100644 --- a/src/conformance/conformance_test/conformance_test.h +++ b/src/conformance/conformance_test/conformance_test.h @@ -16,7 +16,7 @@ #pragma once -#include +#include #include enum MessageType diff --git a/src/conformance/conformance_test/test_ActionPoses.cpp b/src/conformance/conformance_test/test_ActionPoses.cpp index c9c92647..66e1bbd9 100644 --- a/src/conformance/conformance_test/test_ActionPoses.cpp +++ b/src/conformance/conformance_test/test_ActionPoses.cpp @@ -14,18 +14,26 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include "utils.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include "throw_helpers.h" +#include "RGBAImage.h" +#include "common/xr_linear.h" #include "composition_utils.h" -#include +#include "conformance_framework.h" +#include "graphics_plugin.h" +#include "utilities/stringification.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" + #include -#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include using namespace Conformance; diff --git a/src/conformance/conformance_test/test_FrameSubmission.cpp b/src/conformance/conformance_test/test_FrameSubmission.cpp index dc225258..1a343852 100644 --- a/src/conformance/conformance_test/test_FrameSubmission.cpp +++ b/src/conformance/conformance_test/test_FrameSubmission.cpp @@ -14,20 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include -#include -#include "utils.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include "throw_helpers.h" #include "composition_utils.h" +#include "conformance_framework.h" +#include "conformance_utils.h" #include "report.h" +#include "utilities/throw_helpers.h" + +#include #include #include -#include + +#include +#include +#include +#include #define ENUM_LIST(name, val) name, constexpr XrEnvironmentBlendMode SupportedBlendModes[] = {XR_LIST_ENUM_XrEnvironmentBlendMode(ENUM_LIST)}; diff --git a/src/conformance/conformance_test/test_HapticInterrupt.cpp b/src/conformance/conformance_test/test_HapticInterrupt.cpp index dd351d34..89422738 100644 --- a/src/conformance/conformance_test/test_HapticInterrupt.cpp +++ b/src/conformance/conformance_test/test_HapticInterrupt.cpp @@ -14,17 +14,24 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include "utils.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" +#include "RGBAImage.h" +#include "common/xr_linear.h" #include "composition_utils.h" -#include +#include "conformance_framework.h" +#include "graphics_plugin.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" + #include -#include +#include + +#include +#include +#include +#include +#include +#include +#include using namespace Conformance; diff --git a/src/conformance/conformance_test/test_InteractiveThrow.cpp b/src/conformance/conformance_test/test_InteractiveThrow.cpp index 5c9fe67b..c3fdd4f8 100644 --- a/src/conformance/conformance_test/test_InteractiveThrow.cpp +++ b/src/conformance/conformance_test/test_InteractiveThrow.cpp @@ -14,18 +14,16 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include "utils.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include "throw_helpers.h" +#include "common/xr_linear.h" #include "composition_utils.h" +#include "conformance_framework.h" +#include "utilities/throw_helpers.h" +#include "utilities/utils.h" + #include #include -#include + +#include using namespace Conformance; diff --git a/src/conformance/conformance_test/test_LayerComposition.cpp b/src/conformance/conformance_test/test_LayerComposition.cpp index 12b5dc51..d3cf88c8 100644 --- a/src/conformance/conformance_test/test_LayerComposition.cpp +++ b/src/conformance/conformance_test/test_LayerComposition.cpp @@ -14,19 +14,18 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include -#include "utils.h" -#include "throw_helpers.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" +#include "common/xr_linear.h" #include "composition_utils.h" +#include "conformance_framework.h" +#include "utilities/throw_helpers.h" +#include "utilities/utils.h" + #include #include -#include + +#include +#include +#include using namespace Conformance; diff --git a/src/conformance/conformance_test/test_SessionState.cpp b/src/conformance/conformance_test/test_SessionState.cpp index 4714d5cb..f59a14f6 100644 --- a/src/conformance/conformance_test/test_SessionState.cpp +++ b/src/conformance/conformance_test/test_SessionState.cpp @@ -14,21 +14,24 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" #include "matchers.h" -#include -#include -#include -#include -#include -#include + #include #include #include +#include +#include +#include +#include +#include +#include +#include + #define AS_LIST(name, val) name, constexpr XrViewConfigurationType KnownViewTypes[] = {XR_LIST_ENUM_XrViewConfigurationType(AS_LIST)}; #undef AS_LIST diff --git a/src/conformance/conformance_test/test_Swapchains.cpp b/src/conformance/conformance_test/test_Swapchains.cpp index 076b6db8..6ce6484b 100644 --- a/src/conformance/conformance_test/test_Swapchains.cpp +++ b/src/conformance/conformance_test/test_Swapchains.cpp @@ -14,16 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "bitmask_to_string.h" #include "conformance_framework.h" #include "conformance_utils.h" #include "graphics_plugin.h" #include "matchers.h" #include "report.h" #include "swapchain_image_data.h" -#include "swapchain_parameters.h" -#include "types_and_constants.h" -#include "utils.h" +#include "utilities/bitmask_to_string.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" #include #include @@ -73,7 +74,7 @@ namespace Conformance } { INFO("Rendering to the swapchain(s)"); - const XrSwapchainImageBaseHeader* image = swapchainImages->GetGenericColorImage(i); + const XrSwapchainImageBaseHeader* image = swapchainImages->GetGenericColorImage(colorImageIndex); GetGlobalData().graphicsPlugin->ClearImageSlice(image, 0); GetGlobalData().graphicsPlugin->RenderView(projectionView, image, {}); } @@ -206,30 +207,6 @@ namespace Conformance INFO("In the case of XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT we must only allow a single acquire"); CHECK(xrAcquireSwapchainImage(swapchain, &imageAcquireInfo, &index) == XR_ERROR_CALL_ORDER_INVALID); } - else { - // Real apps will acquire images more than once (though they will probably call xrEndFrame too) - // and this tends to trigger annoying-to-fix Vulkan validation errors in the runtime. - INFO("Acquire, wait, then release all the images in turn a second time"); - for (uint32_t i = 0; i < imageCount; ++i) { - CAPTURE(i); - REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrAcquireSwapchainImage(swapchain, nullptr, &indexVector[i])); - } - // Wait/release all the images. - for (uint32_t i = 0; i < imageCount; ++i) { - REQUIRE(GetGlobalData().graphicsPlugin->ValidateSwapchainImageState(swapchain, indexVector[i], imageFormat)); - XrSwapchainImageWaitInfo imageWaitInfo{XR_TYPE_SWAPCHAIN_IMAGE_WAIT_INFO}; - imageWaitInfo.timeout = 500_xrMilliseconds; // Call can block waiting for image to become available for writing. - REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrWaitSwapchainImage(swapchain, &imageWaitInfo)); - - // Another wait should fail with XR_ERROR_CALL_ORDER_INVALID. - CHECK(xrWaitSwapchainImage(swapchain, &imageWaitInfo) == XR_ERROR_CALL_ORDER_INVALID); - - // For odd values of i we exercise that runtimes must accept NULL image release info. - XrSwapchainImageReleaseInfo imageReleaseInfo{XR_TYPE_SWAPCHAIN_IMAGE_RELEASE_INFO}; - const XrSwapchainImageReleaseInfo* imageReleaseInfoToUse = ((i % 2) ? &imageReleaseInfo : nullptr); - REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrReleaseSwapchainImage(swapchain, imageReleaseInfoToUse)); - } - } // To do: Is there a way to exercise xrWaitSwapchainImage XR_TIMEOUT_EXPIRED? It seems that the only // way this can happen is if the runtime is busy with an image despite successfully acquiring it. @@ -626,4 +603,72 @@ namespace Conformance } } } + + TEST_CASE("SwapchainsAcquire", "") + { + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsUsingGraphicsPlugin()) { + // Nothing to check - no graphics plugin means no swapchain + return; + } + + // Set up the session we will use for the testing + AutoBasicSession session(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | + AutoBasicSession::createSwapchains); + + auto graphicsPlugin = globalData.GetGraphicsPlugin(); + + // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. + auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); + CAPTURE(timeout); + + FrameIterator frameIterator(&session); + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + XrSwapchain swapchain{XR_NULL_HANDLE}; + XrExtent2Di widthHeight{0, 0}; // 0,0 means Use defaults. + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(CreateColorSwapchain(session, graphicsPlugin.get(), &swapchain, &widthHeight)); + + uint32_t imageCount = 0; + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrEnumerateSwapchainImages(swapchain, 0, &imageCount, nullptr)); + REQUIRE(imageCount > 0); + std::vector indexVector(imageCount, UINT32_MAX); + + // + { + for (int j = 0; j < 10; ++j) { + { + // Real apps may acquire images more than once (though they will probably call xrEndFrame too) + // and this tends to trigger annoying-to-fix Vulkan validation errors in the runtime. + INFO("Acquire, wait, then release all the images in turn a second time"); + for (uint32_t i = 0; i < imageCount; ++i) { + CAPTURE(i); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrAcquireSwapchainImage(swapchain, nullptr, &indexVector[i])); + } + // Wait/release all the images. + for (uint32_t i = 0; i < imageCount; ++i) { + XrSwapchainImageWaitInfo imageWaitInfo{XR_TYPE_SWAPCHAIN_IMAGE_WAIT_INFO}; + imageWaitInfo.timeout = 500_xrMilliseconds; // Call can block waiting for image to become available for writing. + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrWaitSwapchainImage(swapchain, &imageWaitInfo)); + + // Another wait should fail with XR_ERROR_CALL_ORDER_INVALID. + CHECK(xrWaitSwapchainImage(swapchain, &imageWaitInfo) == XR_ERROR_CALL_ORDER_INVALID); + + // For odd values of i we exercise that runtimes must accept NULL image release info. + XrSwapchainImageReleaseInfo imageReleaseInfo{XR_TYPE_SWAPCHAIN_IMAGE_RELEASE_INFO}; + const XrSwapchainImageReleaseInfo* imageReleaseInfoToUse = ((i % 2) ? &imageReleaseInfo : nullptr); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrReleaseSwapchainImage(swapchain, imageReleaseInfoToUse)); + } + } + + if ((j % 2) == 0) { + runResult = frameIterator.SubmitFrame(); + REQUIRE(runResult == FrameIterator::RunResult::Success); + } + } + } + + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrDestroySwapchain(swapchain)); + } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_ViewConfigurations.cpp b/src/conformance/conformance_test/test_ViewConfigurations.cpp index 87d1dbc1..e269a587 100644 --- a/src/conformance/conformance_test/test_ViewConfigurations.cpp +++ b/src/conformance/conformance_test/test_ViewConfigurations.cpp @@ -14,17 +14,18 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" #include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include + #include #include +#include +#include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp index cec34737..8145d8a1 100644 --- a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp @@ -14,19 +14,19 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" + +#include +#include + #include -#include #include #include -#include #include #include -#include -#include namespace Conformance { @@ -147,9 +147,9 @@ namespace Conformance SECTION("xrCreateInstance debug utils not enabled") { - StringVec enabledApiLayers = globalData.enabledAPILayerNames; + auto enabledApiLayers = StringVec(globalData.enabledAPILayerNames); // Enable only the required platform extensions by default - StringVec enabledExtensions = globalData.requiredPlatformInstanceExtensions; + auto enabledExtensions = StringVec(globalData.requiredPlatformInstanceExtensions); XrInstance instance{XR_NULL_HANDLE}; CleanupInstanceOnScopeExit cleanup(instance); @@ -181,9 +181,9 @@ namespace Conformance // Note that this behavior will be implicitly validated by AutoBasicInstance when skipDebugMessenger // is not passed as an option, but we have an explicit test for this behavior too. - StringVec enabledApiLayers = globalData.enabledAPILayerNames; + auto enabledApiLayers = StringVec(globalData.enabledAPILayerNames); // Enable only the required platform extensions by default - StringVec enabledExtensions = globalData.requiredPlatformInstanceExtensions; + auto enabledExtensions = StringVec(globalData.requiredPlatformInstanceExtensions); XrDebugUtilsMessengerCreateInfoEXT debugInfo{XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; debugInfo.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT | diff --git a/src/conformance/conformance_test/test_XR_EXT_dpad_binding.cpp b/src/conformance/conformance_test/test_XR_EXT_dpad_binding.cpp index 40fc4d0f..91abefa2 100644 --- a/src/conformance/conformance_test/test_XR_EXT_dpad_binding.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_dpad_binding.cpp @@ -14,18 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. +#include "common/xr_linear.h" +#include "composition_utils.h" #include "conformance_framework.h" -#include "utils.h" #include "conformance_utils.h" -#include "composition_utils.h" -#include "throw_helpers.h" #include "input_testinputdevice.h" #include "report.h" -#include +#include "utilities/event_reader.h" +#include "utilities/throw_helpers.h" + #include -#include #include -#include + +#include +#include namespace Conformance { @@ -1171,9 +1173,7 @@ namespace Conformance InitInteractiveInteractionProfiles(interactionProfiles, eControllerComponent::Thumbstick); if (interactionProfiles.size() == 0) { - //! @todo Change to SKIP once catch2 has been updated https://github.com/KhronosGroup/SYCL-CTS/pull/469 - ReportF("Enabled interaction profile(s) doesn't have thumbstick skipping test"); - return; + SKIP("Enabled interaction profile(s) has no thumbstick, skipping test"); } // Setup ActionSet and Actions. @@ -1252,9 +1252,7 @@ namespace Conformance InitInteractiveInteractionProfiles(interactionProfiles, eControllerComponent::Trackpad); if (interactionProfiles.size() == 0) { - //! @todo Change to SKIP once catch2 has been updated https://github.com/KhronosGroup/SYCL-CTS/pull/469 - ReportF("Enabled interaction profile(s) doesn't have trackpad skipping test"); - return; + SKIP("Enabled interaction profile(s) has no thumbstick, skipping test"); } // Setup ActionSet and Actions. diff --git a/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp b/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp index a0065b37..e62710a1 100644 --- a/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp @@ -19,8 +19,9 @@ #include "conformance_framework.h" #include "composition_utils.h" #include "two_call.h" -#include "utils.h" -#include "xr_linear.h" +#include "utilities/utils.h" +#include "utilities/system_properties_helper.h" +#include "common/xr_linear.h" #include using namespace Conformance; @@ -38,15 +39,9 @@ namespace Conformance static constexpr XrVector3f kVectorUp{0, 1, 0}; static constexpr XrVector3f kVectorForward{0, 0, -1}; - static bool SystemSupportsEyeGazeInteraction(XrInstance instance, XrSystemId systemId) - { - XrSystemEyeGazeInteractionPropertiesEXT eyeGazeSystemProperties{XR_TYPE_SYSTEM_EYE_GAZE_INTERACTION_PROPERTIES_EXT}; - XrSystemProperties systemProperties{XR_TYPE_SYSTEM_PROPERTIES, &eyeGazeSystemProperties}; - - REQUIRE(xrGetSystemProperties(instance, systemId, &systemProperties) == XR_SUCCESS); - - return eyeGazeSystemProperties.supportsEyeGazeInteraction != XR_FALSE; - } + static const auto SystemSupportsEyeGazeInteraction = + MakeSystemPropertiesBoolChecker(XrSystemEyeGazeInteractionPropertiesEXT{XR_TYPE_SYSTEM_EYE_GAZE_INTERACTION_PROPERTIES_EXT}, + &XrSystemEyeGazeInteractionPropertiesEXT::supportsEyeGazeInteraction); TEST_CASE("XR_EXT_eye_gaze_interaction", "[XR_EXT_eye_gaze_interaction][interactive][no_auto]") { diff --git a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp index ae9b6af1..ad2e8e72 100644 --- a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp @@ -15,12 +15,13 @@ // limitations under the License. #include "conformance_framework.h" -#include "utils.h" -#include "conformance_utils.h" #include "composition_utils.h" +#include "conformance_utils.h" +#include "utilities/system_properties_helper.h" +#include "utilities/utils.h" #include #include -#include +#include "common/xr_linear.h" using namespace Conformance; @@ -31,15 +32,9 @@ namespace Conformance constexpr int RIGHT_HAND = 1; constexpr int HAND_COUNT = 2; - static bool SystemSupportsHandTracking(XrInstance instance, XrSystemId systemId) - { - XrSystemHandTrackingPropertiesEXT handTrackingSystemProperties{XR_TYPE_SYSTEM_HAND_TRACKING_PROPERTIES_EXT}; - XrSystemProperties systemProperties{XR_TYPE_SYSTEM_PROPERTIES}; - systemProperties.next = &handTrackingSystemProperties; - REQUIRE(xrGetSystemProperties(instance, systemId, &systemProperties) == XR_SUCCESS); - - return handTrackingSystemProperties.supportsHandTracking == XR_TRUE; - } + static const auto SystemSupportsHandTracking = + MakeSystemPropertiesBoolChecker(XrSystemHandTrackingPropertiesEXT{XR_TYPE_SYSTEM_HAND_TRACKING_PROPERTIES_EXT}, + &XrSystemHandTrackingPropertiesEXT::supportsHandTracking); TEST_CASE("XR_EXT_hand_tracking", "") { @@ -69,22 +64,215 @@ namespace Conformance auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); + XrSystemId systemId = instance.systemId; + + AutoBasicSession session(AutoBasicSession::beginSession, instance); + + if (!SystemSupportsHandTracking(instance, systemId)) { + // https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#_create_a_hand_tracker_handle + // If the system does not support hand tracking, runtime must return XR_ERROR_FEATURE_UNSUPPORTED from xrCreateHandTrackerEXT. + // In this case, the runtime must return XR_FALSE for supportsHandTracking in XrSystemHandTrackingPropertiesEXT when the + // function xrGetSystemProperties is called, so that the application can avoid creating a hand tracker. + + for (size_t i = 0; i < HAND_COUNT; ++i) { + XrHandTrackerEXT tracker{XR_NULL_HANDLE}; + XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; + createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; + createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); + REQUIRE(XR_ERROR_FEATURE_UNSUPPORTED == xrCreateHandTrackerEXT(session, &createInfo, &tracker)); + } + + // This runtime does support hand tracking, but this headset does not + // support hand tracking, which is fine. + SKIP("System does not support hand tracking"); + } + + std::array handTracker; + for (size_t i = 0; i < HAND_COUNT; ++i) { + XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; + createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; + createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); + REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); + REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); + } + } + + SECTION("Query joint locations") + { + AutoBasicInstance instance({XR_EXT_HAND_TRACKING_EXTENSION_NAME}, AutoBasicInstance::createSystemId); + + auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); + auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); + auto xrLocateHandJointsEXT = GetInstanceExtensionFunction(instance, "xrLocateHandJointsEXT"); + + XrSystemId systemId = instance.systemId; + if (!SystemSupportsHandTracking(instance, systemId)) { + // This runtime does support hand tracking, but this headset does not + // support hand tracking, which is fine. + SKIP("System does not support hand tracking"); + } + + AutoBasicSession session(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | + AutoBasicSession::createSwapchains, + instance); + + XrSpace localSpace = XR_NULL_HANDLE; + + XrReferenceSpaceCreateInfo localSpaceCreateInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; + localSpaceCreateInfo.poseInReferenceSpace = XrPosefCPP(); + localSpaceCreateInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL; + REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); + + std::array handTracker; + for (size_t i = 0; i < HAND_COUNT; ++i) { + XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; + createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; + createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); + REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); + } + + // Wait until the runtime is ready for us to begin a session + auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); + FrameIterator frameIterator(&session); + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + for (auto hand : {LEFT_HAND, RIGHT_HAND}) { + std::array, HAND_COUNT> jointLocations; + std::array, HAND_COUNT> jointVelocities; + + XrHandJointVelocitiesEXT velocities{XR_TYPE_HAND_JOINT_VELOCITIES_EXT}; + velocities.jointCount = XR_HAND_JOINT_COUNT_EXT; + velocities.jointVelocities = jointVelocities[hand].data(); + + XrHandJointLocationsEXT locations{XR_TYPE_HAND_JOINT_LOCATIONS_EXT}; + locations.next = &velocities; + locations.jointCount = XR_HAND_JOINT_COUNT_EXT; + locations.jointLocations = jointLocations[hand].data(); + + XrHandJointsLocateInfoEXT locateInfo{XR_TYPE_HAND_JOINTS_LOCATE_INFO_EXT}; + locateInfo.baseSpace = localSpace; + locateInfo.time = frameIterator.frameState.predictedDisplayTime; + REQUIRE(XR_SUCCESS == xrLocateHandJointsEXT(handTracker[hand], &locateInfo, &locations)); + + // https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#_locate_hand_joints + for (size_t i = 0; i < jointLocations[hand].size(); ++i) { + if (locations.isActive == XR_TRUE) { + // If the returned isActive is true, the runtime must return all joint locations with both + // XR_SPACE_LOCATION_POSITION_VALID_BIT and XR_SPACE_LOCATION_ORIENTATION_VALID_BIT set. + REQUIRE((jointLocations[hand][i].locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT) != 0); + REQUIRE((jointLocations[hand][i].locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0); + + // If the returned locationFlags has XR_SPACE_LOCATION_POSITION_VALID_BIT set, + // the returned radius must be a positive value. + REQUIRE(jointLocations[hand][i].radius > 0); + + // If an XrHandJointVelocitiesEXT structure is chained to XrHandJointLocationsEXT::next, + // the returned XrHandJointLocationsEXT::isActive is true, and the velocity is observed + // or can be calculated by the runtime, the runtime must fill in the linear velocity of + // each hand joint within the reference frame of baseSpace and set the + // XR_SPACE_VELOCITY_LINEAR_VALID_BIT. + // Similarly, if an XrHandJointVelocitiesEXT structure is chained to + // XrHandJointLocationsEXT::next, the returned XrHandJointLocationsEXT::isActive is true, + // and the angular velocity is observed or can be calculated by the runtime, the runtime + // must fill in the angular velocity of each joint within the reference frame of baseSpace + // and set the XR_SPACE_VELOCITY_ANGULAR_VALID_BIT. + REQUIRE((jointVelocities[hand][i].velocityFlags & XR_SPACE_VELOCITY_LINEAR_VALID_BIT) != 0); + REQUIRE((jointVelocities[hand][i].velocityFlags & XR_SPACE_VELOCITY_ANGULAR_VALID_BIT) != 0); + } + else { + // If the returned isActive is false, it indicates the hand tracker did not detect the hand + // input or the application lost input focus. In this case, the runtime must return all + // jointLocations with neither XR_SPACE_LOCATION_POSITION_VALID_BIT nor + // XR_SPACE_LOCATION_ORIENTATION_VALID_BIT set. + REQUIRE((jointLocations[hand][i].locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT) == 0); + REQUIRE((jointLocations[hand][i].locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) == 0); + } + } + } + + for (size_t i = 0; i < HAND_COUNT; ++i) { + REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); + } + } + + SECTION("Query invalid joint sets") + { + AutoBasicInstance instance({XR_EXT_HAND_TRACKING_EXTENSION_NAME}, AutoBasicInstance::createSystemId); + + auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); + auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); + auto xrLocateHandJointsEXT = GetInstanceExtensionFunction(instance, "xrLocateHandJointsEXT"); + XrSystemId systemId = instance.systemId; if (!SystemSupportsHandTracking(instance, systemId)) { // This runtime does support hand tracking, but this headset does not // support hand tracking, which is fine. SKIP("System does not support hand tracking"); - return; } AutoBasicSession session(AutoBasicSession::beginSession, instance); - XrHandTrackerEXT handTracker[HAND_COUNT]; + XrSpace localSpace = XR_NULL_HANDLE; + + XrReferenceSpaceCreateInfo localSpaceCreateInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; + localSpaceCreateInfo.poseInReferenceSpace = XrPosefCPP(); + localSpaceCreateInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL; + REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); + + std::array handTracker; for (size_t i = 0; i < HAND_COUNT; ++i) { XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); + } + + // Wait until the runtime is ready for us to begin a session + auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); + FrameIterator frameIterator(&session); + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + // The application must input jointCount as described by the XrHandJointSetEXT when creating the XrHandTrackerEXT. + // Otherwise, the runtime must return XR_ERROR_VALIDATION_FAILURE. + static const uint32_t INVALID_JOINT_COUNT = XR_HAND_JOINT_COUNT_EXT - 1; + + // Firstly test without joint velocities. + for (auto hand : {LEFT_HAND, RIGHT_HAND}) { + std::array, HAND_COUNT> jointLocations; + + XrHandJointLocationsEXT locations{XR_TYPE_HAND_JOINT_LOCATIONS_EXT}; + locations.jointCount = INVALID_JOINT_COUNT; + locations.jointLocations = jointLocations[hand].data(); + + XrHandJointsLocateInfoEXT locateInfo{XR_TYPE_HAND_JOINTS_LOCATE_INFO_EXT}; + locateInfo.baseSpace = localSpace; + locateInfo.time = frameIterator.frameState.predictedDisplayTime; + REQUIRE(XR_ERROR_VALIDATION_FAILURE == xrLocateHandJointsEXT(handTracker[hand], &locateInfo, &locations)); + } + + // Same test again but this time with invalid joint velocity count + for (auto hand : {LEFT_HAND, RIGHT_HAND}) { + std::array, HAND_COUNT> jointLocations; + std::array, HAND_COUNT> jointVelocities; + + XrHandJointVelocitiesEXT velocities{XR_TYPE_HAND_JOINT_VELOCITIES_EXT}; + velocities.jointCount = INVALID_JOINT_COUNT; + velocities.jointVelocities = jointVelocities[hand].data(); + + XrHandJointLocationsEXT locations{XR_TYPE_HAND_JOINT_LOCATIONS_EXT}; + locations.next = &velocities; + locations.jointCount = XR_HAND_JOINT_COUNT_EXT; + locations.jointLocations = jointLocations[hand].data(); + + XrHandJointsLocateInfoEXT locateInfo{XR_TYPE_HAND_JOINTS_LOCATE_INFO_EXT}; + locateInfo.baseSpace = localSpace; + locateInfo.time = frameIterator.frameState.predictedDisplayTime; + REQUIRE(XR_ERROR_VALIDATION_FAILURE == xrLocateHandJointsEXT(handTracker[hand], &locateInfo, &locations)); + } + + for (size_t i = 0; i < HAND_COUNT; ++i) { REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); } } diff --git a/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp b/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp index a2b52bf4..0207a3cb 100644 --- a/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2019-2022, The Khronos Group Inc. +// Copyright (c) 2019-2023, The Khronos Group Inc. // // SPDX-License-Identifier: Apache-2.0 // @@ -14,12 +14,14 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" +#include "two_call.h" #include #include +#include #include -#include "two_call.h" +#include using namespace Conformance; @@ -51,8 +53,7 @@ namespace Conformance AutoBasicSession session(AutoBasicSession::OptionFlags::createSession, instance); std::vector refSpaceTypes = CHECK_TWO_CALL(XrReferenceSpaceType, {}, xrEnumerateReferenceSpaces, session); - - REQUIRE(std::find(refSpaceTypes.begin(), refSpaceTypes.end(), XR_REFERENCE_SPACE_TYPE_LOCAL_FLOOR_EXT) != refSpaceTypes.end()); + REQUIRE_THAT(refSpaceTypes, Catch::Matchers::Contains(XR_REFERENCE_SPACE_TYPE_LOCAL_FLOOR_EXT)); XrReferenceSpaceCreateInfo localFloorCreateInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; localFloorCreateInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL_FLOOR_EXT; diff --git a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp index c1bae987..5391e7fd 100644 --- a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2019-2021, The Khronos Group Inc. +// Copyright (c) 2019-2023, The Khronos Group Inc. // // SPDX-License-Identifier: Apache-2.0 // @@ -14,18 +14,27 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include "utils.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include "throw_helpers.h" +#include "RGBAImage.h" +#include "common/xr_linear.h" #include "composition_utils.h" +#include "conformance_framework.h" +#include "graphics_plugin.h" +#include "utilities/stringification.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include #include -#include + +#include +#include +#include +#include +#include +#include +#include +#include using namespace Conformance; @@ -315,7 +324,7 @@ namespace Conformance auto update = [&](const XrFrameState& frameState) { std::vector renderedCubes; - const std::array activeActionSets = {{actionSet, XR_NULL_PATH}}; + const std::array activeActionSets = {{{actionSet, XR_NULL_PATH}}}; XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; syncInfo.activeActionSets = activeActionSets.data(); syncInfo.countActiveActionSets = (uint32_t)activeActionSets.size(); diff --git a/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp b/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp index d9c9740e..ee44c0dc 100644 --- a/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp @@ -14,16 +14,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include -#include -#include -#include -#include #include -#include namespace Conformance { diff --git a/src/conformance/conformance_test/test_XR_EXT_plane_detection.cpp b/src/conformance/conformance_test/test_XR_EXT_plane_detection.cpp new file mode 100644 index 00000000..ceb03f55 --- /dev/null +++ b/src/conformance/conformance_test/test_XR_EXT_plane_detection.cpp @@ -0,0 +1,860 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include "utilities/utils.h" +#include "conformance_utils.h" +#include "composition_utils.h" +#include "mesh_projection_layer.h" + +#include +#include +#include +#include + +#include + +using namespace Conformance; + +namespace mapbox +{ + namespace util + { + + template <> + struct nth<0, XrVector2f> + { + inline static auto get(const XrVector2f& t) + { + return t.x; + }; + }; + template <> + struct nth<1, XrVector2f> + { + inline static auto get(const XrVector2f& t) + { + return t.y; + }; + }; + + } // namespace util +} // namespace mapbox + +namespace Conformance +{ + static constexpr XrVector3f Up{0, 1, 0}; + static XrPlaneDetectionCapabilityFlagsEXT SystemPlaneDetectionCapabilities(XrInstance instance, XrSystemId systemId) + { + XrSystemPlaneDetectionPropertiesEXT planeDetectionSystemProperties{XR_TYPE_SYSTEM_PLANE_DETECTION_PROPERTIES_EXT}; + XrSystemProperties systemProperties{XR_TYPE_SYSTEM_PROPERTIES}; + systemProperties.next = &planeDetectionSystemProperties; + + REQUIRE(XR_SUCCESS == xrGetSystemProperties(instance, systemId, &systemProperties)); + return planeDetectionSystemProperties.supportedFeatures; + } + + static bool SystemSupportsEXTPlaneDetection(XrInstance instance, XrSystemId systemId) + { + return SystemPlaneDetectionCapabilities(instance, systemId) & XR_PLANE_DETECTION_CAPABILITY_PLANE_DETECTION_BIT_EXT; + } + + TEST_CASE("XR_EXT_plane_detection", "[XR_EXT_plane_detection]") + { + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_PLANE_DETECTION_EXTENSION_NAME)) { + // Runtime does not support extension - it should not be possible to get function pointers. + AutoBasicInstance instance; + ValidateInstanceExtensionFunctionNotSupported(instance, "xrCreatePlaneDetectorEXT"); + return; + } + + SECTION("Extension not enabled") + { + if (!globalData.IsInstanceExtensionEnabled(XR_EXT_PLANE_DETECTION_EXTENSION_NAME)) { + AutoBasicInstance instance; + ValidateInstanceExtensionFunctionNotSupported(instance, "xrCreatePlaneDetectorEXT"); + } + else { + WARN(XR_EXT_PLANE_DETECTION_EXTENSION_NAME " force-enabled, cannot test behavior when extension is disabled."); + } + } + + SECTION("Create and Destroy") + { + AutoBasicInstance instance({XR_EXT_PLANE_DETECTION_EXTENSION_NAME}, AutoBasicInstance::createSystemId); + XrSystemId systemId = instance.systemId; + auto xrCreatePlaneDetectorEXT = + GetInstanceExtensionFunction(instance, "xrCreatePlaneDetectorEXT"); + auto xrDestroyPlaneDetectorEXT = + GetInstanceExtensionFunction(instance, "xrDestroyPlaneDetectorEXT"); + + if (!SystemSupportsEXTPlaneDetection(instance, systemId)) { + // This runtime does support plane detection tracking, but this system does not, that is fine. + SKIP("System does not support plane detection"); + } + + AutoBasicSession session(AutoBasicSession::beginSession, instance); + + // pass not initialized structure + XrPlaneDetectorCreateInfoEXT createInfo{}; + XrPlaneDetectorEXT detection = XR_NULL_HANDLE; + REQUIRE(XR_ERROR_VALIDATION_FAILURE == xrCreatePlaneDetectorEXT(session, &createInfo, &detection)); + + createInfo.type = XR_TYPE_PLANE_DETECTOR_CREATE_INFO_EXT; + createInfo.flags = XR_PLANE_DETECTOR_ENABLE_CONTOUR_BIT_EXT; + + REQUIRE(XR_SUCCESS == xrCreatePlaneDetectorEXT(session, &createInfo, &detection)); + REQUIRE(XR_SUCCESS == xrDestroyPlaneDetectorEXT(detection)); + } + } + + static void RunPlaneTest(const std::vector& orientations, const char* instructions, + XrPlaneDetectorSemanticTypeEXT autoCompleteSemanticType = XR_PLANE_DETECTOR_SEMANTIC_TYPE_UNDEFINED_EXT, + bool forceOrientationNullptr = false) + { + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_PLANE_DETECTION_EXTENSION_NAME)) { + SKIP(); + } + + if (!globalData.IsUsingGraphicsPlugin()) { + SKIP(); + } + + CompositionHelper compositionHelper("XR_EXT_plane_detection", {XR_EXT_PLANE_DETECTION_EXTENSION_NAME}); + XrPlaneDetectionCapabilityFlagsEXT flags = + SystemPlaneDetectionCapabilities(compositionHelper.GetInstance(), compositionHelper.GetSystemId()); + + if ((flags & XR_PLANE_DETECTION_CAPABILITY_PLANE_DETECTION_BIT_EXT) == 0) { + SKIP("System does not support plane detection"); + } + + switch (autoCompleteSemanticType) { + case XR_PLANE_DETECTOR_SEMANTIC_TYPE_UNDEFINED_EXT: + break; + case XR_PLANE_DETECTOR_SEMANTIC_TYPE_CEILING_EXT: + if ((flags & XR_PLANE_DETECTION_CAPABILITY_SEMANTIC_CEILING_BIT_EXT) == 0) { + // TODO convert to SKIP? compare ctsxml output in both cases + INFO("Semantic ceiling not supported"); + return; + } + break; + case XR_PLANE_DETECTOR_SEMANTIC_TYPE_FLOOR_EXT: + if ((flags & XR_PLANE_DETECTION_CAPABILITY_SEMANTIC_FLOOR_BIT_EXT) == 0) { + // TODO convert to SKIP? compare ctsxml output in both cases + INFO("Semantic floor not supported"); + return; + } + break; + case XR_PLANE_DETECTOR_SEMANTIC_TYPE_WALL_EXT: + if ((flags & XR_PLANE_DETECTION_CAPABILITY_SEMANTIC_WALL_BIT_EXT) == 0) { + // TODO convert to SKIP? compare ctsxml output in both cases + INFO("Semantic wall not supported"); + return; + } + break; + case XR_PLANE_DETECTOR_SEMANTIC_TYPE_PLATFORM_EXT: + if ((flags & XR_PLANE_DETECTION_CAPABILITY_SEMANTIC_PLATFORM_BIT_EXT) == 0) { + // TODO convert to SKIP? compare ctsxml output in both cases + INFO("Semantic platform not supported"); + return; + } + break; + default: + WARN("Unexpected Semantic Type requested"); + return; + } + + XrInstance instance = compositionHelper.GetInstance(); + + auto xrCreatePlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrCreatePlaneDetectorEXT"); + auto xrDestroyPlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrDestroyPlaneDetectorEXT"); + auto xrBeginPlaneDetectionEXT = GetInstanceExtensionFunction(instance, "xrBeginPlaneDetectionEXT"); + auto xrGetPlaneDetectionStateEXT = + GetInstanceExtensionFunction(instance, "xrGetPlaneDetectionStateEXT"); + auto xrGetPlaneDetectionsEXT = GetInstanceExtensionFunction(instance, "xrGetPlaneDetectionsEXT"); + + const XrSpace localSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_LOCAL, XrPosefCPP{}); + const XrSpace viewSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_VIEW, XrPosefCPP{}); + + // Set up composition projection layer and swapchains (one swapchain per view). + std::vector swapchains; + XrCompositionLayerProjection* const projLayer = compositionHelper.CreateProjectionLayer(localSpace); + { + const std::vector viewProperties = compositionHelper.EnumerateConfigurationViews(); + for (uint32_t j = 0; j < projLayer->viewCount; j++) { + const XrSwapchain swapchain = compositionHelper.CreateSwapchain(compositionHelper.DefaultColorSwapchainCreateInfo( + viewProperties[j].recommendedImageRectWidth, viewProperties[j].recommendedImageRectHeight)); + const_cast(projLayer->views[j].subImage) = compositionHelper.MakeDefaultSubImage(swapchain, 0); + swapchains.push_back(swapchain); + } + } + + const std::vector subactionPaths{StringToPath(compositionHelper.GetInstance(), "/user/hand/left"), + StringToPath(compositionHelper.GetInstance(), "/user/hand/right")}; + + XrActionSet actionSet; + XrAction completeAction; + { + XrActionSetCreateInfo actionSetInfo{XR_TYPE_ACTION_SET_CREATE_INFO}; + strcpy(actionSetInfo.actionSetName, "plane_detection_test"); + strcpy(actionSetInfo.localizedActionSetName, "Plane Detection Test"); + XRC_CHECK_THROW_XRCMD(xrCreateActionSet(compositionHelper.GetInstance(), &actionSetInfo, &actionSet)) + + XrActionCreateInfo actionInfo{XR_TYPE_ACTION_CREATE_INFO}; + actionInfo.actionType = XR_ACTION_TYPE_BOOLEAN_INPUT; + strcpy(actionInfo.actionName, "complete_test"); + strcpy(actionInfo.localizedActionName, "Complete test"); + XRC_CHECK_THROW_XRCMD(xrCreateAction(actionSet, &actionInfo, &completeAction)) + } + + const std::vector bindings = { + {completeAction, StringToPath(compositionHelper.GetInstance(), "/user/hand/left/input/select/click")}, + {completeAction, StringToPath(compositionHelper.GetInstance(), "/user/hand/right/input/select/click")}, + }; + + XrInteractionProfileSuggestedBinding suggestedBindings{XR_TYPE_INTERACTION_PROFILE_SUGGESTED_BINDING}; + suggestedBindings.interactionProfile = StringToPath(compositionHelper.GetInstance(), "/interaction_profiles/khr/simple_controller"); + suggestedBindings.suggestedBindings = bindings.data(); + suggestedBindings.countSuggestedBindings = (uint32_t)bindings.size(); + XRC_CHECK_THROW_XRCMD(xrSuggestInteractionProfileBindings(compositionHelper.GetInstance(), &suggestedBindings)) + + XrSessionActionSetsAttachInfo attachInfo{XR_TYPE_SESSION_ACTION_SETS_ATTACH_INFO}; + attachInfo.actionSets = &actionSet; + attachInfo.countActionSets = 1; + XRC_CHECK_THROW_XRCMD(xrAttachSessionActionSets(compositionHelper.GetSession(), &attachInfo)) + + compositionHelper.BeginSession(); + + XrPlaneDetectorCreateInfoEXT createInfo{XR_TYPE_PLANE_DETECTOR_CREATE_INFO_EXT}; + createInfo.flags = XR_PLANE_DETECTOR_ENABLE_CONTOUR_BIT_EXT; + XrPlaneDetectorEXT detection = XR_NULL_HANDLE; + REQUIRE(XR_SUCCESS == xrCreatePlaneDetectorEXT(compositionHelper.GetSession(), &createInfo, &detection)); + + // Create the instructional quad layer placed to the left. + XrCompositionLayerQuad* const instructionsQuad = + compositionHelper.CreateQuadLayer(compositionHelper.CreateStaticSwapchainImage(CreateTextImage(1024, 512, instructions, 48)), + localSpace, 1.0f, {{0, 0, 0, 1}, {-0.2f, 0, -1.0f}}); + XrQuaternionf_CreateFromAxisAngle(&instructionsQuad->pose.orientation, &Up, 10 * MATH_PI / 180); + + enum DetectState + { + IDLE, + WAITING, + PROCESSING + }; + DetectState detect_state = IDLE; + std::vector renderedCubes; + + auto update = [&](const XrFrameState& frameState) { + const std::array activeActionSets = {{{actionSet, XR_NULL_PATH}}}; + XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; + syncInfo.activeActionSets = activeActionSets.data(); + syncInfo.countActiveActionSets = (uint32_t)activeActionSets.size(); + XRC_CHECK_THROW_XRCMD(xrSyncActions(compositionHelper.GetSession(), &syncInfo)) + + // if an autoCompleteSemanticType is specified it will be used to complete the + // test. + if (autoCompleteSemanticType == XR_PLANE_DETECTOR_SEMANTIC_TYPE_UNDEFINED_EXT) { + XrActionStateGetInfo completeActionGetInfo{XR_TYPE_ACTION_STATE_GET_INFO}; + completeActionGetInfo.action = completeAction; + XrActionStateBoolean completeActionState{XR_TYPE_ACTION_STATE_BOOLEAN}; + XRC_CHECK_THROW_XRCMD(xrGetActionStateBoolean(compositionHelper.GetSession(), &completeActionGetInfo, &completeActionState)) + if (completeActionState.currentState == XR_TRUE && completeActionState.changedSinceLastSync) { + return false; + } + } + + switch (detect_state) { + case IDLE: { + XrPosef pose{}; + pose.position = XrVector3f{0.0f, 0.0f, 0.0f}; + pose.orientation = XrQuaternionf{0.0f, 0.0f, 0.0f, 1.0f}; + + XrPlaneDetectorBeginInfoEXT beginInfo{XR_TYPE_PLANE_DETECTOR_BEGIN_INFO_EXT}; + if (forceOrientationNullptr) { + beginInfo.orientationCount = 0; + beginInfo.orientations = nullptr; + } + else { + beginInfo.orientationCount = (uint32_t)orientations.size(); + beginInfo.orientations = orientations.data(); + } + beginInfo.minArea = 0.1f; + beginInfo.maxPlanes = 100; + beginInfo.boundingBoxPose = pose; + beginInfo.boundingBoxExtent = XrExtent3DfEXT{10.0f, 10.0f, 10.0f}; + beginInfo.time = frameState.predictedDisplayTime; + beginInfo.baseSpace = viewSpace; + REQUIRE(XR_SUCCESS == xrBeginPlaneDetectionEXT(detection, &beginInfo)); + detect_state = WAITING; + } break; + case WAITING: { + + // If GetPlaneDetectionStateEXT has not yet returned XR_PLANE_DETECTION_STATE_DONE_EXT + // calling xrGetPlaneDetectionsEXT must return XR_ERROR_CALL_ORDER_INVALID + XrPlaneDetectorGetInfoEXT getInfo{XR_TYPE_PLANE_DETECTOR_GET_INFO_EXT}; + getInfo.baseSpace = localSpace; + getInfo.time = frameState.predictedDisplayTime; + XrPlaneDetectorLocationsEXT locations{XR_TYPE_PLANE_DETECTOR_LOCATIONS_EXT}; + REQUIRE(XR_ERROR_CALL_ORDER_INVALID == xrGetPlaneDetectionsEXT(detection, &getInfo, &locations)); + + XrPlaneDetectionStateEXT state; + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionStateEXT(detection, &state)); + REQUIRE((state == XR_PLANE_DETECTION_STATE_PENDING_EXT || state == XR_PLANE_DETECTION_STATE_DONE_EXT || + state == XR_PLANE_DETECTION_STATE_ERROR_EXT)); + switch (state) { + case XR_PLANE_DETECTION_STATE_DONE_EXT: + detect_state = PROCESSING; + break; + case XR_PLANE_DETECTION_STATE_ERROR_EXT: + detect_state = IDLE; + break; + case XR_PLANE_DETECTION_STATE_PENDING_EXT: + break; + default: + break; + } + } break; + case PROCESSING: { + XrPlaneDetectorGetInfoEXT getInfo{XR_TYPE_PLANE_DETECTOR_GET_INFO_EXT}; + getInfo.baseSpace = localSpace; + getInfo.time = frameState.predictedDisplayTime; + XrPlaneDetectorLocationsEXT locations{XR_TYPE_PLANE_DETECTOR_LOCATIONS_EXT}; + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionsEXT(detection, &getInfo, &locations)); + + if (locations.planeLocationCountOutput == 0) { + break; + } + + renderedCubes.clear(); + std::vector location_vector; + location_vector.resize(locations.planeLocationCountOutput); + for (auto& location : location_vector) { + location.type = XR_TYPE_PLANE_DETECTOR_LOCATION_EXT; + location.next = nullptr; + } + locations.planeLocations = location_vector.data(); + locations.planeLocationCapacityInput = (uint32_t)location_vector.size(); + + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionsEXT(detection, &getInfo, &locations)); + for (auto& location : location_vector) { + + if (autoCompleteSemanticType != XR_PLANE_DETECTOR_SEMANTIC_TYPE_UNDEFINED_EXT) { + if (location.semanticType == autoCompleteSemanticType) { + // DONE! + return false; + } + } + + renderedCubes.push_back(Cube{/* pose */ {location.pose.orientation, location.pose.position}, + /* scale: */ {location.extents.width, location.extents.height, 0.01f}}); + } + + detect_state = IDLE; + + } break; + } + + auto viewData = compositionHelper.LocateViews(localSpace, frameState.predictedDisplayTime); + const auto& viewState = std::get(viewData); + + std::vector layers; + if (viewState.viewStateFlags & XR_VIEW_STATE_POSITION_VALID_BIT && + viewState.viewStateFlags & XR_VIEW_STATE_ORIENTATION_VALID_BIT) { + const auto& views = std::get>(viewData); + + // Render into each viewport of the wide swapchain using the projection layer view fov and pose. + for (size_t view = 0; view < views.size(); view++) { + compositionHelper.AcquireWaitReleaseImage(swapchains[view], // + [&](const XrSwapchainImageBaseHeader* swapchainImage) { + GetGlobalData().graphicsPlugin->ClearImageSlice(swapchainImage); + const_cast(projLayer->views[view].fov) = views[view].fov; + const_cast(projLayer->views[view].pose) = views[view].pose; + GetGlobalData().graphicsPlugin->RenderView( + projLayer->views[view], swapchainImage, + RenderParams().Draw(renderedCubes)); + }); + } + + layers.push_back({reinterpret_cast(projLayer)}); + } + + layers.push_back({reinterpret_cast(instructionsQuad)}); + + compositionHelper.EndFrame(frameState.predictedDisplayTime, layers); + + return compositionHelper.PollEvents(); + }; + + RenderLoop(compositionHelper.GetSession(), update).Loop(); + + REQUIRE(XR_SUCCESS == xrDestroyPlaneDetectorEXT(detection)); + } + + TEST_CASE("XR_EXT_plane_detection-V", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_VERTICAL_EXT}, + "Planes should be rendered at the vertical surfaces, " + "the blue faces should face inward. " + "Press the select button on either controller to pass the test."); + } + + TEST_CASE("XR_EXT_plane_detection-HU", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_UPWARD_EXT}, + "Planes should be rendered at the horizontal surfaces with upward normals, " + "the blue faces should face upward (e.g. floors). " + "Press the select button on either controller to pass the test."); + } + + TEST_CASE("XR_EXT_plane_detection-HD", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_DOWNWARD_EXT}, + "Planes should be rendered at the horizontal surfaces with downward normals, " + "the blue faces should face downward (e.g. ceilings). " + "Press the select button on either controller to pass the test."); + } + + TEST_CASE("XR_EXT_plane_detection-A", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_ARBITRARY_EXT}, + "Planes should be rendered at the non horizontal/vertical surfaces. " + "Press the select button on either controller to pass the test."); + } + + TEST_CASE("XR_EXT_plane_detection-empty-list", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({}, + "All planes should be rendered. " + "Press the select button on either controller to pass the test."); + } + + TEST_CASE("XR_EXT_plane_detection-nullptr", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({}, + "All planes should be rendered. " + "Press the select button on either controller to pass the test.", + XR_PLANE_DETECTOR_SEMANTIC_TYPE_UNDEFINED_EXT, true); + } + + TEST_CASE("XR_EXT_plane_detection-ceiling", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_DOWNWARD_EXT}, "Make sure a ceiling is detected in the scene.", + XR_PLANE_DETECTOR_SEMANTIC_TYPE_CEILING_EXT); + } + + TEST_CASE("XR_EXT_plane_detection-floor", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_UPWARD_EXT}, "Make sure a floor is detected in the scene.", + XR_PLANE_DETECTOR_SEMANTIC_TYPE_FLOOR_EXT); + } + + TEST_CASE("XR_EXT_plane_detection-wall", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_VERTICAL_EXT}, "Make sure a wall is detected in the scene.", + XR_PLANE_DETECTOR_SEMANTIC_TYPE_WALL_EXT); + } + + TEST_CASE("XR_EXT_plane_detection-platform", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_UPWARD_EXT}, "Make sure a platform is detected in the scene.", + XR_PLANE_DETECTOR_SEMANTIC_TYPE_PLATFORM_EXT); + } + + TEST_CASE("XR_EXT_plane_detection-invalid-arguments", "[XR_EXT_plane_detection]") + { + // basic setup stuff + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_PLANE_DETECTION_EXTENSION_NAME)) { + SKIP(); + } + + CompositionHelper compositionHelper("XR_EXT_plane_detection", {XR_EXT_PLANE_DETECTION_EXTENSION_NAME}); + + if (!SystemSupportsEXTPlaneDetection(compositionHelper.GetInstance(), compositionHelper.GetSystemId())) { + SKIP("System does not support plane detection"); + } + + XrInstance instance = compositionHelper.GetInstance(); + + auto xrCreatePlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrCreatePlaneDetectorEXT"); + auto xrDestroyPlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrDestroyPlaneDetectorEXT"); + auto xrBeginPlaneDetectionEXT = GetInstanceExtensionFunction(instance, "xrBeginPlaneDetectionEXT"); + const XrSpace localSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_LOCAL, XrPosefCPP{}); + const XrSpace viewSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_VIEW, XrPosefCPP{}); + + // Set up composition projection layer and swapchains (one swapchain per view). + std::vector swapchains; + XrCompositionLayerProjection* const projLayer = compositionHelper.CreateProjectionLayer(localSpace); + { + const std::vector viewProperties = compositionHelper.EnumerateConfigurationViews(); + for (uint32_t j = 0; j < projLayer->viewCount; j++) { + const XrSwapchain swapchain = compositionHelper.CreateSwapchain(compositionHelper.DefaultColorSwapchainCreateInfo( + viewProperties[j].recommendedImageRectWidth, viewProperties[j].recommendedImageRectHeight)); + const_cast(projLayer->views[j].subImage) = compositionHelper.MakeDefaultSubImage(swapchain, 0); + swapchains.push_back(swapchain); + } + } + + compositionHelper.BeginSession(); + + XrPlaneDetectorCreateInfoEXT createInfo{XR_TYPE_PLANE_DETECTOR_CREATE_INFO_EXT}; + createInfo.flags = XR_PLANE_DETECTOR_ENABLE_CONTOUR_BIT_EXT; + XrPlaneDetectorEXT detection = XR_NULL_HANDLE; + REQUIRE(XR_SUCCESS == xrCreatePlaneDetectorEXT(compositionHelper.GetSession(), &createInfo, &detection)); + + // Lambda to create the instructional quad layer placed to the left. + auto makeInstructionsQuad = [&](const char* instructions) { + XrCompositionLayerQuad* const instructionsQuad = compositionHelper.CreateQuadLayer( + compositionHelper.CreateStaticSwapchainImage(CreateTextImage(1024, 512, instructions, 48)), localSpace, 1.0f, + {{0, 0, 0, 1}, {-0.2f, 0, -1.0f}}); + XrQuaternionf_CreateFromAxisAngle(&instructionsQuad->pose.orientation, &Up, 10 * MATH_PI / 180); + }; + + // Configure the XrPlaneDetectorBeginInfoEXT correctly first, before making it invalid in sections. + XrPosef pose{}; + pose.position = XrVector3f{0.0f, 0.0f, 0.0f}; + pose.orientation = XrQuaternionf{0.0f, 0.0f, 0.0f, 1.0f}; + + XrPlaneDetectorBeginInfoEXT beginInfo{XR_TYPE_PLANE_DETECTOR_BEGIN_INFO_EXT}; + beginInfo.minArea = 0.1f; + beginInfo.maxPlanes = 100; + + std::vector orientations = {XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_UPWARD_EXT}; + beginInfo.orientationCount = (uint32_t)orientations.size(); + beginInfo.orientations = orientations.data(); + + beginInfo.boundingBoxPose = pose; + beginInfo.boundingBoxExtent = XrExtent3DfEXT{10.0f, 10.0f, 10.0f}; + + beginInfo.baseSpace = viewSpace; + + beginInfo.time = 0; + + SECTION("invalid-filters") + { + // intentionally do this wrong: + beginInfo.orientationCount = 1; + beginInfo.orientations = nullptr; + + makeInstructionsQuad("Testing null filters with count"); + + RenderLoop(compositionHelper.GetSession(), [&](const XrFrameState& frameState) { + beginInfo.time = frameState.predictedDisplayTime; + REQUIRE(XR_ERROR_VALIDATION_FAILURE == xrBeginPlaneDetectionEXT(detection, &beginInfo)); + return false; + }).Loop(); + } + SECTION("invalid-time") + { + + makeInstructionsQuad("Testing invalid time"); + + RenderLoop(compositionHelper.GetSession(), [&](const XrFrameState& /* frameState */) { + beginInfo.time = 0; + REQUIRE(XR_ERROR_TIME_INVALID == xrBeginPlaneDetectionEXT(detection, &beginInfo)); + return false; + }).Loop(); + } + SECTION("invalid-pose") + { + + pose.orientation = XrQuaternionf{0.0f, 0.0f, 0.0f, 0.0f}; + + makeInstructionsQuad("Testing invalid pose"); + + RenderLoop(compositionHelper.GetSession(), [&](const XrFrameState& frameState) { + beginInfo.time = frameState.predictedDisplayTime; + beginInfo.boundingBoxPose = pose; + REQUIRE(XR_ERROR_POSE_INVALID == xrBeginPlaneDetectionEXT(detection, &beginInfo)); + return false; + }).Loop(); + } + + REQUIRE(XR_SUCCESS == xrDestroyPlaneDetectorEXT(detection)); + } + + static bool IsClockWise(const XrVector2f* points, size_t count) + { + float area = 0.0f; + for (size_t i = 0; i < count; i++) { + const auto& p1 = points[i]; + const auto& p2 = points[(i + 1) % count]; + area += (p2.x - p1.x) * (p2.y + p1.y); + } + return area > 0.0f; + } + + static void RunPlaneContourTest(const std::vector& orientations, const char* exampleImage, + const char* instructions) + { + + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_PLANE_DETECTION_EXTENSION_NAME)) { + SKIP(); + } + + if (!globalData.IsUsingGraphicsPlugin()) { + SKIP(); + } + + CompositionHelper compositionHelper("XR_EXT_plane_detection", {XR_EXT_PLANE_DETECTION_EXTENSION_NAME}); + + if (!SystemSupportsEXTPlaneDetection(compositionHelper.GetInstance(), compositionHelper.GetSystemId())) { + SKIP("System does not support plane detection"); + } + + InteractiveLayerManager interactiveLayerManager(compositionHelper, exampleImage, instructions); + + compositionHelper.GetInteractionManager().AttachActionSets(); + + MeshProjectionLayerHelper meshProjectionLayerHelper(compositionHelper); + + XrInstance instance = compositionHelper.GetInstance(); + + auto xrCreatePlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrCreatePlaneDetectorEXT"); + auto xrDestroyPlaneDetectorEXT = GetInstanceExtensionFunction(instance, "xrDestroyPlaneDetectorEXT"); + auto xrBeginPlaneDetectionEXT = GetInstanceExtensionFunction(instance, "xrBeginPlaneDetectionEXT"); + auto xrGetPlaneDetectionStateEXT = + GetInstanceExtensionFunction(instance, "xrGetPlaneDetectionStateEXT"); + auto xrGetPlaneDetectionsEXT = GetInstanceExtensionFunction(instance, "xrGetPlaneDetectionsEXT"); + auto xrGetPlanePolygonBufferEXT = + GetInstanceExtensionFunction(instance, "xrGetPlanePolygonBufferEXT"); + + const XrSpace localSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_LOCAL, XrPosefCPP{}); + const XrSpace viewSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_VIEW, XrPosefCPP{}); + + // Set up composition projection layer and swapchains (one swapchain per view). + std::vector swapchains; + XrCompositionLayerProjection* const projLayer = compositionHelper.CreateProjectionLayer(localSpace); + { + const std::vector viewProperties = compositionHelper.EnumerateConfigurationViews(); + for (uint32_t j = 0; j < projLayer->viewCount; j++) { + const XrSwapchain swapchain = compositionHelper.CreateSwapchain(compositionHelper.DefaultColorSwapchainCreateInfo( + viewProperties[j].recommendedImageRectWidth, viewProperties[j].recommendedImageRectHeight)); + const_cast(projLayer->views[j].subImage) = compositionHelper.MakeDefaultSubImage(swapchain, 0); + swapchains.push_back(swapchain); + } + } + + compositionHelper.BeginSession(); + + XrPlaneDetectorCreateInfoEXT createInfo{XR_TYPE_PLANE_DETECTOR_CREATE_INFO_EXT}; + createInfo.flags = XR_PLANE_DETECTOR_ENABLE_CONTOUR_BIT_EXT; + XrPlaneDetectorEXT detection = XR_NULL_HANDLE; + REQUIRE(XR_SUCCESS == xrCreatePlaneDetectorEXT(compositionHelper.GetSession(), &createInfo, &detection)); + + enum DetectState + { + IDLE, + WAITING, + PROCESSING, + RETRIEVING, + }; + DetectState detect_state = IDLE; + + struct MeshData + { + std::vector indices; + std::vector vertices; + XrPosef pose{}; + }; + + std::vector meshes; + std::future> retrieval; + + auto update = [&](const XrFrameState& frameState) { + switch (detect_state) { + case IDLE: { + XrPosef pose{}; + pose.position = XrVector3f{0.0f, 0.0f, 0.0f}; + pose.orientation = XrQuaternionf{0.0f, 0.0f, 0.0f, 1.0f}; + + XrPlaneDetectorBeginInfoEXT beginInfo{XR_TYPE_PLANE_DETECTOR_BEGIN_INFO_EXT}; + beginInfo.orientationCount = (uint32_t)orientations.size(); + beginInfo.orientations = orientations.data(); + beginInfo.minArea = 0.1f; + beginInfo.maxPlanes = 100; + beginInfo.boundingBoxPose = pose; + beginInfo.boundingBoxExtent = XrExtent3DfEXT{10.0f, 10.0f, 10.0f}; + beginInfo.time = frameState.predictedDisplayTime; + beginInfo.baseSpace = viewSpace; + REQUIRE(XR_SUCCESS == xrBeginPlaneDetectionEXT(detection, &beginInfo)); + detect_state = WAITING; + } break; + case WAITING: { + XrPlaneDetectionStateEXT state; + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionStateEXT(detection, &state)); + REQUIRE((state == XR_PLANE_DETECTION_STATE_PENDING_EXT || state == XR_PLANE_DETECTION_STATE_DONE_EXT || + state == XR_PLANE_DETECTION_STATE_ERROR_EXT)); + switch (state) { + case XR_PLANE_DETECTION_STATE_DONE_EXT: + detect_state = PROCESSING; + break; + case XR_PLANE_DETECTION_STATE_ERROR_EXT: + detect_state = IDLE; + break; + case XR_PLANE_DETECTION_STATE_PENDING_EXT: + break; + default: + break; + } + } break; + case PROCESSING: { + XrPlaneDetectorGetInfoEXT getInfo{XR_TYPE_PLANE_DETECTOR_GET_INFO_EXT}; + getInfo.baseSpace = localSpace; + getInfo.time = frameState.predictedDisplayTime; + XrPlaneDetectorLocationsEXT locations{XR_TYPE_PLANE_DETECTOR_LOCATIONS_EXT}; + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionsEXT(detection, &getInfo, &locations)); + + if (locations.planeLocationCountOutput == 0) { + break; + } + + std::vector location_vector; + location_vector.resize(locations.planeLocationCountOutput); + for (auto& location : location_vector) { + location.type = XR_TYPE_PLANE_DETECTOR_LOCATION_EXT; + location.next = nullptr; + } + locations.planeLocations = location_vector.data(); + locations.planeLocationCapacityInput = (uint32_t)location_vector.size(); + + REQUIRE(XR_SUCCESS == xrGetPlaneDetectionsEXT(detection, &getInfo, &locations)); + detect_state = RETRIEVING; + + retrieval = std::async(std::launch::async, [detection, location_vector, xrGetPlanePolygonBufferEXT]() { + std::vector local_meshes; + for (auto& location : location_vector) { + + std::vector> polygon; + MeshData md; + + for (uint32_t polygonBufferIndex = 0; polygonBufferIndex < location.polygonBufferCount; polygonBufferIndex++) { + + XrPlaneDetectorPolygonBufferEXT polygonBuffer{XR_TYPE_PLANE_DETECTOR_POLYGON_BUFFER_EXT}; + + REQUIRE(XR_SUCCESS == + xrGetPlanePolygonBufferEXT(detection, location.planeId, polygonBufferIndex, &polygonBuffer)); + REQUIRE(polygonBuffer.vertexCountOutput > 0); + + std::vector vertices; + vertices.resize(polygonBuffer.vertexCountOutput); + polygonBuffer.vertexCapacityInput = (uint32_t)vertices.size(); + polygonBuffer.vertices = vertices.data(); + REQUIRE(XR_SUCCESS == + xrGetPlanePolygonBufferEXT(detection, location.planeId, polygonBufferIndex, &polygonBuffer)); + + XrMatrix4x4f transform; + XrVector3f scale{1.0f, 1.0f, 1.0f}; + XrMatrix4x4f_CreateTranslationRotationScale(&transform, &location.pose.position, &location.pose.orientation, + &scale); + CAPTURE(polygonBufferIndex); + if (polygonBufferIndex == 0) { + // hull is counter clock-wise. + REQUIRE(IsClockWise(polygonBuffer.vertices, polygonBuffer.vertexCountOutput) == false); + } + else { + // holes are clock-wise + REQUIRE(IsClockWise(polygonBuffer.vertices, polygonBuffer.vertexCountOutput) == true); + } + + polygon.push_back(vertices); + + for (const XrVector2f& vertex : vertices) { + XrVector3f source = {vertex.x, vertex.y, 0.0f}; + md.vertices.push_back({source, Geometry::DarkBlue}); + } + } + + md.indices = mapbox::earcut(polygon); + md.pose = location.pose; + std::reverse(md.indices.begin(), md.indices.end()); + local_meshes.push_back(md); + } + return local_meshes; + }); + + } break; + case RETRIEVING: { + using namespace std::chrono_literals; + if (retrieval.wait_for(0ms) == std::future_status::ready) { + meshes.clear(); + auto source_meshes = retrieval.get(); + for (const auto& mesh_data : source_meshes) { + auto mesh = GetGlobalData().graphicsPlugin->MakeSimpleMesh(mesh_data.indices, mesh_data.vertices); + meshes.emplace_back(mesh, mesh_data.pose); + } + detect_state = IDLE; + } + } break; + } // case + + auto viewData = compositionHelper.LocateViews(localSpace, frameState.predictedDisplayTime); + const auto& viewState = std::get(viewData); + std::vector layers; + if (viewState.viewStateFlags & XR_VIEW_STATE_POSITION_VALID_BIT && + viewState.viewStateFlags & XR_VIEW_STATE_ORIENTATION_VALID_BIT) { + const auto& views = std::get>(viewData); + + // Render into each viewport of the wide swapchain using the projection layer view fov and pose. + for (size_t view = 0; view < views.size(); view++) { + compositionHelper.AcquireWaitReleaseImage(swapchains[view], // + [&](const XrSwapchainImageBaseHeader* swapchainImage) { + GetGlobalData().graphicsPlugin->ClearImageSlice(swapchainImage); + const_cast(projLayer->views[view].fov) = views[view].fov; + const_cast(projLayer->views[view].pose) = views[view].pose; + GetGlobalData().graphicsPlugin->RenderView( + projLayer->views[view], swapchainImage, RenderParams().Draw(meshes)); + }); + } + + layers.push_back({reinterpret_cast(projLayer)}); + } + return interactiveLayerManager.EndFrame(frameState, layers); + }; + + RenderLoop(compositionHelper.GetSession(), update).Loop(); + + REQUIRE(XR_SUCCESS == xrDestroyPlaneDetectorEXT(detection)); + } + + TEST_CASE("XR_EXT_plane_detection-contour-HU", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneContourTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_UPWARD_EXT}, "ext_plane_detection_contour.png", + "This should show the plane contours of all upward horizontal planes."); + } + + TEST_CASE("XR_EXT_plane_detection-contour-HD", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneContourTest({XR_PLANE_DETECTOR_ORIENTATION_HORIZONTAL_DOWNWARD_EXT}, "ext_plane_detection_contour.png", + "This should show the plane contours of all downward horizontal planes."); + } + + TEST_CASE("XR_EXT_plane_detection-contour-V", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneContourTest({XR_PLANE_DETECTOR_ORIENTATION_VERTICAL_EXT}, "ext_plane_detection_contour.png", + "This should show the plane contours of all vertical planes."); + } + + TEST_CASE("XR_EXT_plane_detection-contour-A", "[scenario][interactive][no_auto][XR_EXT_plane_detection]") + { + RunPlaneContourTest({XR_PLANE_DETECTOR_ORIENTATION_ARBITRARY_EXT}, "ext_plane_detection_contour.png", + "This should show the plane contours of all non vertical / horizontal (arbitrary) planes."); + } + +} // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_EXT_thermal_query.cpp b/src/conformance/conformance_test/test_XR_EXT_thermal_query.cpp index bc9dcae6..4caf22b2 100644 --- a/src/conformance/conformance_test/test_XR_EXT_thermal_query.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_thermal_query.cpp @@ -14,13 +14,15 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" + #include #include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp index b5e837e1..bff5dd31 100644 --- a/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp @@ -14,24 +14,24 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_D3D11 #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include -#include -#ifdef XR_USE_GRAPHICS_API_D3D11 +#include #include #include #include // For Microsoft::WRL::ComPtr +#include +#include + using namespace Microsoft::WRL; namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp index be3a5bbb..7bd3e69a 100644 --- a/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp @@ -14,24 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_D3D12 #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include #include - -#ifdef XR_USE_GRAPHICS_API_D3D12 #include #include #include // For Microsoft::WRL::ComPtr +#include +#include + using namespace Microsoft::WRL; namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp index c47f20e5..429e460c 100644 --- a/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp @@ -14,23 +14,21 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_OPENGL_ES #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include -#include -#include +#include "xr_dependencies.h" -#ifdef XR_USE_GRAPHICS_API_OPENGL_ES +#include -#include "xr_dependencies.h" +#include #include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp index e212df4e..490500d6 100644 --- a/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp @@ -14,23 +14,22 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_OPENGL + #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include #include - -#ifdef XR_USE_GRAPHICS_API_OPENGL - #include "xr_dependencies.h" #include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp index 455cfcfb..a9ba932f 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp @@ -14,20 +14,22 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_generator.h" -#include "bitmask_to_string.h" -#include -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/generator.h" +#include "utilities/bitmask_generator.h" +#include "utilities/bitmask_to_string.h" + #include #include +#include +#include +#include +#include +#include +#include + namespace Conformance { // This implements an automated programmatic test of the cubemap layer. However, a separate visual diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp index c3dd846c..624d320c 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp @@ -14,20 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_generator.h" -#include "bitmask_to_string.h" -#include -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/bitmask_generator.h" +#include "utilities/bitmask_to_string.h" + #include #include +#include +#include +#include +#include +#include + namespace Conformance { // This implements an automated programmatic test of the cylinder layer. However, a separate visual diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp index ab40e27a..0b1d18ed 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp @@ -14,19 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_generator.h" -#include -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/bitmask_generator.h" + #include #include +#include +#include +#include + namespace Conformance { // This implements an automated programmatic test of depth layers. However, a separate visual diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp index 96e22202..64deccc5 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp @@ -14,20 +14,22 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_generator.h" -#include "bitmask_to_string.h" -#include -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/bitmask_generator.h" +#include "utilities/bitmask_to_string.h" +#include "utilities/generator.h" + #include #include +#include +#include +#include +#include +#include +#include + namespace Conformance { // This implements an automated programmatic test of the equirect layer. However, a separate visual diff --git a/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp b/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp index 03838caf..fa8718c9 100644 --- a/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp @@ -14,7 +14,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" #include diff --git a/src/conformance/conformance_test/test_XR_KHR_headless.cpp b/src/conformance/conformance_test/test_XR_KHR_headless.cpp index d5f3a1fc..2c266c43 100644 --- a/src/conformance/conformance_test/test_XR_KHR_headless.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_headless.cpp @@ -14,7 +14,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" #include diff --git a/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp b/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp index 32c145b5..5a2fe5d5 100644 --- a/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp @@ -14,24 +14,35 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "matchers.h" -#include "utils.h" -#include "conformance_utils.h" +#include "composition_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" +#include "mesh_projection_layer.h" #include "two_call_struct_metadata.h" #include "two_call_struct_tests.h" -#include "composition_utils.h" -#include "mesh_projection_layer.h" -#include "earcut.hpp" -#include -#include -#include -#include -#include +#include "type_utils.h" +#include "matchers.h" +#include "utilities/Geometry.h" +#include "utilities/types_and_constants.h" + +#include "nonstd/type.hpp" + +#include #include #include +#include #include +#include +#include +#include +#include +#include +#include +#include +#include + namespace Conformance { constexpr XrColor4f BrightRed = {1.0, 0.0, 0.0, 1.0f}; diff --git a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp index c8481f3f..fb12e8cc 100644 --- a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp @@ -14,22 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_VULKAN + #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/types_and_constants.h" +#include "xr_dependencies.h" + #include #include - -#ifdef XR_USE_GRAPHICS_API_VULKAN -#include "xr_dependencies.h" #include +#include +#include +#include + namespace Conformance { TEST_CASE("XR_KHR_vulkan_enable", "") diff --git a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp index c6c6a8ec..7c458dd4 100644 --- a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp @@ -14,22 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" +#ifdef XR_USE_GRAPHICS_API_VULKAN + #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/types_and_constants.h" +#include "xr_dependencies.h" + #include #include - -#ifdef XR_USE_GRAPHICS_API_VULKAN -#include "xr_dependencies.h" #include +#include +#include +#include + namespace Conformance { TEST_CASE("XR_KHR_vulkan_enable2", "") diff --git a/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp b/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp index 5ced8a4f..78edae70 100644 --- a/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp @@ -14,7 +14,8 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#ifdef XR_USE_PLATFORM_WIN32 +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" #include @@ -24,13 +25,10 @@ #include "xr_dependencies.h" #include -#ifdef XR_USE_PLATFORM_WIN32 #include -#endif namespace Conformance { -#ifdef XR_USE_PLATFORM_WIN32 TEST_CASE("XR_KHR_win32_convert_performance_counter_time", "") { GlobalData& globalData = GetGlobalData(); @@ -141,5 +139,6 @@ namespace Conformance REQUIRE(qpcAfter > qpcBefore); } } -#endif // XR_USE_PLATFORM_WIN32 } // namespace Conformance + // +#endif // XR_USE_PLATFORM_WIN32 diff --git a/src/conformance/conformance_test/test_XR_META_headset_id.cpp b/src/conformance/conformance_test/test_XR_META_headset_id.cpp index eacfc584..f41885d5 100644 --- a/src/conformance/conformance_test/test_XR_META_headset_id.cpp +++ b/src/conformance/conformance_test/test_XR_META_headset_id.cpp @@ -15,25 +15,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" +#include "utilities/utils.h" + #include -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" +#include "utilities/system_properties_helper.h" #include +#include +#include + namespace Conformance { - static inline XrUuidEXT queryHeadsetId(XrInstance instance, XrSystemId systemId) - { - XrSystemHeadsetIdPropertiesMETA headsetId = {XR_TYPE_SYSTEM_HEADSET_ID_PROPERTIES_META}; - XrSystemProperties systemProperties{XR_TYPE_SYSTEM_PROPERTIES}; - systemProperties.next = &headsetId; - - REQUIRE(XR_SUCCESS == xrGetSystemProperties(instance, systemId, &systemProperties)); - return headsetId.id; - } + static const auto QueryHeadsetId = MakeSystemPropertiesChecker( + XrSystemHeadsetIdPropertiesMETA{XR_TYPE_SYSTEM_HEADSET_ID_PROPERTIES_META}, &XrSystemHeadsetIdPropertiesMETA::id); TEST_CASE("XR_META_headset_id", "[XR_META_headset_id]") { @@ -49,7 +47,7 @@ namespace Conformance AutoBasicInstance instance(AutoBasicInstance::createSystemId); XrSystemId systemId = instance.systemId; - XrUuidEXT headsetId = queryHeadsetId(instance, systemId); + XrUuidEXT headsetId = QueryHeadsetId(instance, systemId); // Validate headsetid has NOT been filled in XrUuidEXT empty{}; @@ -67,7 +65,7 @@ namespace Conformance SECTION("Valid UUID returned") { - XrUuidEXT headsetId = queryHeadsetId(instance, systemId); + XrUuidEXT headsetId = QueryHeadsetId(instance, systemId); // Validate headsetid has been filled in XrUuidEXT empty{}; @@ -76,8 +74,8 @@ namespace Conformance SECTION("Consistent UUID returned") { - XrUuidEXT headsetId1 = queryHeadsetId(instance, systemId); - XrUuidEXT headsetId2 = queryHeadsetId(instance, systemId); + XrUuidEXT headsetId1 = QueryHeadsetId(instance, systemId); + XrUuidEXT headsetId2 = QueryHeadsetId(instance, systemId); // Validate headsetid is consistent REQUIRE(memcmp(&headsetId1, &headsetId2, sizeof(XrUuidEXT)) == 0); diff --git a/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp b/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp index 4798f9e1..25b63a78 100644 --- a/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp +++ b/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp @@ -15,18 +15,18 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "two_call.h" -#include -#include -#include -#include -#include + #include #include +#include +#include +#include +#include + namespace Conformance { TEST_CASE("XR_META_performance_metrics", "") diff --git a/src/conformance/conformance_test/test_XR_MND_headless.cpp b/src/conformance/conformance_test/test_XR_MND_headless.cpp index 0e437f5a..277a3879 100644 --- a/src/conformance/conformance_test/test_XR_MND_headless.cpp +++ b/src/conformance/conformance_test/test_XR_MND_headless.cpp @@ -14,17 +14,14 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" #include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include + #include #include +#include + namespace Conformance { TEST_CASE("XR_MND_headless", "") diff --git a/src/conformance/conformance_test/test_XrCompositionLayerProjection.cpp b/src/conformance/conformance_test/test_XrCompositionLayerProjection.cpp index e723c350..54738e6a 100644 --- a/src/conformance/conformance_test/test_XrCompositionLayerProjection.cpp +++ b/src/conformance/conformance_test/test_XrCompositionLayerProjection.cpp @@ -14,15 +14,24 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_generator.h" -#include "bitmask_to_string.h" -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/bitmask_generator.h" +#include "utilities/bitmask_to_string.h" +#include "utilities/generator.h" + #include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include namespace Conformance { diff --git a/src/conformance/conformance_test/test_XrCompositionLayerQuad.cpp b/src/conformance/conformance_test/test_XrCompositionLayerQuad.cpp index 2d84005e..00384567 100644 --- a/src/conformance/conformance_test/test_XrCompositionLayerQuad.cpp +++ b/src/conformance/conformance_test/test_XrCompositionLayerQuad.cpp @@ -14,17 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include "throw_helpers.h" -#include "bitmask_generator.h" -#include "bitmask_to_string.h" -#include +#include "conformance_utils.h" +#include "utilities/bitmask_generator.h" +#include "utilities/generator.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" + #include -#include #include +#include +#include +#include +#include +#include +#include + namespace Conformance { TEST_CASE("XrCompositionLayerQuad", "") diff --git a/src/conformance/conformance_test/test_actions.cpp b/src/conformance/conformance_test/test_actions.cpp index 8b978b77..ce37068f 100644 --- a/src/conformance/conformance_test/test_actions.cpp +++ b/src/conformance/conformance_test/test_actions.cpp @@ -14,25 +14,40 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "catch2/catch_message.hpp" -#include "utils.h" -#include "report.h" -#include "two_call.h" -#include "report.h" -#include "conformance_utils.h" -#include "bitmask_to_string.h" -#include "conformance_framework.h" +#include "action_utils.h" #include "composition_utils.h" +#include "conformance_framework.h" +#include "conformance_utils.h" #include "input_testinputdevice.h" -#include "action_utils.h" +#include "report.h" +#include "two_call.h" +#include "utilities/bitmask_to_string.h" +#include "utilities/event_reader.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include +#include #include #include + +#include #include -#include -#include +#include +#include +#include +#include +#include +#include +#include +#include #include +#include #include +#include +#include +#include +#include using namespace std::chrono_literals; using namespace Conformance; diff --git a/src/conformance/conformance_test/test_multithreading.cpp b/src/conformance/conformance_test/test_multithreading.cpp index 721eb6cc..75559484 100644 --- a/src/conformance/conformance_test/test_multithreading.cpp +++ b/src/conformance/conformance_test/test_multithreading.cpp @@ -1,4 +1,4 @@ -// Copyright (c) 2017 The Khronos Group Inc. +// Copyright (c) 2017-2023, The Khronos Group Inc. // // SPDX-License-Identifier: Apache-2.0 // @@ -14,26 +14,28 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "swapchain_image_data.h" -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" +#include "utilities/utils.h" + +#include +#include #include #include -#include -#include -#include +#include +#include #include #include #include -#include #include -#include #include -#include -#include +#include +#include +#include // Include all dependencies of openxr_platform as configured #include "xr_dependencies.h" diff --git a/src/conformance/conformance_test/test_xrCreateInstance.cpp b/src/conformance/conformance_test/test_xrCreateInstance.cpp index 6db0be7a..e7ab8232 100644 --- a/src/conformance/conformance_test/test_xrCreateInstance.cpp +++ b/src/conformance/conformance_test/test_xrCreateInstance.cpp @@ -14,18 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "catch2/catch_message.hpp" -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include #include +#include +#include + namespace Conformance { @@ -60,7 +59,7 @@ namespace Conformance updateCreateInfoApiLayers(); // Enable only the required platform extensions by default - StringVec enabledExtensions = globalData.requiredPlatformInstanceExtensions; + auto enabledExtensions = StringVec(globalData.requiredPlatformInstanceExtensions); // Call this to update createInfo after modifying enabledExtensions. auto updateCreateInfoExtensions = [&] { diff --git a/src/conformance/conformance_test/test_xrCreateReferenceSpace.cpp b/src/conformance/conformance_test/test_xrCreateReferenceSpace.cpp index 8c0e9645..21f534e7 100644 --- a/src/conformance/conformance_test/test_xrCreateReferenceSpace.cpp +++ b/src/conformance/conformance_test/test_xrCreateReferenceSpace.cpp @@ -14,13 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "two_call.h" -#include + #include #include +#include +#include +#include +#include namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrCreateSession.cpp b/src/conformance/conformance_test/test_xrCreateSession.cpp index a339fbd2..a29d4612 100644 --- a/src/conformance/conformance_test/test_xrCreateSession.cpp +++ b/src/conformance/conformance_test/test_xrCreateSession.cpp @@ -14,18 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/types_and_constants.h" + #include +#include #include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrCreateSwapchain.cpp b/src/conformance/conformance_test/test_xrCreateSwapchain.cpp index 6273e040..ac179090 100644 --- a/src/conformance/conformance_test/test_xrCreateSwapchain.cpp +++ b/src/conformance/conformance_test/test_xrCreateSwapchain.cpp @@ -14,15 +14,22 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "swapchain_parameters.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" #include "two_call.h" -#include "bitmask_generator.h" -#include +#include "utilities/swapchain_parameters.h" +#include "utilities/types_and_constants.h" + +#include #include #include +#include +#include +#include +#include + namespace Conformance { static inline void checkCreateSwapchain(AutoBasicSession& session, int64_t format, const XrViewConfigurationView& vcv) diff --git a/src/conformance/conformance_test/test_xrEnumerateApiLayerProperties.cpp b/src/conformance/conformance_test/test_xrEnumerateApiLayerProperties.cpp index 15d7768a..ac6e61b3 100644 --- a/src/conformance/conformance_test/test_xrEnumerateApiLayerProperties.cpp +++ b/src/conformance/conformance_test/test_xrEnumerateApiLayerProperties.cpp @@ -14,17 +14,16 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" -#include "conformance_framework.h" -#include -#include -#include -#include -#include + #include #include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrEnumerateEnvironmentBlendModes.cpp b/src/conformance/conformance_test/test_xrEnumerateEnvironmentBlendModes.cpp index b03d4eb5..81b98126 100644 --- a/src/conformance/conformance_test/test_xrEnumerateEnvironmentBlendModes.cpp +++ b/src/conformance/conformance_test/test_xrEnumerateEnvironmentBlendModes.cpp @@ -14,20 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "matchers.h" -#include -#include -#include -#include -#include + #include #include #include #include +#include +#include +#include +#include + using Catch::Matchers::VectorContains; #define AS_LIST(name, val) name, diff --git a/src/conformance/conformance_test/test_xrEnumerateInstanceExtensionProperties.cpp b/src/conformance/conformance_test/test_xrEnumerateInstanceExtensionProperties.cpp index 54fa8f37..a0af3c21 100644 --- a/src/conformance/conformance_test/test_xrEnumerateInstanceExtensionProperties.cpp +++ b/src/conformance/conformance_test/test_xrEnumerateInstanceExtensionProperties.cpp @@ -14,17 +14,16 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" #include "conformance_utils.h" -#include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include #include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrEnumerateReferenceSpaces.cpp b/src/conformance/conformance_test/test_xrEnumerateReferenceSpaces.cpp index afcd9d73..df984a41 100644 --- a/src/conformance/conformance_test/test_xrEnumerateReferenceSpaces.cpp +++ b/src/conformance/conformance_test/test_xrEnumerateReferenceSpaces.cpp @@ -14,14 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "conformance_utils.h" #include "conformance_framework.h" -#include "two_call.h" +#include "conformance_utils.h" #include "matchers.h" +#include "two_call.h" +#include "utilities/types_and_constants.h" + #include #include #include +#include +#include +#include + namespace Conformance { using Catch::Matchers::VectorContains; diff --git a/src/conformance/conformance_test/test_xrEnumerateSwapchainFormats.cpp b/src/conformance/conformance_test/test_xrEnumerateSwapchainFormats.cpp index 48dca458..68008ecd 100644 --- a/src/conformance/conformance_test/test_xrEnumerateSwapchainFormats.cpp +++ b/src/conformance/conformance_test/test_xrEnumerateSwapchainFormats.cpp @@ -17,9 +17,12 @@ #include "conformance_utils.h" #include "conformance_framework.h" #include "two_call.h" + #include #include +#include + namespace Conformance { TEST_CASE("xrEnumerateSwapchainFormats") diff --git a/src/conformance/conformance_test/test_xrGetInstanceProcAddr.cpp b/src/conformance/conformance_test/test_xrGetInstanceProcAddr.cpp index 9e06cf2c..3b11e3b5 100644 --- a/src/conformance/conformance_test/test_xrGetInstanceProcAddr.cpp +++ b/src/conformance/conformance_test/test_xrGetInstanceProcAddr.cpp @@ -14,18 +14,21 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include #include +#include +#include +#include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrGetInstanceProperties.cpp b/src/conformance/conformance_test/test_xrGetInstanceProperties.cpp index 3c0d1a64..9e2deb32 100644 --- a/src/conformance/conformance_test/test_xrGetInstanceProperties.cpp +++ b/src/conformance/conformance_test/test_xrGetInstanceProperties.cpp @@ -14,17 +14,16 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include #include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrGetReferenceSpaceBoundsRect.cpp b/src/conformance/conformance_test/test_xrGetReferenceSpaceBoundsRect.cpp index 88ae4357..56a6d4a5 100644 --- a/src/conformance/conformance_test/test_xrGetReferenceSpaceBoundsRect.cpp +++ b/src/conformance/conformance_test/test_xrGetReferenceSpaceBoundsRect.cpp @@ -14,13 +14,19 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "conformance_utils.h" #include "conformance_framework.h" -#include "two_call.h" +#include "conformance_utils.h" #include "matchers.h" +#include "two_call.h" +#include "utilities/types_and_constants.h" + #include #include +#include +#include +#include + namespace Conformance { TEST_CASE("xrGetReferenceSpaceBoundsRect", "") diff --git a/src/conformance/conformance_test/test_xrGetSystem.cpp b/src/conformance/conformance_test/test_xrGetSystem.cpp index b74fb583..def4e6db 100644 --- a/src/conformance/conformance_test/test_xrGetSystem.cpp +++ b/src/conformance/conformance_test/test_xrGetSystem.cpp @@ -14,17 +14,15 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" + #include #include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrGetSystemProperties.cpp b/src/conformance/conformance_test/test_xrGetSystemProperties.cpp index acc33d4b..0bc6fb1c 100644 --- a/src/conformance/conformance_test/test_xrGetSystemProperties.cpp +++ b/src/conformance/conformance_test/test_xrGetSystemProperties.cpp @@ -14,17 +14,16 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" +#include "utilities/utils.h" #include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include + #include #include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrLinear.cpp b/src/conformance/conformance_test/test_xrLinear.cpp index 58c6d0ac..43468d6d 100644 --- a/src/conformance/conformance_test/test_xrLinear.cpp +++ b/src/conformance/conformance_test/test_xrLinear.cpp @@ -14,9 +14,12 @@ // See the License for the specific language governing permissions and // limitations under the License. +#include "common/xr_linear.h" + #include #include -#include + +#include namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrLocateSpace.cpp b/src/conformance/conformance_test/test_xrLocateSpace.cpp index 3d2da1c8..99c2c6bc 100644 --- a/src/conformance/conformance_test/test_xrLocateSpace.cpp +++ b/src/conformance/conformance_test/test_xrLocateSpace.cpp @@ -14,14 +14,20 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "conformance_utils.h" #include "conformance_framework.h" -#include "bitmask_to_string.h" -#include "two_call.h" -#include +#include "conformance_utils.h" +#include "utilities/bitmask_to_string.h" +#include "utilities/types_and_constants.h" + #include #include +#include +#include +#include +#include +#include + namespace Conformance { TEST_CASE("xrLocateSpace", "") diff --git a/src/conformance/conformance_test/test_xrPathToString.cpp b/src/conformance/conformance_test/test_xrPathToString.cpp index e54b31cd..5bb87546 100644 --- a/src/conformance/conformance_test/test_xrPathToString.cpp +++ b/src/conformance/conformance_test/test_xrPathToString.cpp @@ -14,14 +14,9 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" #include "conformance_utils.h" -#include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "utilities/utils.h" + #include #include diff --git a/src/conformance/conformance_test/test_xrPollEvent.cpp b/src/conformance/conformance_test/test_xrPollEvent.cpp index 70071d99..10d96278 100644 --- a/src/conformance/conformance_test/test_xrPollEvent.cpp +++ b/src/conformance/conformance_test/test_xrPollEvent.cpp @@ -14,17 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include #include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrRequestExitSession.cpp b/src/conformance/conformance_test/test_xrRequestExitSession.cpp index 8e2cf270..d4456f36 100644 --- a/src/conformance/conformance_test/test_xrRequestExitSession.cpp +++ b/src/conformance/conformance_test/test_xrRequestExitSession.cpp @@ -14,9 +14,8 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" #include "conformance_utils.h" -#include "conformance_framework.h" + #include #include diff --git a/src/conformance/conformance_test/test_xrResultToString.cpp b/src/conformance/conformance_test/test_xrResultToString.cpp index 6e9c32bc..7e951cea 100644 --- a/src/conformance/conformance_test/test_xrResultToString.cpp +++ b/src/conformance/conformance_test/test_xrResultToString.cpp @@ -14,18 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/stringification.h" +#include "utilities/types_and_constants.h" + #include +#include #include +#include +#include +#include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrStringToPath.cpp b/src/conformance/conformance_test/test_xrStringToPath.cpp index ef0c328c..f1b66618 100644 --- a/src/conformance/conformance_test/test_xrStringToPath.cpp +++ b/src/conformance/conformance_test/test_xrStringToPath.cpp @@ -14,17 +14,18 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" -#include +#include "conformance_utils.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + +#include +#include + #include -#include #include #include #include -#include -#include namespace Conformance { diff --git a/src/conformance/conformance_test/test_xrStructureTypeToString.cpp b/src/conformance/conformance_test/test_xrStructureTypeToString.cpp index f1ebd36d..3a4f9c03 100644 --- a/src/conformance/conformance_test/test_xrStructureTypeToString.cpp +++ b/src/conformance/conformance_test/test_xrStructureTypeToString.cpp @@ -14,19 +14,23 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "utils.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "matchers.h" -#include -#include -#include -#include -#include +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include +#include #include #include +#include +#include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/framework/CMakeLists.txt b/src/conformance/framework/CMakeLists.txt new file mode 100644 index 00000000..2281d7c8 --- /dev/null +++ b/src/conformance/framework/CMakeLists.txt @@ -0,0 +1,91 @@ +# Copyright (c) 2019-2023, The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 + +set(VULKAN_SHADERS ${CMAKE_CURRENT_SOURCE_DIR}/vulkan_shaders/frag.glsl + ${CMAKE_CURRENT_SOURCE_DIR}/vulkan_shaders/vert.glsl) + +run_xr_xml_generate( + conformance_generator.py function_info.cpp + ${PROJECT_SOURCE_DIR}/src/scripts/template_function_info.cpp) + +add_library( + conformance_framework STATIC + action_utils.cpp + catch_reporter_cts.cpp + composition_utils.cpp + conformance_framework.cpp + conformance_utils.cpp + environment.cpp + graphics_plugin_d3d11.cpp + graphics_plugin_d3d12.cpp + graphics_plugin_factory.cpp + graphics_plugin_opengl.cpp + graphics_plugin_opengles.cpp + graphics_plugin_vulkan.cpp + input_testinputdevice.cpp + mesh_projection_layer.cpp + platform_plugin_android.cpp + platform_plugin_posix.cpp + platform_plugin_win32.cpp + report.cpp + RGBAImage.cpp + swapchain_image_data.cpp + xml_test_environment.cpp + ${VULKAN_SHADERS} + ${CMAKE_CURRENT_BINARY_DIR}/function_info.cpp) + +target_link_libraries( + conformance_framework PUBLIC conformance_utilities OpenXR::openxr_loader + Threads::Threads Catch2) + +target_include_directories( + conformance_framework + PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/.. + # Strong types for integers, etc. + ${PROJECT_SOURCE_DIR}/src/external/type-lite/include + # Backport of std::span functionality to pre-C++17 + ${PROJECT_SOURCE_DIR}/src/external/span-lite/include + # for xr_linear.h: + ${PROJECT_SOURCE_DIR}/src/common + # for openxr.h: + ${PROJECT_BINARY_DIR}/include + PRIVATE ${CMAKE_CURRENT_SOURCE_DIR} + # for compiled shaders + ${CMAKE_CURRENT_BINARY_DIR} + # for STB utilities, etc + ${PROJECT_SOURCE_DIR}/src/external) + +compile_glsl(run_conformance_test_glsl_compiles ${VULKAN_SHADERS}) + +add_dependencies(conformance_framework generate_openxr_header + run_conformance_test_glsl_compiles) + +source_group("Vulkan Shaders" FILES ${VULKAN_SHADERS}) + +if(Vulkan_FOUND) + target_include_directories(conformance_framework + PUBLIC ${Vulkan_INCLUDE_DIRS}) + target_link_libraries(conformance_framework PUBLIC ${Vulkan_LIBRARY}) +endif() + +if(TARGET openxr-gfxwrapper) + target_link_libraries(conformance_framework PUBLIC openxr-gfxwrapper) +endif() + +if(WIN32) + + target_compile_definitions(conformance_framework + PUBLIC _CRT_SECURE_NO_WARNINGS) + if(MSVC) + target_compile_options(conformance_framework + PUBLIC /Zc:wchar_t /Zc:forScope /W4 /WX /wd4996) + + # Right now can't build this on MinGW because of directxcolors, etc. + target_link_libraries(conformance_framework PUBLIC d3d11 d3d12 d3dcompiler + dxgi) + else() + target_compile_definitions(conformance_framework + PUBLIC MISSING_DIRECTX_COLORS) + endif() +endif() diff --git a/src/conformance/framework/RGBAImage.cpp b/src/conformance/framework/RGBAImage.cpp index 6e6fb6d1..32e9d709 100644 --- a/src/conformance/framework/RGBAImage.cpp +++ b/src/conformance/framework/RGBAImage.cpp @@ -2,18 +2,14 @@ // // SPDX-License-Identifier: Apache-2.0 -#include -#include -#include -#include -#include -#include #include "RGBAImage.h" #include "conformance_framework.h" #ifdef XR_USE_PLATFORM_ANDROID #include "unique_asset.h" + +#include #endif // Only one compilation unit can have the STB implementations. @@ -22,6 +18,18 @@ #include "stb/stb_image.h" #include "stb/stb_truetype.h" +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + namespace { // Convert R32G32B32A_FLOAT to R8G8B8A8_UNORM. diff --git a/src/conformance/framework/RGBAImage.h b/src/conformance/framework/RGBAImage.h index f87789ae..c40b7914 100644 --- a/src/conformance/framework/RGBAImage.h +++ b/src/conformance/framework/RGBAImage.h @@ -4,9 +4,9 @@ #pragma once -#include #include -#include + +#include namespace Conformance { diff --git a/src/conformance/framework/action_utils.h b/src/conformance/framework/action_utils.h index d48f7b5c..ac0a6e6b 100644 --- a/src/conformance/framework/action_utils.h +++ b/src/conformance/framework/action_utils.h @@ -16,12 +16,18 @@ #pragma once -#include -#include - +#include "RGBAImage.h" #include "composition_utils.h" -#include "event_reader.h" #include "input_testinputdevice.h" +#include "utilities/event_reader.h" + +#include + +#include +#include +#include +#include +#include namespace Conformance { diff --git a/src/conformance/framework/catch_reporter_cts.h b/src/conformance/framework/catch_reporter_cts.h index c14f5e00..72697852 100644 --- a/src/conformance/framework/catch_reporter_cts.h +++ b/src/conformance/framework/catch_reporter_cts.h @@ -10,7 +10,7 @@ // Based on the upstream JunitReporter -#include "utils.h" +#include "utilities/utils.h" XRC_DISABLE_MSVC_WARNING(4324) diff --git a/src/conformance/framework/composition_utils.cpp b/src/conformance/framework/composition_utils.cpp index 7a593768..64dc6d9a 100644 --- a/src/conformance/framework/composition_utils.cpp +++ b/src/conformance/framework/composition_utils.cpp @@ -15,21 +15,23 @@ // limitations under the License. #include "composition_utils.h" -#include "swapchain_image_data.h" -#include "types_and_constants.h" -#include "utils.h" -#include "report.h" + #include "conformance_framework.h" -#include "throw_helpers.h" -#include -#include -#include -#include -#include -#include -#include +#include "swapchain_image_data.h" +#include "utilities/event_reader.h" +#include "utilities/throw_helpers.h" +#include "utilities/xrduration_literals.h" + +#include + #include -#include + +#include +#include +#include +#include +#include +#include using namespace std::chrono_literals; diff --git a/src/conformance/framework/composition_utils.h b/src/conformance/framework/composition_utils.h index 7ace09c6..f20cdff5 100644 --- a/src/conformance/framework/composition_utils.h +++ b/src/conformance/framework/composition_utils.h @@ -16,25 +16,36 @@ #pragma once +#include "RGBAImage.h" +#include "common/xr_linear.h" +#include "conformance_framework.h" +#include "conformance_utils.h" +#include "graphics_plugin.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" + +#include "catch2/catch_test_macros.hpp" +#include + +#include +#include +#include +#include +#include #include -#include -#include #include -#include -#include -#include -#include #include -#include -#include -#include "graphics_plugin.h" -#include "event_reader.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include "throw_helpers.h" +#include +#include +#include +#include namespace Conformance { + class EventQueue; + class EventReader; + class ISwapchainImageData; + RGBAImage CreateTextImage(int width, int height, const char* text, int fontHeight); XrPath StringToPath(XrInstance instance, const char* pathStr); @@ -409,6 +420,13 @@ namespace Conformance } m_viewSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_VIEW, XrPosef{{0, 0, 0, 1}, {0, 0, 0}}); + m_localSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_LOCAL, XrPosef{{0, 0, 0, 1}, {0, 0, 0}}); + + // description quad to the left, example image quad to the right, counter-rotated 15 degrees towards the viewer + m_descriptionQuadSpace = compositionHelper.CreateReferenceSpace( + XR_REFERENCE_SPACE_TYPE_VIEW, {Quat::FromAxisAngle(UpVector, 15 * MATH_PI / 180), {-0.5f, 0, -1.5f}}); + m_exampleQuadSpace = compositionHelper.CreateReferenceSpace( + XR_REFERENCE_SPACE_TYPE_VIEW, {Quat::FromAxisAngle(UpVector, -15 * MATH_PI / 180), {0.5f, 0, -1.5f}}); // Load example screenshot if available and set up the quad layer for it. { @@ -423,16 +441,14 @@ namespace Conformance } // Create a quad to the right of the help text. - m_exampleQuad = compositionHelper.CreateQuadLayer(exampleSwapchain, m_viewSpace, 1.25f, {Quat::Identity, {0.5f, 0, -1.5f}}); - XrQuaternionf_CreateFromAxisAngle(&m_exampleQuad->pose.orientation, &UpVector, -15 * MATH_PI / 180); + m_exampleQuad = compositionHelper.CreateQuadLayer(exampleSwapchain, m_exampleQuadSpace, 1.25f); } // Set up the quad layer for showing the help text to the left of the example image. m_descriptionQuad = compositionHelper.CreateQuadLayer( - m_compositionHelper.CreateStaticSwapchainImage(CreateTextImage(768, 768, descriptionText, 48)), m_viewSpace, 0.75f, - {Quat::Identity, {-0.5f, 0, -1.5f}}); + compositionHelper.CreateStaticSwapchainImage(CreateTextImage(768, 768, descriptionText, 48)), m_descriptionQuadSpace, + 0.75f); m_descriptionQuad->layerFlags |= XR_COMPOSITION_LAYER_BLEND_TEXTURE_SOURCE_ALPHA_BIT; - XrQuaternionf_CreateFromAxisAngle(&m_descriptionQuad->pose.orientation, &UpVector, 15 * MATH_PI / 180); constexpr uint32_t actionsWidth = 768, actionsHeight = 128; m_sceneActionsSwapchain = compositionHelper.CreateStaticSwapchainImage( @@ -454,17 +470,21 @@ namespace Conformance bool EndFrame(const XrFrameState& frameState, std::vector layers = {}) { - bool keepRunning = AppendLayers(layers); + bool keepRunning = AppendLayers(layers, frameState.predictedDisplayTime); keepRunning &= m_compositionHelper.PollEvents(); m_compositionHelper.EndFrame(frameState.predictedDisplayTime, std::move(layers)); return keepRunning; } private: - bool AppendLayers(std::vector& layers) + bool AppendLayers(std::vector& layers, XrTime predictedDisplayTime) { + LayerMode layerMode = GetLayerMode(); + LayerMode lastLayerMode = m_lastLayerMode; + m_lastLayerMode = layerMode; + // Add layer(s) based on the interaction mode. - switch (GetLayerMode()) { + switch (layerMode) { case LayerMode::Scene: m_actionsQuad->subImage = m_compositionHelper.MakeDefaultSubImage(m_sceneActionsSwapchain); layers.push_back(reinterpret_cast(m_actionsQuad)); @@ -475,6 +495,27 @@ namespace Conformance break; case LayerMode::Help: + if (lastLayerMode != LayerMode::Help) { + // convert a quad's reference space to local space when the help menu is opened + // this avoids view-locking the help, allowing the user to read it more naturally + auto placeQuad = [&](XrCompositionLayerQuad* quad, XrSpace quadSpace) { + XrSpaceLocation quadInLocalSpace{XR_TYPE_SPACE_LOCATION}; + XRC_CHECK_THROW_XRCMD(xrLocateSpace(quadSpace, m_localSpace, predictedDisplayTime, &quadInLocalSpace)); + if (quadInLocalSpace.locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT && + quadInLocalSpace.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) { + quad->space = m_localSpace; + quad->pose = quadInLocalSpace.pose; + } + else { + // xrLocateSpace didn't return a valid pose, fall back to view space + quad->space = quadSpace; + quad->pose = XrPosefCPP{}; + } + }; + placeQuad(m_descriptionQuad, m_descriptionQuadSpace); + placeQuad(m_exampleQuad, m_exampleQuadSpace); + } + layers.push_back(reinterpret_cast(m_descriptionQuad)); layers.push_back(reinterpret_cast(m_exampleQuad)); @@ -525,11 +566,15 @@ namespace Conformance XrAction m_menu{XR_NULL_HANDLE}; XrSpace m_viewSpace; + XrSpace m_localSpace; XrSwapchain m_sceneActionsSwapchain; XrSwapchain m_helpActionsSwapchain; + LayerMode m_lastLayerMode{LayerMode::Scene}; XrCompositionLayerQuad* m_actionsQuad; XrCompositionLayerQuad* m_descriptionQuad; + XrSpace m_descriptionQuadSpace; XrCompositionLayerQuad* m_exampleQuad; + XrSpace m_exampleQuadSpace; std::vector m_sceneLayers; }; } // namespace Conformance diff --git a/src/conformance/framework/conformance_framework.cpp b/src/conformance/framework/conformance_framework.cpp index f8e0094f..a4dcb037 100644 --- a/src/conformance/framework/conformance_framework.cpp +++ b/src/conformance/framework/conformance_framework.cpp @@ -15,25 +15,35 @@ // limitations under the License. #include "conformance_framework.h" + +#include "graphics_plugin.h" +#include "platform_plugin.h" #include "report.h" -#include "utils.h" #include "two_call_util.h" -#include -#include -#include -#include +#include "utilities/throw_helpers.h" +#include "utilities/utils.h" #include -#ifdef XR_USE_PLATFORM_ANDROID -#include -#include // for AWINDOW_FLAG_KEEP_SCREEN_ON -#include // for native window JNI -#include -#endif /// XR_USE_PLATFORM_ANDROID +#include +#include +#include +#include +#include +#include namespace Conformance { + + /// This list of instance extensions is safe to always enable if available. + static constexpr std::initializer_list kEnableIfAvailableInstanceExtensionNames = { + XR_KHR_COMPOSITION_LAYER_CUBE_EXTENSION_NAME, XR_KHR_COMPOSITION_LAYER_CYLINDER_EXTENSION_NAME}; + + /// The name of the required conformance layer + static constexpr auto kConformanceLayerName = "XR_APILAYER_KHRONOS_runtime_conformance"; + + static constexpr auto kGetSystemPollingTimeout = std::chrono::seconds(10); + static std::unique_ptr globalDataInstance; void ResetGlobalData() @@ -77,6 +87,8 @@ namespace Conformance AppendSprintf(result, " fileLineLoggingEnabled: %s\n", fileLineLoggingEnabled ? "yes" : "no"); + AppendSprintf(result, " pollGetSystem: %s\n", pollGetSystem ? "yes" : "no"); + AppendSprintf(result, " debugMode: %s", debugMode ? "yes" : "no"); return result; @@ -92,15 +104,15 @@ namespace Conformance XR_VERSION_PATCH(apiVersion)); AppendSprintf(reportString, "Graphics system: %s\n", globalData.options.graphicsPlugin.c_str()); AppendSprintf(reportString, "Present API layers:\n"); - for (const std::string& apiLayerName : globalData.enabledAPILayerNames) { - AppendSprintf(reportString, " %s\n", apiLayerName.c_str()); + for (const char* const& apiLayerName : globalData.enabledAPILayerNames) { + AppendSprintf(reportString, " %s\n", apiLayerName); } if (globalData.enabledAPILayerNames.empty()) { AppendSprintf(reportString, " \n"); } AppendSprintf(reportString, "Tested instance extensions:\n"); - for (const std::string& extensionName : globalData.enabledInstanceExtensionNames) { - AppendSprintf(reportString, " %s\n", extensionName.c_str()); + for (const char* const& extensionName : globalData.enabledInstanceExtensionNames) { + AppendSprintf(reportString, " %s\n", extensionName); } if (globalData.enabledInstanceExtensionNames.empty()) { AppendSprintf(reportString, " \n"); @@ -187,12 +199,11 @@ namespace Conformance availableAPILayerNames.emplace_back(value.layerName); } - static const auto CONFORMANCE_LAYER_NAME = "XR_APILAYER_KHRONOS_runtime_conformance"; const auto e = availableAPILayerNames.end(); - bool hasConfLayer = (e != std::find(availableAPILayerNames.begin(), e, CONFORMANCE_LAYER_NAME)); + bool hasConfLayer = (e != std::find(availableAPILayerNames.begin(), e, kConformanceLayerName)); if (hasConfLayer) { if (!globalData.options.invalidHandleValidation) { - enabledAPILayerNames.push_back_unique(CONFORMANCE_LAYER_NAME); + enabledAPILayerNames.push_back_unique(kConformanceLayerName); useDebugMessenger = true; } else { @@ -223,12 +234,8 @@ namespace Conformance availableInstanceExtensionNames.emplace_back(value.extensionName); } - // This list of instance extensions is safe to always enable if available. - std::vector enableIfAvailableInstanceExtensionNames = {XR_KHR_COMPOSITION_LAYER_CUBE_EXTENSION_NAME, - XR_KHR_COMPOSITION_LAYER_CYLINDER_EXTENSION_NAME}; - - for (auto& value : enableIfAvailableInstanceExtensionNames) { - auto& avail = availableInstanceExtensionNames; + for (auto& value : kEnableIfAvailableInstanceExtensionNames) { + const auto& avail = availableInstanceExtensionNames; if (std::find(avail.begin(), avail.end(), value) != avail.end()) { enabledInstanceExtensionNames.push_back_unique(value); } @@ -282,6 +289,77 @@ namespace Conformance return false; } + // Find XrSystemId (for later use and to ensure device is connected/available for whatever that means in a given runtime) + XrSystemId systemId = XR_NULL_SYSTEM_ID; + const XrSystemGetInfo systemGetInfo = {XR_TYPE_SYSTEM_GET_INFO, nullptr, options.formFactorValue}; + + auto tryGetSystem = [&] { + XrResult result = xrGetSystem(autoInstance, &systemGetInfo, &systemId); + if (result != XR_SUCCESS && result != XR_ERROR_FORM_FACTOR_UNAVAILABLE) { + // Anything else is a real error + ReportF("GlobalData::Initialize: xrGetSystem failed with result: %s.", ResultToString(result)); + return false; + } + return true; + }; + + if (options.pollGetSystem) { + ReportF( + "GlobalData::Initialize: xrGetSystem will be polled until success or timeout, as requested. This behavior may be less compatible with applications."); + + const auto timeout = std::chrono::steady_clock::now() + kGetSystemPollingTimeout; + while (systemId == XR_NULL_SYSTEM_ID && std::chrono::steady_clock::now() < timeout) { + if (!tryGetSystem()) { + return false; + } + // pause briefly before trying again + std::this_thread::sleep_for(std::chrono::milliseconds{50}); + } + + if (systemId == XR_NULL_SYSTEM_ID) { + ReportF("GlobalData::Initialize: xrGetSystem polling timed out without success after %f", + std::chrono::duration_cast>(kGetSystemPollingTimeout).count()); + return false; + } + } + else { + // just try once + if (!tryGetSystem()) { + return false; + } + if (systemId == XR_NULL_SYSTEM_ID) { + ReportF("GlobalData::Initialize: xrGetSystem did not return a system ID on the first call, not proceeding with tests."); + return false; + } + } + + // Find available blend modes + result = doTwoCallInPlace(availableBlendModes, xrEnumerateEnvironmentBlendModes, autoInstance.GetInstance(), systemId, + options.viewConfigurationValue); + if (XR_FAILED(result)) { + ReportF("GlobalData::Initialize: xrEnumerateEnvironmentBlendModes failed with result: %s", ResultToString(result)); + return false; + } + if (options.environmentBlendMode.empty()) { + // Default to the first enumerated blend mode + options.environmentBlendModeValue = availableBlendModes.front(); + // convert to string, indicating auto selection + switch (options.environmentBlendModeValue) { + case XR_ENVIRONMENT_BLEND_MODE_OPAQUE: + options.environmentBlendMode = "opaque (auto-selected)"; + break; + case XR_ENVIRONMENT_BLEND_MODE_ADDITIVE: + options.environmentBlendMode = "additive (auto-selected)"; + break; + case XR_ENVIRONMENT_BLEND_MODE_ALPHA_BLEND: + options.environmentBlendMode = "alphablend (auto-selected)"; + break; + default: + XRC_THROW("Got unrecognized environment blend mode value as the front of the enumerated list."); + break; + } + } + isInitialized = true; return true; } diff --git a/src/conformance/framework/conformance_framework.h b/src/conformance/framework/conformance_framework.h index ee9f7223..20bc7f7e 100644 --- a/src/conformance/framework/conformance_framework.h +++ b/src/conformance/framework/conformance_framework.h @@ -16,19 +16,26 @@ #pragma once -#include -#include -#include -#include -#include -#include +#include "conformance_utils.h" +#include "utilities/stringification.h" +#include "utilities/types_and_constants.h" +#include "utilities/utils.h" + #include #include -#include "utils.h" -#include "conformance_utils.h" -#include "platform_plugin.h" -#include "graphics_plugin.h" + #include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include #ifdef XR_USE_PLATFORM_WIN32 #include "windows.h" @@ -112,12 +119,6 @@ void Conformance_Android_Detach_Current_Thread(); /// @} -#define XRC_CHECK_STRINGIFY(x) #x -#define XRC_TO_STRING(x) XRC_CHECK_STRINGIFY(x) - -/// Represents a compile time file and line location as a single string. -#define XRC_FILE_AND_LINE __FILE__ ":" XRC_TO_STRING(__LINE__) - #if defined(XR_USE_PLATFORM_ANDROID) #define ATTACH_THREAD Conformance_Android_Attach_Current_Thread() #define DETACH_THREAD Conformance_Android_Detach_Current_Thread() @@ -133,6 +134,8 @@ void Conformance_Android_Detach_Current_Thread(); namespace Conformance { + struct IGraphicsPlugin; + struct IPlatformPlugin; /// Specifies runtime options for the application. /// String options are case-insensitive. /// Each of these can be specified from the command line via a command of the same name as @@ -168,9 +171,9 @@ namespace Conformance XrViewConfigurationType viewConfigurationValue{XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO}; /// Options include "opaque" "additive" "alphablend". See enum XrEnvironmentBlendMode. - /// Default is opaque. - std::string environmentBlendMode{"Opaque"}; - XrEnvironmentBlendMode environmentBlendModeValue{XR_ENVIRONMENT_BLEND_MODE_OPAQUE}; + /// Default is the first enumerated value + std::string environmentBlendMode{}; + XrEnvironmentBlendMode environmentBlendModeValue{(XrEnvironmentBlendMode)0}; /// Options can vary depending on their platform availability. If a requested API layer is /// not supported then the test fails. @@ -206,6 +209,10 @@ namespace Conformance /// Default is true (enabled). bool fileLineLoggingEnabled{true}; + /// If true then xrGetSystem will be attempted repeatedly for a limited time at the beginning of a run + /// before beginning a test case. + bool pollGetSystem{false}; + /// Defines if executing in debug mode. By default this follows the build type. bool debugMode { @@ -418,6 +425,9 @@ namespace Conformance /// The interaction profiles that have been requested to be tested. StringVec enabledInteractionProfiles; + /// The environment blend modes available for the view configuration type. + std::vector availableBlendModes; + /// Whether each controller is to be used during testing bool leftHandUnderTest{false}; bool rightHandUnderTest{false}; diff --git a/src/conformance/framework/conformance_utils.cpp b/src/conformance/framework/conformance_utils.cpp index eb3f4557..3a90faff 100644 --- a/src/conformance/framework/conformance_utils.cpp +++ b/src/conformance/framework/conformance_utils.cpp @@ -14,29 +14,29 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include - -#include "utils.h" -#include "report.h" -#include "two_call_util.h" -#include "conformance_utils.h" #include "conformance_framework.h" +#include "conformance_utils.h" #include "graphics_plugin.h" -#include "throw_helpers.h" +#include "platform_plugin.h" +#include "two_call_util.h" +#include "utilities/throw_helpers.h" +#include "utilities/utils.h" -#include "xr_dependencies.h" -#include +#include #include -#include +#include +#include + #include -#include #include +#include #include -#include -#include -#include +#include +#include +#include #include +#include #ifdef _WIN32 #include @@ -51,14 +51,6 @@ namespace Conformance { - // We keep our own copy of this as opposed to calling the xrResultToString function, because our - // purpose here it to validate the runtime's implementation of xrResultToString. - - const ResultStringMap& GetResultStringMap() - { - static const ResultStringMap resultStringMap{XR_LIST_ENUM_XrResult(XRC_ENUM_NAME_PAIR)}; - return resultStringMap; - } const std::map GetNumberExtensionMap() { @@ -67,17 +59,6 @@ namespace Conformance return myMap; } - const char* ResultToString(XrResult result) - { - auto it = GetResultStringMap().find(result); - - if (it != GetResultStringMap().end()) { - return it->second; - } - - return ""; - } - std::string PathToString(XrInstance instance, XrPath path) { uint32_t count = 0; @@ -1215,4 +1196,16 @@ namespace Conformance return result; } + + std::ostream& operator<<(std::ostream& os, AutoBasicInstance const& inst) + { + OutputHandle(os, inst.GetInstance()); + return os; + } + + std::ostream& operator<<(std::ostream& os, AutoBasicSession const& sess) + { + OutputHandle(os, sess.GetSession()); + return os; + } } // namespace Conformance diff --git a/src/conformance/framework/conformance_utils.h b/src/conformance/framework/conformance_utils.h index 7174f343..d770f113 100644 --- a/src/conformance/framework/conformance_utils.h +++ b/src/conformance/framework/conformance_utils.h @@ -17,23 +17,20 @@ #pragma once +#include "utilities/types_and_constants.h" +#include "utilities/xrduration_literals.h" + #include -#include -#include + #include -#include -#include -#include -#include #include +#include +#include #include - -#include -#include - -#include "utils.h" -#include "types_and_constants.h" -#include "event_reader.h" +#include +#include +#include +#include namespace Conformance { @@ -42,25 +39,6 @@ namespace Conformance // Forward declarations struct IGraphicsPlugin; - /// We keep a private auto-generated map of all results and their string versions. - typedef std::map ResultStringMap; - - const ResultStringMap& GetResultStringMap(); - - /// @addtogroup cts_framework - /// @{ - - /// Returns a string for a given XrResult, based on our accounting of the result strings, and not - /// based on the xrResultToString function. - /// Returns "" if the result is not recognized. - /// - /// Example usage: - /// ``` - /// XrResult result = xrPollEvent(instance, &eventData); - /// printf("%d: %s, resut, ResultToString(result)); - /// ``` - const char* ResultToString(XrResult result); - /// PathToString /// /// Returns a string for a given XrPath if it exists, else "" @@ -389,42 +367,6 @@ namespace Conformance } } - /// Example usage: - /// ``` - /// XrDuration timeout = 10_xrSeconds; - /// ``` - inline constexpr XrDuration operator"" _xrSeconds(unsigned long long value) - { - return (static_cast(value) * 1000 * 1000 * 1000); // Convert seconds to XrDuration nanoseconds. - } - - /// Example usage: - /// ``` - /// XrDuration timeout = 10_xrMilliseconds; - /// ``` - inline constexpr XrDuration operator"" _xrMilliseconds(unsigned long long value) - { - return (static_cast(value) * 1000 * 1000); // Convert milliseconds to XrDuration nanoseconds. - } - - /// Example usage: - /// ``` - /// XrDuration timeout = 10_xrMicroseconds; - /// ``` - inline constexpr XrDuration operator"" _xrMicroseconds(unsigned long long value) - { - return (static_cast(value) * 1000); // Convert microseconds to XrDuration nanoseconds. - } - - /// Example usage: - /// ``` - /// XrDuration timeout = 10_xrNanoseconds; - /// ``` - inline constexpr XrDuration operator"" _xrNanoseconds(unsigned long long value) - { - return static_cast(value); // XrDuration is already in nanoseconds - } - /// Implements a single-run stopwatch using std::chrono. class Stopwatch { @@ -565,6 +507,10 @@ namespace Conformance XrSystemId systemId{XR_NULL_SYSTEM_ID}; }; + /// Output operator for the `XrInstance` handle in a @ref AutoBasicInstance + /// @relates AutoBasicInstance + std::ostream& operator<<(std::ostream& os, AutoBasicInstance const& inst); + /// Finds an XrSystemId suitable for testing of additional functionality. XrResult FindBasicSystem(XrInstance instance, XrSystemId* systemId); @@ -696,6 +642,10 @@ namespace Conformance std::vector environmentBlendModeVector; }; + /// Output operator for the `XrSession` handle in a @ref AutoBasicSession + /// @relates AutoBasicSession + std::ostream& operator<<(std::ostream& os, AutoBasicSession const& sess); + bool WaitUntilPredicateWithTimeout(const std::function& predicate, const std::chrono::nanoseconds timeout, const std::chrono::nanoseconds delay); diff --git a/src/conformance/framework/graphics_plugin.h b/src/conformance/framework/graphics_plugin.h index 100b8dd7..a26b00f4 100644 --- a/src/conformance/framework/graphics_plugin.h +++ b/src/conformance/framework/graphics_plugin.h @@ -18,7 +18,7 @@ #include "platform_plugin.h" #include "conformance_utils.h" -#include "Geometry.h" +#include "utilities/Geometry.h" #include "RGBAImage.h" #include diff --git a/src/conformance/framework/graphics_plugin_d3d11.cpp b/src/conformance/framework/graphics_plugin_d3d11.cpp index 16dc3ca8..6b2ac971 100644 --- a/src/conformance/framework/graphics_plugin_d3d11.cpp +++ b/src/conformance/framework/graphics_plugin_d3d11.cpp @@ -14,28 +14,30 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "graphics_plugin.h" - #if defined(XR_USE_GRAPHICS_API_D3D11) && !defined(MISSING_DIRECTX_COLORS) -#include "swapchain_image_data.h" -#include "swapchain_parameters.h" -#include "graphics_plugin_impl_helpers.h" +#include "graphics_plugin.h" +#include "common/xr_linear.h" #include "conformance_framework.h" -#include "throw_helpers.h" -#include "Geometry.h" +#include "graphics_plugin_impl_helpers.h" +#include "swapchain_image_data.h" +#include "utilities/Geometry.h" +#include "utilities/d3d_common.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" + +#include + +#include + +#include +#include +#include #include #include // For Microsoft::WRL::ComPtr -#include -#include -#include -#include -#include + #include #include -#include - -#include "d3d_common.h" using namespace Microsoft::WRL; using namespace DirectX; @@ -288,7 +290,7 @@ namespace Conformance GetInstanceExtensionFunction(instance, "xrGetD3D11GraphicsRequirementsKHR"); XrResult result = xrGetD3D11GraphicsRequirementsKHR(instance, systemId, &graphicsRequirements); - CHECK(ValidateResultAllowed("xrGetD3D11GraphicsRequirementsKHR", result)); + XRC_CHECK_THROW(ValidateResultAllowed("xrGetD3D11GraphicsRequirementsKHR", result)); if (XR_FAILED(result)) { // Log result? return false; @@ -480,6 +482,7 @@ namespace Conformance bool D3D11GraphicsPlugin::ValidateSwapchainImages(int64_t /*imageFormat*/, const SwapchainCreateTestParameters* tp, XrSwapchain swapchain, uint32_t* imageCount) const noexcept(false) { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. *imageCount = 0; // Zero until set below upon success. std::vector swapchainImageVector; diff --git a/src/conformance/framework/graphics_plugin_d3d12.cpp b/src/conformance/framework/graphics_plugin_d3d12.cpp index 5ffb45b8..e24c89c8 100644 --- a/src/conformance/framework/graphics_plugin_d3d12.cpp +++ b/src/conformance/framework/graphics_plugin_d3d12.cpp @@ -14,49 +14,39 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "graphics_plugin.h" - #if defined(XR_USE_GRAPHICS_API_D3D12) && !defined(MISSING_DIRECTX_COLORS) -#include "swapchain_image_data.h" -#include "swapchain_parameters.h" +#include "graphics_plugin.h" +#include "common/xr_linear.h" #include "conformance_framework.h" #include "graphics_plugin_impl_helpers.h" -#include "Geometry.h" -#include "throw_helpers.h" +#include "swapchain_image_data.h" +#include "utilities/Geometry.h" +#include "utilities/align_to.h" +#include "utilities/array_size.h" +#include "utilities/d3d_common.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" + +#include + +#include +#include +#include #include #include // For Microsoft::WRL::ComPtr -#include -#include -#include + #include + #include #include #include -#include - -#include -#include "d3d_common.h" using namespace Microsoft::WRL; using namespace DirectX; namespace Conformance { - // The equivalent of C++17 std::size. A helper to get the dimension for an array. - template - constexpr size_t ArraySize(const T (&/*unused*/)[Size]) noexcept - { - return Size; - } - - template - constexpr uint32_t AlignTo(uint32_t n) - { - static_assert((alignment & (alignment - 1)) == 0, "The alignment must be power-of-two"); - return (n + alignment - 1) & ~(alignment - 1); - } - ComPtr CreateBuffer(ID3D12Device* d3d12Device, uint32_t size, D3D12_HEAP_TYPE heapType) { D3D12_RESOURCE_STATES d3d12ResourceState; @@ -154,7 +144,7 @@ namespace Conformance } XRC_CHECK_THROW_HRCMD(cmdList->Close()); - CHECK(ExecuteCommandList(cmdList.Get())); + XRC_CHECK_THROW(ExecuteCommandList(cmdList.Get())); } }; @@ -476,7 +466,7 @@ namespace Conformance GetInstanceExtensionFunction(instance, "xrGetD3D12GraphicsRequirementsKHR"); XrResult result = xrGetD3D12GraphicsRequirementsKHR(instance, systemId, &graphicsRequirements); - CHECK(ValidateResultAllowed("xrGetD3D12GraphicsRequirementsKHR", result)); + XRC_CHECK_THROW(ValidateResultAllowed("xrGetD3D12GraphicsRequirementsKHR", result)); if (FAILED(result)) { // Log result? return false; @@ -568,7 +558,7 @@ namespace Conformance XRC_CHECK_THROW_HRCMD(fence->SetName(L"CTS fence")); fenceEvent = ::CreateEvent(nullptr, FALSE, FALSE, nullptr); - CHECK(fenceEvent != nullptr); + XRC_CHECK_THROW(fenceEvent != nullptr); m_cubeMesh = MakeCubeMesh(); @@ -665,7 +655,7 @@ namespace Conformance cmdList->CopyTextureRegion(&dstLocation, 0 /* X */, 0 /* Y */, 0 /* Z */, &srcLocation, nullptr); XRC_CHECK_THROW_HRCMD(cmdList->Close()); - CHECK(ExecuteCommandList(cmdList.Get())); + XRC_CHECK_THROW(ExecuteCommandList(cmdList.Get())); WaitForGpu(); } @@ -689,6 +679,7 @@ namespace Conformance bool D3D12GraphicsPlugin::ValidateSwapchainImages(int64_t /*imageFormat*/, const SwapchainCreateTestParameters* tp, XrSwapchain swapchain, uint32_t* imageCount) const noexcept(false) { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. *imageCount = 0; // Zero until set below upon success. std::vector swapchainImageVector; @@ -754,6 +745,7 @@ namespace Conformance bool D3D12GraphicsPlugin::ValidateSwapchainImageState(XrSwapchain swapchain, uint32_t index, int64_t imageFormat) const { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. std::vector swapchainImageVector; uint32_t countOutput; @@ -978,7 +970,7 @@ namespace Conformance cmdList->ClearDepthStencilView(depthStencilView, D3D12_CLEAR_FLAG_DEPTH, 1.0f, 0, 0, nullptr); XRC_CHECK_THROW_HRCMD(cmdList->Close()); - CHECK(ExecuteCommandList(cmdList.Get())); + XRC_CHECK_THROW(ExecuteCommandList(cmdList.Get())); } inline MeshHandle D3D12GraphicsPlugin::MakeSimpleMesh(span idx, span vtx) @@ -1122,7 +1114,7 @@ namespace Conformance } XRC_CHECK_THROW_HRCMD(cmdList->Close()); - CHECK(ExecuteCommandList(cmdList.Get())); + XRC_CHECK_THROW(ExecuteCommandList(cmdList.Get())); // TODO: Track down exactly why this wait is needed. // On some drivers and/or hardware the test is generating the same image for the left and right eye, diff --git a/src/conformance/framework/graphics_plugin_factory.cpp b/src/conformance/framework/graphics_plugin_factory.cpp index 4b56f99e..08e7dc06 100644 --- a/src/conformance/framework/graphics_plugin_factory.cpp +++ b/src/conformance/framework/graphics_plugin_factory.cpp @@ -15,7 +15,7 @@ // limitations under the License. #include "graphics_plugin.h" -#include "utils.h" +#include "utilities/utils.h" #include namespace Conformance diff --git a/src/conformance/framework/graphics_plugin_opengl.cpp b/src/conformance/framework/graphics_plugin_opengl.cpp index 32807ed3..5bdc4881 100644 --- a/src/conformance/framework/graphics_plugin_opengl.cpp +++ b/src/conformance/framework/graphics_plugin_opengl.cpp @@ -14,32 +14,28 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include "graphics_plugin.h" - #ifdef XR_USE_GRAPHICS_API_OPENGL +#include "common/xr_linear.h" +#include "conformance_framework.h" +#include "graphics_plugin.h" +#include "graphics_plugin_impl_helpers.h" #include "report.h" #include "swapchain_image_data.h" -#include "graphics_plugin_impl_helpers.h" -#include "swapchain_parameters.h" - +#include "utilities/Geometry.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" #include "xr_dependencies.h" -#include "conformance_framework.h" -#include "throw_helpers.h" -#include "Geometry.h" - #include #include #include + #include #include +#include -// Why was this needed? hello_xr doesn't need these #defines -//#include "graphics_plugin_opengl_loader.h" #include "gfxwrapper_opengl.h" -#include "xr_linear.h" // clang-format off @@ -605,6 +601,8 @@ namespace Conformance XrGraphicsBindingOpenGLXlibKHR graphicsBinding{XR_TYPE_GRAPHICS_BINDING_OPENGL_XLIB_KHR}; #elif defined(XR_USE_PLATFORM_XCB) XrGraphicsBindingOpenGLXcbKHR graphicsBinding{XR_TYPE_GRAPHICS_BINDING_OPENGL_XCB_KHR}; +#elif defined(XR_USE_PLATFORM_MACOS) +#error OpenGL bindings for Mac have not been implemented #else #error "Platform not (yet) supported." #endif @@ -691,7 +689,7 @@ namespace Conformance GetInstanceExtensionFunction(instance, "xrGetOpenGLGraphicsRequirementsKHR"); XrResult result = xrGetOpenGLGraphicsRequirementsKHR(instance, systemId, &graphicsRequirements); - CHECK(ValidateResultAllowed("xrGetOpenGLGraphicsRequirementsKHR", result)); + XRC_CHECK_THROW(ValidateResultAllowed("xrGetOpenGLGraphicsRequirementsKHR", result)); if (XR_FAILED(result)) { // Log result? return false; @@ -723,26 +721,28 @@ namespace Conformance graphicsBinding.hDC = window.context.hDC; graphicsBinding.hGLRC = window.context.hGLRC; #elif defined(XR_USE_PLATFORM_XLIB) - REQUIRE(window.context.xDisplay != nullptr); + XRC_CHECK_THROW(window.context.xDisplay != nullptr); graphicsBinding.xDisplay = window.context.xDisplay; graphicsBinding.visualid = window.context.visualid; graphicsBinding.glxFBConfig = window.context.glxFBConfig; graphicsBinding.glxDrawable = window.context.glxDrawable; graphicsBinding.glxContext = window.context.glxContext; #elif defined(XR_USE_PLATFORM_XCB) - REQUIRE(window.context.connection != nullptr); + XRC_CHECK_THROW(window.context.connection != nullptr); graphicsBinding.connection = window.context.connection; graphicsBinding.screenNumber = window.context.screen_number; graphicsBinding.fbconfigid = window.context.fbconfigid; graphicsBinding.visualid = window.context.visualid; graphicsBinding.glxDrawable = window.context.glxDrawable; graphicsBinding.glxContext = window.context.glxContext; +#elif defined(XR_USE_PLATFORM_MACOS) +#error OpenGL bindings for Mac have not been implemented #else #error "Platform not (yet) supported." #endif GLenum error = glGetError(); - CHECK(error == GL_NO_ERROR); + XRC_CHECK_THROW(error == GL_NO_ERROR); GLint major, minor; glGetIntegerv(GL_MAJOR_VERSION, &major); @@ -810,6 +810,7 @@ namespace Conformance ReportF("GL %s (0x%x): %s", sev, id, message); #else (void)severity; + (void)message; #endif // !defined(OS_APPLE_MACOS) } @@ -1107,6 +1108,7 @@ namespace Conformance bool OpenGLGraphicsPlugin::ValidateSwapchainImages(int64_t imageFormat, const SwapchainCreateTestParameters* tp, XrSwapchain swapchain, uint32_t* imageCount) const { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. *imageCount = 0; // Zero until set below upon success. std::vector swapchainImageVector; diff --git a/src/conformance/framework/graphics_plugin_opengles.cpp b/src/conformance/framework/graphics_plugin_opengles.cpp index 705fbc28..e6800012 100644 --- a/src/conformance/framework/graphics_plugin_opengles.cpp +++ b/src/conformance/framework/graphics_plugin_opengles.cpp @@ -18,22 +18,32 @@ #ifdef XR_USE_GRAPHICS_API_OPENGL_ES -#include "swapchain_image_data.h" -#include "swapchain_parameters.h" -#include "graphics_plugin_impl_helpers.h" -#include "conformance_framework.h" -#include "Geometry.h" #include "common/gfxwrapper_opengl.h" -#include +#include "common/xr_linear.h" +#include "conformance_framework.h" +#include "graphics_plugin_impl_helpers.h" +#include "report.h" +#include "swapchain_image_data.h" +#include "utilities/Geometry.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" #include "xr_dependencies.h" + +#include #include #include -#include -#include -#include -#include -#include -#include "report.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include #define GL(glcmd) \ { \ @@ -50,6 +60,7 @@ namespace Conformance { + struct IPlatformPlugin; static const char* VertexShaderGlsl = R"_( #version 320 es @@ -370,7 +381,7 @@ namespace Conformance std::tie(swapchainData, imageIndex) = m_swapchainImageDataMap.GetDataAndIndexFromBasePointer(swapchainImage); // auto imageInfoIt = m_imageInfo.find(swapchainImage); - // CHECK(imageInfoIt != m_imageInfo.end()); + // XRC_CHECK_THROW(imageInfoIt != m_imageInfo.end()); // SwapchainInfo& swapchainInfo = m_swapchainInfo[imageInfoIt->second.swapchainIndex]; // GLuint arraySize = swapchainInfo.createInfo.arraySize; @@ -414,7 +425,7 @@ namespace Conformance GetInstanceExtensionFunction(instance, "xrGetOpenGLESGraphicsRequirementsKHR"); XrResult result = xrGetOpenGLESGraphicsRequirementsKHR(instance, systemId, &graphicsRequirements); - CHECK(ValidateResultAllowed("xrGetOpenGLESGraphicsRequirementsKHR", result)); + XRC_CHECK_THROW(ValidateResultAllowed("xrGetOpenGLESGraphicsRequirementsKHR", result)); if (XR_FAILED(result)) { // Log result? return false; @@ -443,8 +454,8 @@ namespace Conformance // Initialize the binding once we have a context { - REQUIRE(window.display != EGL_NO_DISPLAY); - REQUIRE(window.context.context != EGL_NO_CONTEXT); + XRC_CHECK_THROW(window.display != EGL_NO_DISPLAY); + XRC_CHECK_THROW(window.context.context != EGL_NO_CONTEXT); graphicsBinding = {XR_TYPE_GRAPHICS_BINDING_OPENGL_ES_ANDROID_KHR}; graphicsBinding.display = window.display; graphicsBinding.config = (EGLConfig)0; @@ -452,7 +463,7 @@ namespace Conformance } GLenum error = glGetError(); - CHECK(error == GL_NO_ERROR); + XRC_CHECK_THROW(error == GL_NO_ERROR); GLint major, minor; GL(glGetIntegerv(GL_MAJOR_VERSION, &major)); @@ -859,6 +870,7 @@ namespace Conformance bool OpenGLESGraphicsPlugin::ValidateSwapchainImages(int64_t imageFormat, const SwapchainCreateTestParameters* tp, XrSwapchain swapchain, uint32_t* imageCount) const { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. *imageCount = 0; // Zero until set below upon success. std::vector swapchainImageVector; diff --git a/src/conformance/framework/graphics_plugin_vulkan.cpp b/src/conformance/framework/graphics_plugin_vulkan.cpp index 63759e61..92ca5ddc 100644 --- a/src/conformance/framework/graphics_plugin_vulkan.cpp +++ b/src/conformance/framework/graphics_plugin_vulkan.cpp @@ -14,33 +14,46 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include #include "graphics_plugin.h" -#include "openxr/openxr.h" #ifdef XR_USE_GRAPHICS_API_VULKAN -#include "vulkan_utils.h" - -#include -#include -#include +#include "RGBAImage.h" +#include "common/hex_and_handles.h" +#include "common/xr_linear.h" +#include "graphics_plugin_impl_helpers.h" #include "report.h" -#include "hex_and_handles.h" #include "swapchain_image_data.h" -#include "graphics_plugin_impl_helpers.h" -#include "throw_helpers.h" -#include "swapchain_parameters.h" -#include "conformance_framework.h" +#include "utilities/Geometry.h" +#include "utilities/swapchain_parameters.h" +#include "utilities/throw_helpers.h" +#include "utilities/vulkan_utils.h" #include "xr_dependencies.h" -#include "Geometry.h" -#include + +#include #include +#include +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#ifdef USE_CHECKPOINTS +#include +#endif namespace Conformance { + struct IPlatformPlugin; #ifdef USE_ONLINE_VULKAN_SHADERC constexpr char VertexShaderGlsl[] = @@ -349,7 +362,7 @@ namespace Conformance break; } - CHECK(foundFmt < surfFmtCount); + XRC_CHECK_THROW(foundFmt < surfFmtCount); uint32_t presentModeCount = 0; XRC_CHECK_THROW_VKCMD(vkGetPhysicalDeviceSurfacePresentModesKHR(m_vkPhysicalDevice, surface, &presentModeCount, nullptr)); @@ -367,7 +380,7 @@ namespace Conformance VkBool32 presentable = false; XRC_CHECK_THROW_VKCMD(vkGetPhysicalDeviceSurfaceSupportKHR(m_vkPhysicalDevice, m_queueFamilyIndex, surface, &presentable)); - CHECK(presentable); + XRC_CHECK_THROW(presentable); VkSwapchainCreateInfoKHR swapchainInfo{VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR}; swapchainInfo.flags = 0; @@ -1623,6 +1636,7 @@ namespace Conformance bool VulkanGraphicsPlugin::ValidateSwapchainImages(int64_t /*imageFormat*/, const SwapchainCreateTestParameters* /*tp*/, XrSwapchain swapchain, uint32_t* imageCount) const noexcept(false) { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. *imageCount = 0; // Zero until set below upon success. std::vector swapchainImageVector; @@ -1666,6 +1680,7 @@ namespace Conformance bool VulkanGraphicsPlugin::ValidateSwapchainImageState(XrSwapchain /*swapchain*/, uint32_t /*index*/, int64_t /*imageFormat*/) const { + // OK to use CHECK and REQUIRE in here because this is always called from within a test. // TODO: check VK_ACCESS_*? Mandate this in the spec? return true; } diff --git a/src/conformance/framework/input_testinputdevice.cpp b/src/conformance/framework/input_testinputdevice.cpp index 33e085d6..53f34765 100644 --- a/src/conformance/framework/input_testinputdevice.cpp +++ b/src/conformance/framework/input_testinputdevice.cpp @@ -14,20 +14,28 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include -#include -#include -#include - -#include - #include "input_testinputdevice.h" + +#include "composition_utils.h" #include "conformance_framework.h" #include "conformance_utils.h" - -#include "throw_helpers.h" #include "two_call.h" -#include "types_and_constants.h" +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" + +#include + +#include + +#include +#include +#include +#include +#include +#include +#include +#include +#include using namespace std::chrono_literals; diff --git a/src/conformance/framework/input_testinputdevice.h b/src/conformance/framework/input_testinputdevice.h index 252ab08e..c42ea52d 100644 --- a/src/conformance/framework/input_testinputdevice.h +++ b/src/conformance/framework/input_testinputdevice.h @@ -16,11 +16,17 @@ #pragma once +#include + #include -#include "composition_utils.h" +#include +#include +#include +#include namespace Conformance { + struct InteractionManager; struct InputSourcePathData { const char* Path; diff --git a/src/conformance/framework/mesh_projection_layer.cpp b/src/conformance/framework/mesh_projection_layer.cpp index 5208dfcb..af3990f6 100644 --- a/src/conformance/framework/mesh_projection_layer.cpp +++ b/src/conformance/framework/mesh_projection_layer.cpp @@ -16,8 +16,18 @@ #include "mesh_projection_layer.h" #include "composition_utils.h" +#include "conformance_framework.h" #include "graphics_plugin.h" +#include "utilities/utils.h" +#include "nonstd/span.hpp" + +#include + +#include +#include +#include +#include #include namespace diff --git a/src/conformance/framework/mesh_projection_layer.h b/src/conformance/framework/mesh_projection_layer.h index 5a2af938..68812140 100644 --- a/src/conformance/framework/mesh_projection_layer.h +++ b/src/conformance/framework/mesh_projection_layer.h @@ -16,16 +16,12 @@ #pragma once -#include -#include -#include -#include "utils.h" -#include "report.h" -#include "conformance_utils.h" -#include "conformance_framework.h" #include "composition_utils.h" +#include "graphics_plugin.h" + #include -#include + +#include namespace Conformance { diff --git a/src/conformance/framework/platform_plugin_android.cpp b/src/conformance/framework/platform_plugin_android.cpp index d59641e6..ec094582 100644 --- a/src/conformance/framework/platform_plugin_android.cpp +++ b/src/conformance/framework/platform_plugin_android.cpp @@ -18,28 +18,16 @@ #ifdef XR_USE_PLATFORM_ANDROID -#include -#include -#include // for AWINDOW_FLAG_KEEP_SCREEN_ON -#include // for native window JNI - #include "conformance_framework.h" - -#ifdef XR_USE_GRAPHICS_API_VULKAN -#define VK_USE_PLATFORM_ANDROID_KHR -#include -#endif /// XR_USE_GRAPHICS_API_VULKAN - -#ifdef XR_USE_GRAPHICS_API_OPENGL_ES -#include -#include -#include -#include -#endif /// XR_USE_GRAPHICS_API_OPENGL_ES +#include "xr_dependencies.h" #include #include +#include +#include +#include + namespace Conformance { diff --git a/src/conformance/framework/platform_plugin_posix.cpp b/src/conformance/framework/platform_plugin_posix.cpp index ba8fbec4..776768b9 100644 --- a/src/conformance/framework/platform_plugin_posix.cpp +++ b/src/conformance/framework/platform_plugin_posix.cpp @@ -14,10 +14,10 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "platform_plugin.h" - #if defined(XR_OS_APPLE) || defined(XR_OS_LINUX) +#include "platform_plugin.h" + namespace Conformance { diff --git a/src/conformance/framework/platform_plugin_win32.cpp b/src/conformance/framework/platform_plugin_win32.cpp index c0deb958..ca521433 100644 --- a/src/conformance/framework/platform_plugin_win32.cpp +++ b/src/conformance/framework/platform_plugin_win32.cpp @@ -14,10 +14,10 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "platform_plugin.h" - #ifdef XR_USE_PLATFORM_WIN32 +#include "platform_plugin.h" + #include namespace Conformance diff --git a/src/conformance/framework/swapchain_image_data.cpp b/src/conformance/framework/swapchain_image_data.cpp index 1d9188b4..c1631dd2 100644 --- a/src/conformance/framework/swapchain_image_data.cpp +++ b/src/conformance/framework/swapchain_image_data.cpp @@ -3,9 +3,12 @@ // SPDX-License-Identifier: Apache-2.0 #include "swapchain_image_data.h" -#include "throw_helpers.h" -#include "types_and_constants.h" -#include "conformance_framework.h" + +#include "utilities/throw_helpers.h" +#include "utilities/types_and_constants.h" +#include "utilities/xrduration_literals.h" + +#include #include #include diff --git a/src/conformance/framework/swapchain_image_data.h b/src/conformance/framework/swapchain_image_data.h index e215eaa7..acb6449d 100644 --- a/src/conformance/framework/swapchain_image_data.h +++ b/src/conformance/framework/swapchain_image_data.h @@ -4,17 +4,16 @@ #pragma once -#include "throw_helpers.h" -#include "types_and_constants.h" - #include + #include +#include #include -#include -#include -#include #include -#include +#include +#include +#include +#include namespace Conformance { diff --git a/src/conformance/framework/two_call_struct.h b/src/conformance/framework/two_call_struct.h index 0b7ede87..46b71963 100644 --- a/src/conformance/framework/two_call_struct.h +++ b/src/conformance/framework/two_call_struct.h @@ -18,7 +18,7 @@ #pragma once #include "type_utils.h" -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" #include "hex_and_handles.h" #include diff --git a/src/conformance/framework/two_call_struct_tests.h b/src/conformance/framework/two_call_struct_tests.h index 8dee2e7c..ffba6810 100644 --- a/src/conformance/framework/two_call_struct_tests.h +++ b/src/conformance/framework/two_call_struct_tests.h @@ -18,7 +18,7 @@ #pragma once #include "two_call_struct.h" -#include "throw_helpers.h" +#include "utilities/throw_helpers.h" #include "hex_and_handles.h" #include diff --git a/src/conformance/framework/types_and_constants.cpp b/src/conformance/framework/types_and_constants.cpp deleted file mode 100644 index 2221f5fa..00000000 --- a/src/conformance/framework/types_and_constants.cpp +++ /dev/null @@ -1,88 +0,0 @@ -// Copyright (c) 2019-2023, The Khronos Group Inc. -// -// SPDX-License-Identifier: Apache-2.0 -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -#include -#include "utils.h" -#include "report.h" -#include "two_call_util.h" -#include "types_and_constants.h" -#include "conformance_framework.h" -#include "xr_dependencies.h" -#include "openxr/openxr_platform.h" -#include "openxr/openxr_reflection.h" -#include "graphics_plugin.h" - -#include -#include -#include -#include -#include -#include -#include -#include - -#ifdef _WIN32 -#include -#endif -#ifdef _MSC_VER -#pragma warning(disable : 4312) // ': conversion from 'int' to 'XrInstance' of greater size) -#endif - -#ifdef Success -#undef Success // Some platforms #define Success as 0, breaking its usage as an enumerant. -#endif - -namespace Conformance -{ - constexpr size_t HEX_DIGITS_FOR_HANDLE = 8; - - std::ostream& operator<<(std::ostream& os, NullHandleType const& /*unused*/) - { - os << "XR_NULL_HANDLE"; - return os; - } - - template - static inline void OutputHandle(std::ostream& os, T handle) - { - if (handle == XR_NULL_HANDLE) { - os << "XR_NULL_HANDLE"; - } - else { - std::ostringstream oss; - oss << "0x" << std::hex << std::setw(HEX_DIGITS_FOR_HANDLE) << std::setfill('0'); -#if XR_PTR_SIZE == 8 - oss << reinterpret_cast(handle); -#else - oss << static_cast(handle); -#endif - os << oss.str(); - } - } - - std::ostream& operator<<(std::ostream& os, AutoBasicInstance const& inst) - { - OutputHandle(os, inst.GetInstance()); - return os; - } - - std::ostream& operator<<(std::ostream& os, AutoBasicSession const& sess) - { - OutputHandle(os, sess.GetSession()); - return os; - } - -} // namespace Conformance diff --git a/src/conformance/conformance_test/vulkan_shaders/frag.glsl b/src/conformance/framework/vulkan_shaders/frag.glsl similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/frag.glsl rename to src/conformance/framework/vulkan_shaders/frag.glsl diff --git a/src/conformance/conformance_test/vulkan_shaders/frag.spv b/src/conformance/framework/vulkan_shaders/frag.spv similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/frag.spv rename to src/conformance/framework/vulkan_shaders/frag.spv diff --git a/src/conformance/conformance_test/vulkan_shaders/frag.spv.license b/src/conformance/framework/vulkan_shaders/frag.spv.license similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/frag.spv.license rename to src/conformance/framework/vulkan_shaders/frag.spv.license diff --git a/src/conformance/conformance_test/vulkan_shaders/vert.glsl b/src/conformance/framework/vulkan_shaders/vert.glsl similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/vert.glsl rename to src/conformance/framework/vulkan_shaders/vert.glsl diff --git a/src/conformance/conformance_test/vulkan_shaders/vert.spv b/src/conformance/framework/vulkan_shaders/vert.spv similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/vert.spv rename to src/conformance/framework/vulkan_shaders/vert.spv diff --git a/src/conformance/conformance_test/vulkan_shaders/vert.spv.license b/src/conformance/framework/vulkan_shaders/vert.spv.license similarity index 100% rename from src/conformance/conformance_test/vulkan_shaders/vert.spv.license rename to src/conformance/framework/vulkan_shaders/vert.spv.license diff --git a/src/conformance/framework/xml_test_environment.cpp b/src/conformance/framework/xml_test_environment.cpp index ab0c3597..165f09a8 100644 --- a/src/conformance/framework/xml_test_environment.cpp +++ b/src/conformance/framework/xml_test_environment.cpp @@ -3,11 +3,14 @@ // SPDX-License-Identifier: Apache-2.0 #include "xml_test_environment.h" -#include +#include "graphics_plugin.h" -#include "catch2/internal/catch_xmlwriter.hpp" #include "conformance_framework.h" -#include "hex_and_handles.h" +#include "common/hex_and_handles.h" + +#include + +#include #define CTS_XML_NS_PREFIX "cts" #define CTS_XML_NS_PREFIX_QUALIFIER CTS_XML_NS_PREFIX ":" @@ -140,6 +143,7 @@ namespace Conformance } xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "fileLineLoggingEnabled").writeAttribute("value", options.fileLineLoggingEnabled); + xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "pollGetSystem").writeAttribute("value", options.pollGetSystem); xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "debugMode").writeAttribute("value", options.debugMode); } @@ -149,14 +153,14 @@ namespace Conformance { auto e2 = xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "activeAPILayers"); - for (const std::string& apiLayerName : globalData.enabledAPILayerNames) { - xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "layer").writeAttribute("name", apiLayerName.c_str()); + for (const char* const& apiLayerName : globalData.enabledAPILayerNames) { + xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "layer").writeAttribute("name", apiLayerName); } } { auto e2 = xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "activeInstanceExtensions"); - for (const std::string& extensionName : globalData.enabledInstanceExtensionNames) { + for (const char* const& extensionName : globalData.enabledInstanceExtensionNames) { xml.scopedElement(CTS_XML_NS_PREFIX_QUALIFIER "extension").writeAttribute("name", extensionName); } } diff --git a/src/conformance/gradle/wrapper/gradle-wrapper.jar b/src/conformance/gradle/wrapper/gradle-wrapper.jar index 62d4c053550b91381bbd28b1afc82d634bf73a8a..249e5832f090a2944b7473328c07c9755baa3196 100644 GIT binary patch delta 21827 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 D86?w= diff --git a/src/conformance/gradle/wrapper/gradle-wrapper.properties b/src/conformance/gradle/wrapper/gradle-wrapper.properties index 0f80bbf5..8049c684 100644 --- a/src/conformance/gradle/wrapper/gradle-wrapper.properties +++ b/src/conformance/gradle/wrapper/gradle-wrapper.properties @@ -1,5 +1,5 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists -distributionUrl=https\://services.gradle.org/distributions/gradle-7.0.2-bin.zip +distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-bin.zip zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/src/conformance/gradlew b/src/conformance/gradlew index f1f48b2b..a69d9cb6 100755 --- a/src/conformance/gradlew +++ b/src/conformance/gradlew @@ -1,9 +1,7 @@ -#!/usr/bin/env sh +#!/bin/sh # -# Copyright 2015 the original author or authors. -# -# SPDX-License-Identifier: Apache-2.0 +# Copyright © 2015-2021 the original authors. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. @@ -19,67 +17,101 @@ # ############################################################################## -## -## Gradle start up script for UN*X -## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# ############################################################################## # Attempt to set APP_HOME + # Resolve links: $0 may be a link -PRG="$0" -# Need this for relative symlinks. -while [ -h "$PRG" ] ; do - ls=`ls -ld "$PRG"` - link=`expr "$ls" : '.*-> \(.*\)$'` - if expr "$link" : '/.*' > /dev/null; then - PRG="$link" - else - PRG=`dirname "$PRG"`"/$link" - fi +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac done -SAVED="`pwd`" -cd "`dirname \"$PRG\"`/" >/dev/null -APP_HOME="`pwd -P`" -cd "$SAVED" >/dev/null + +APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit APP_NAME="Gradle" -APP_BASE_NAME=`basename "$0"` +APP_BASE_NAME=${0##*/} # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' # Use the maximum available, or set MAX_FD != -1 to use that value. -MAX_FD="maximum" +MAX_FD=maximum warn () { echo "$*" -} +} >&2 die () { echo echo "$*" echo exit 1 -} +} >&2 # OS specific support (must be 'true' or 'false'). cygwin=false msys=false darwin=false nonstop=false -case "`uname`" in - CYGWIN* ) - cygwin=true - ;; - Darwin* ) - darwin=true - ;; - MINGW* ) - msys=true - ;; - NONSTOP* ) - nonstop=true - ;; +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; esac CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar @@ -89,9 +121,9 @@ CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar if [ -n "$JAVA_HOME" ] ; then if [ -x "$JAVA_HOME/jre/sh/java" ] ; then # IBM's JDK on AIX uses strange locations for the executables - JAVACMD="$JAVA_HOME/jre/sh/java" + JAVACMD=$JAVA_HOME/jre/sh/java else - JAVACMD="$JAVA_HOME/bin/java" + JAVACMD=$JAVA_HOME/bin/java fi if [ ! -x "$JAVACMD" ] ; then die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME @@ -100,7 +132,7 @@ Please set the JAVA_HOME variable in your environment to match the location of your Java installation." fi else - JAVACMD="java" + JAVACMD=java which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. Please set the JAVA_HOME variable in your environment to match the @@ -108,80 +140,101 @@ location of your Java installation." fi # Increase the maximum file descriptors if we can. -if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then - MAX_FD_LIMIT=`ulimit -H -n` - if [ $? -eq 0 ] ; then - if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then - MAX_FD="$MAX_FD_LIMIT" - fi - ulimit -n $MAX_FD - if [ $? -ne 0 ] ; then - warn "Could not set maximum file descriptor limit: $MAX_FD" - fi - else - warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" - fi +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac fi -# For Darwin, add options to specify how the application appears in the dock -if $darwin; then - GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" -fi +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. # For Cygwin or MSYS, switch paths to Windows format before running java -if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then - APP_HOME=`cygpath --path --mixed "$APP_HOME"` - CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` - - JAVACMD=`cygpath --unix "$JAVACMD"` - - # We build the pattern for arguments to be converted via cygpath - ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` - SEP="" - for dir in $ROOTDIRSRAW ; do - ROOTDIRS="$ROOTDIRS$SEP$dir" - SEP="|" - done - OURCYGPATTERN="(^($ROOTDIRS))" - # Add a user-defined pattern to the cygpath arguments - if [ "$GRADLE_CYGPATTERN" != "" ] ; then - OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" - fi +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + # Now convert the arguments - kludge to limit ourselves to /bin/sh - i=0 - for arg in "$@" ; do - CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` - CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option - - if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition - eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` - else - eval `echo args$i`="\"$arg\"" + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) fi - i=`expr $i + 1` + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg done - case $i in - 0) set -- ;; - 1) set -- "$args0" ;; - 2) set -- "$args0" "$args1" ;; - 3) set -- "$args0" "$args1" "$args2" ;; - 4) set -- "$args0" "$args1" "$args2" "$args3" ;; - 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; - 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; - 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; - 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; - 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; - esac fi -# Escape application args -save () { - for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done - echo " " -} -APP_ARGS=`save "$@"` +# Collect all arguments for the java command; +# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of +# shell script including quotes and variable substitutions, so put them in +# double quotes to make sure that they get re-expanded; and +# * put everything else in single quotes, so that it's not re-expanded. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Stop when "xargs" is not available. +if ! command -v xargs >/dev/null 2>&1 +then + die "xargs is not available" +fi + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# -# Collect all arguments for the java command, following the shell quoting and substitution rules -eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' exec "$JAVACMD" "$@" diff --git a/src/conformance/gradlew.bat b/src/conformance/gradlew.bat index bf0b614f..f127cfd4 100644 --- a/src/conformance/gradlew.bat +++ b/src/conformance/gradlew.bat @@ -1,7 +1,5 @@ @rem @rem Copyright 2015 the original author or authors. -@rem -@rem SPDX-License-Identifier: Apache-2.0 @rem @rem Licensed under the Apache License, Version 2.0 (the "License"); @rem you may not use this file except in compliance with the License. @@ -16,7 +14,7 @@ @rem limitations under the License. @rem -@if "%DEBUG%" == "" @echo off +@if "%DEBUG%"=="" @echo off @rem ########################################################################## @rem @rem Gradle startup script for Windows @@ -27,7 +25,7 @@ if "%OS%"=="Windows_NT" setlocal set DIRNAME=%~dp0 -if "%DIRNAME%" == "" set DIRNAME=. +if "%DIRNAME%"=="" set DIRNAME=. set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% @@ -42,7 +40,7 @@ if defined JAVA_HOME goto findJavaFromJavaHome set JAVA_EXE=java.exe %JAVA_EXE% -version >NUL 2>&1 -if "%ERRORLEVEL%" == "0" goto init +if %ERRORLEVEL% equ 0 goto execute echo. echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. @@ -56,7 +54,7 @@ goto fail set JAVA_HOME=%JAVA_HOME:"=% set JAVA_EXE=%JAVA_HOME%/bin/java.exe -if exist "%JAVA_EXE%" goto init +if exist "%JAVA_EXE%" goto execute echo. echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% @@ -66,21 +64,6 @@ echo location of your Java installation. goto fail -:init -@rem Get command-line arguments, handling Windows variants - -if not "%OS%" == "Windows_NT" goto win9xME_args - -:win9xME_args -@rem Slurp the command line arguments. -set CMD_LINE_ARGS= -set _SKIP=2 - -:win9xME_args_slurp -if "x%~1" == "x" goto execute - -set CMD_LINE_ARGS=%* - :execute @rem Setup the command line @@ -88,17 +71,19 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar @rem Execute Gradle -"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* :end @rem End local scope for the variables with windows NT shell -if "%ERRORLEVEL%"=="0" goto mainEnd +if %ERRORLEVEL% equ 0 goto mainEnd :fail rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of rem the _cmd.exe /c_ return code! -if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 -exit /b 1 +set EXIT_CODE=%ERRORLEVEL% +if %EXIT_CODE% equ 0 set EXIT_CODE=1 +if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE% +exit /b %EXIT_CODE% :mainEnd if "%OS%"=="Windows_NT" endlocal diff --git a/src/conformance/gradlew.bat.license b/src/conformance/gradlew.bat.license new file mode 100644 index 00000000..a6bf9762 --- /dev/null +++ b/src/conformance/gradlew.bat.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright 2015 the original author or authors. +SPDX-License-Identifier: Apache-2.0 diff --git a/src/conformance/gradlew.license b/src/conformance/gradlew.license new file mode 100644 index 00000000..69286a08 --- /dev/null +++ b/src/conformance/gradlew.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright 2015-2021 the original author or authors. +SPDX-License-Identifier: Apache-2.0 diff --git a/src/conformance/platform_specific/AndroidManifest.xml b/src/conformance/platform_specific/AndroidManifest.xml index cba09670..32f17c06 100644 --- a/src/conformance/platform_specific/AndroidManifest.xml +++ b/src/conformance/platform_specific/AndroidManifest.xml @@ -1,54 +1,73 @@ - - - - - - - - - - - + package="org.khronos.openxr.cts" + android:installLocation="auto" + android:versionCode="1" + android:versionName="1.0"> + + + + + + + + + + + + + + - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/conformance/platform_specific/android_main.cpp b/src/conformance/platform_specific/android_main.cpp index d14fa102..3fb36f29 100644 --- a/src/conformance/platform_specific/android_main.cpp +++ b/src/conformance/platform_specific/android_main.cpp @@ -15,7 +15,7 @@ // limitations under the License. #include "conformance/platform_specific/android_intent_extras.h" -#include "utils.h" +#include "utilities/utils.h" #include #include diff --git a/src/conformance/utilities/CMakeLists.txt b/src/conformance/utilities/CMakeLists.txt new file mode 100644 index 00000000..9a86511f --- /dev/null +++ b/src/conformance/utilities/CMakeLists.txt @@ -0,0 +1,34 @@ +# Copyright (c) 2019-2023, The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 + +add_library( + conformance_utilities STATIC + bitmask_generator.cpp + bitmask_to_string.cpp + d3d_common.cpp + event_reader.cpp + Geometry.cpp + stringification.cpp + types_and_constants.cpp + utils.cpp) + +target_include_directories( + conformance_utilities + PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/.. + # Backport of std::span functionality to pre-C++17 + ${PROJECT_SOURCE_DIR}/src/external/span-lite/include + # for openxr.h: + ${PROJECT_BINARY_DIR}/include) + +add_dependencies(conformance_utilities generate_openxr_header) + +if(GLSLANG_VALIDATOR AND NOT GLSLC_COMMAND) + target_compile_definitions(conformance_utilities PUBLIC USE_GLSLANGVALIDATOR) +endif() + +if(Vulkan_FOUND) + target_include_directories(conformance_utilities + PUBLIC ${Vulkan_INCLUDE_DIRS}) + target_link_libraries(conformance_utilities PUBLIC ${Vulkan_LIBRARY}) +endif() diff --git a/src/conformance/framework/Geometry.cpp b/src/conformance/utilities/Geometry.cpp similarity index 98% rename from src/conformance/framework/Geometry.cpp rename to src/conformance/utilities/Geometry.cpp index ff67e4c1..7eaf0309 100644 --- a/src/conformance/framework/Geometry.cpp +++ b/src/conformance/utilities/Geometry.cpp @@ -4,6 +4,8 @@ #include "Geometry.h" +#include + namespace Geometry { diff --git a/src/conformance/framework/Geometry.h b/src/conformance/utilities/Geometry.h similarity index 100% rename from src/conformance/framework/Geometry.h rename to src/conformance/utilities/Geometry.h diff --git a/src/conformance/utilities/align_to.h b/src/conformance/utilities/align_to.h new file mode 100644 index 00000000..c737f806 --- /dev/null +++ b/src/conformance/utilities/align_to.h @@ -0,0 +1,20 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#pragma once + +#include + +namespace Conformance +{ + /// Rounds up the parameter size @p n to be a multiple of @p alignment + template + constexpr std::uint32_t AlignTo(std::uint32_t n) + { + static_assert((alignment & (alignment - 1)) == 0, "The alignment must be power-of-two"); + // Add one less than the alignment: this will give us the largest possible value that we might need to round to. + // Then, mask off the bits that are lower order than the alignment. + return (n + alignment - 1) & ~(alignment - 1); + } +} // namespace Conformance diff --git a/src/conformance/utilities/array_size.h b/src/conformance/utilities/array_size.h new file mode 100644 index 00000000..e1c1351f --- /dev/null +++ b/src/conformance/utilities/array_size.h @@ -0,0 +1,18 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#pragma once + +#include + +namespace Conformance +{ + /// The equivalent of C++17 std::size. A helper to get the dimension for an array. + template + constexpr std::size_t ArraySize(const T (&unused)[Size]) noexcept + { + (void)unused; + return Size; + } +} // namespace Conformance diff --git a/src/conformance/framework/bitmask_generator.cpp b/src/conformance/utilities/bitmask_generator.cpp similarity index 94% rename from src/conformance/framework/bitmask_generator.cpp rename to src/conformance/utilities/bitmask_generator.cpp index cd3997b6..6bd4c6a5 100644 --- a/src/conformance/framework/bitmask_generator.cpp +++ b/src/conformance/utilities/bitmask_generator.cpp @@ -21,9 +21,12 @@ */ #include "bitmask_generator.h" +#include "generator.h" +#include #include #include +#include namespace Conformance { @@ -106,11 +109,11 @@ namespace Conformance GeneratorWrapper bitmaskGeneratorIncluding0(std::initializer_list const& bits) { - return {BitmaskGenerator::create(true, bits)}; + return GeneratorWrapper(BitmaskGenerator::create(true, bits)); } GeneratorWrapper bitmaskGenerator(std::initializer_list const& bits) { - return {BitmaskGenerator::create(false, bits)}; + return GeneratorWrapper(BitmaskGenerator::create(false, bits)); } } // namespace Conformance diff --git a/src/conformance/framework/bitmask_generator.h b/src/conformance/utilities/bitmask_generator.h similarity index 98% rename from src/conformance/framework/bitmask_generator.h rename to src/conformance/utilities/bitmask_generator.h index a0fbd6d3..31d9b1d3 100644 --- a/src/conformance/framework/bitmask_generator.h +++ b/src/conformance/utilities/bitmask_generator.h @@ -29,8 +29,8 @@ #include "generator.h" +#include #include -#include namespace Conformance { diff --git a/src/conformance/framework/bitmask_to_string.cpp b/src/conformance/utilities/bitmask_to_string.cpp similarity index 96% rename from src/conformance/framework/bitmask_to_string.cpp rename to src/conformance/utilities/bitmask_to_string.cpp index 2b102e8c..50fdb9a9 100644 --- a/src/conformance/framework/bitmask_to_string.cpp +++ b/src/conformance/utilities/bitmask_to_string.cpp @@ -3,11 +3,14 @@ // SPDX-License-Identifier: Apache-2.0 #include "bitmask_to_string.h" + +#include "common/hex_and_handles.h" + +#include + #include #include -#include "conformance_framework.h" -#include "hex_and_handles.h" -#include +#include namespace Conformance { diff --git a/src/conformance/framework/bitmask_to_string.h b/src/conformance/utilities/bitmask_to_string.h similarity index 98% rename from src/conformance/framework/bitmask_to_string.h rename to src/conformance/utilities/bitmask_to_string.h index eb95ad0c..eb31b963 100644 --- a/src/conformance/framework/bitmask_to_string.h +++ b/src/conformance/utilities/bitmask_to_string.h @@ -4,12 +4,13 @@ #pragma once -#include "conformance_framework.h" +#include #include #include #include -#include "openxr/openxr.h" +#include +#include namespace Conformance { diff --git a/src/conformance/framework/d3d_common.cpp b/src/conformance/utilities/d3d_common.cpp similarity index 98% rename from src/conformance/framework/d3d_common.cpp rename to src/conformance/utilities/d3d_common.cpp index 20964a3f..ca878e4c 100644 --- a/src/conformance/framework/d3d_common.cpp +++ b/src/conformance/utilities/d3d_common.cpp @@ -4,16 +4,18 @@ #if (defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12)) && !defined(MISSING_DIRECTX_COLORS) -#include // For Microsoft::WRL::ComPtr +#include "d3d_common.h" + #include "swapchain_parameters.h" -#include "conformance_framework.h" -#include +#include "common/xr_linear.h" +#include "throw_helpers.h" + +#include // For Microsoft::WRL::ComPtr + #include #include #include -#include "d3d_common.h" -#include "throw_helpers.h" using namespace Microsoft::WRL; using namespace DirectX; @@ -266,16 +268,15 @@ namespace Conformance // Verify that the image format is known. If it's not known then this test needs to be // updated to recognize new DXGI formats. - CAPTURE(imageFormat); - CHECK_MSG(it != dxgiSwapchainTestMap.end(), "Unknown DXGI image format."); + XRC_CHECK_THROW_MSG(it != dxgiSwapchainTestMap.end(), "Unknown DXGI image format."); if (it == dxgiSwapchainTestMap.end()) { return false; } // Verify that imageFormat is not a typeless type. Only regular types are allowed to // be returned by the runtime for enumerated image formats. - CAPTURE(it->second.imageFormatName); - CHECK_MSG(!it->second.mutableFormat, "Typeless DXGI image formats must not be enumerated by runtimes."); + XRC_CHECK_THROW_MSG(!it->second.mutableFormat, + "Typeless DXGI image formats must not be enumerated by runtimes: " + it->second.imageFormatName); if (it->second.mutableFormat) { return false; } diff --git a/src/conformance/framework/d3d_common.h b/src/conformance/utilities/d3d_common.h similarity index 98% rename from src/conformance/framework/d3d_common.h rename to src/conformance/utilities/d3d_common.h index 0e6a6c29..29245aea 100644 --- a/src/conformance/framework/d3d_common.h +++ b/src/conformance/utilities/d3d_common.h @@ -6,13 +6,14 @@ #if (defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12)) && !defined(MISSING_DIRECTX_COLORS) #include -#include "xr_linear.h" +#include "common/xr_linear.h" #include "swapchain_parameters.h" #include #include #include #include + #include #include diff --git a/src/conformance/framework/event_reader.cpp b/src/conformance/utilities/event_reader.cpp similarity index 98% rename from src/conformance/framework/event_reader.cpp rename to src/conformance/utilities/event_reader.cpp index b90caff8..80e8ff85 100644 --- a/src/conformance/framework/event_reader.cpp +++ b/src/conformance/utilities/event_reader.cpp @@ -15,9 +15,11 @@ // limitations under the License. #include "event_reader.h" -#include "conformance_framework.h" + #include "throw_helpers.h" +#include + namespace Conformance { diff --git a/src/conformance/framework/event_reader.h b/src/conformance/utilities/event_reader.h similarity index 98% rename from src/conformance/framework/event_reader.h rename to src/conformance/utilities/event_reader.h index 775c21b5..1637f61c 100644 --- a/src/conformance/framework/event_reader.h +++ b/src/conformance/utilities/event_reader.h @@ -16,9 +16,10 @@ #pragma once +#include +#include #include #include -#include namespace Conformance { diff --git a/src/conformance/framework/generator.h b/src/conformance/utilities/generator.h similarity index 97% rename from src/conformance/framework/generator.h rename to src/conformance/utilities/generator.h index 4776c06e..85a2c7ca 100644 --- a/src/conformance/framework/generator.h +++ b/src/conformance/utilities/generator.h @@ -105,7 +105,7 @@ class GeneratorWrapper std::unique_ptr> inner_; public: - GeneratorWrapper(std::unique_ptr>&& inner) : inner_(std::move(inner)) + explicit GeneratorWrapper(std::unique_ptr>&& inner) : inner_(std::move(inner)) { } diff --git a/src/conformance/utilities/stringification.cpp b/src/conformance/utilities/stringification.cpp new file mode 100644 index 00000000..b19ec833 --- /dev/null +++ b/src/conformance/utilities/stringification.cpp @@ -0,0 +1,35 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#include "stringification.h" +#include "utils.h" + +#include +#include + +#include + +namespace Conformance +{ + + // We keep our own copy of this as opposed to calling the xrResultToString function, because our + // purpose here it to validate the runtime's implementation of xrResultToString. + + const ResultStringMap& GetResultStringMap() + { + static const ResultStringMap resultStringMap{XR_LIST_ENUM_XrResult(XRC_ENUM_NAME_PAIR)}; + return resultStringMap; + } + + const char* ResultToString(XrResult result) + { + auto it = GetResultStringMap().find(result); + + if (it != GetResultStringMap().end()) { + return it->second; + } + + return ""; + } +} // namespace Conformance diff --git a/src/conformance/utilities/stringification.h b/src/conformance/utilities/stringification.h new file mode 100644 index 00000000..616d8073 --- /dev/null +++ b/src/conformance/utilities/stringification.h @@ -0,0 +1,37 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#pragma once + +#include +#include + +namespace Conformance +{ + /// We keep a private auto-generated map of all results and their string versions. + typedef std::map ResultStringMap; + + const ResultStringMap& GetResultStringMap(); + + /// @addtogroup cts_framework + /// @{ + + /// Returns a string for a given XrResult, based on our accounting of the result strings, and not + /// based on the xrResultToString function. + /// Returns "" if the result is not recognized. + /// + /// Example usage: + /// ``` + /// XrResult result = xrPollEvent(instance, &eventData); + /// printf("%d: %s, resut, ResultToString(result)); + /// ``` + const char* ResultToString(XrResult result); + +#define XRC_CHECK_STRINGIFY(x) #x +#define XRC_TO_STRING(x) XRC_CHECK_STRINGIFY(x) + +/// Represents a compile time file and line location as a single string. +#define XRC_FILE_AND_LINE __FILE__ ":" XRC_TO_STRING(__LINE__) + +} // namespace Conformance diff --git a/src/conformance/framework/swapchain_parameters.h b/src/conformance/utilities/swapchain_parameters.h similarity index 99% rename from src/conformance/framework/swapchain_parameters.h rename to src/conformance/utilities/swapchain_parameters.h index d036bd91..5b2a83d4 100644 --- a/src/conformance/framework/swapchain_parameters.h +++ b/src/conformance/utilities/swapchain_parameters.h @@ -16,7 +16,7 @@ #pragma once -#include +#include "utils.h" #include #include diff --git a/src/conformance/utilities/system_properties_helper.h b/src/conformance/utilities/system_properties_helper.h new file mode 100644 index 00000000..bdbfbd40 --- /dev/null +++ b/src/conformance/utilities/system_properties_helper.h @@ -0,0 +1,129 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#pragma once + +#include +#include +#include "throw_helpers.h" + +namespace Conformance +{ + /// A generic wrapper for xrGetSystemProperties that returns only a single member of an extension struct. + /// + /// Mainly for use by @ref SystemPropertiesChecker and @ref SystemPropertiesBoolChecker + template + static inline MemberType GetSystemPropertiesValue(const SysPropsExtStruct& emptyExtStruct, + MemberType SysPropsExtStruct::*pointerToMember, XrInstance instance, + XrSystemId systemId) + { + SysPropsExtStruct extSystemProperties{emptyExtStruct}; + XrSystemProperties systemProperties{XR_TYPE_SYSTEM_PROPERTIES, &extSystemProperties}; + XrResult result = xrGetSystemProperties(instance, systemId, &systemProperties); + if (result != XR_SUCCESS) { + XRC_THROW_XRRESULT(result, xrGetSystemProperties); + } + + return (&extSystemProperties)->*pointerToMember; + } + + /// A functor class template that you can call with XrInstance and XrSystemId to get the value of a field in a struct chained to XrSystemProperties. + /// + /// @see MakeSystemPropertiesChecker for easy creation in the absence of C++17 class type parameter deduction. + template + class SystemPropertiesChecker + { + public: + /// Pointer to data member type + using MemberObjectPointer = MemberType SysPropsExtStruct::*; + + static_assert(std::is_pod{}, "Extension structs must be plain-old data"); + + /// Constructor + /// + /// @param emptyExtStruct An empty but initialized extension struct to chain on to XrSystemProperties. + /// Make sure `type` is initialized. The `next` pointer will be cleared before use. + /// @param memberToReturn A pointer to data member corresponding to the member of the extension struct to retrieve. + SystemPropertiesChecker(const SysPropsExtStruct& emptyExtStruct, MemberObjectPointer memberToReturn) noexcept + : m_emptyExtStruct(emptyExtStruct), m_memberToReturn(memberToReturn) + { + m_emptyExtStruct.next = nullptr; + } + + /// Functor call operator: Will call xrGetSystemProperties with your instance and systemId, and your struct chained on. + /// Will return the value of the member for which you provided a pointer + MemberType operator()(XrInstance instance, XrSystemId systemId) const + { + return GetSystemPropertiesValue(m_emptyExtStruct, m_memberToReturn, instance, systemId); + } + + private: + SysPropsExtStruct m_emptyExtStruct; + MemberObjectPointer m_memberToReturn; + }; + + /// Create a functor that you can call with XrInstance and XrSystemId to get the value of a member in a struct chained to XrSystemProperties. + /// + /// Helper function to deduce the type params of @ref SystemPropertiesChecker from the empty struct passed as the first argument and the pointer to data member passed as the second. + /// + /// @param emptyExtStruct An empty but initialized extension struct to chain on to XrSystemProperties. Make sure `type` is initialized. The `next` pointer will be cleared before use. + /// @param memberToReturn A pointer to data member corresponding to the member of the extension struct to retrieve. + template + static inline SystemPropertiesChecker + MakeSystemPropertiesChecker(const SysPropsExtStruct& emptyExtStruct, MemberType SysPropsExtStruct::*memberToReturn) noexcept + { + return SystemPropertiesChecker{emptyExtStruct, memberToReturn}; + } + + /// A functor class template that you can call with XrInstance and XrSystemId to get the value of a boolean field in a struct chained to XrSystemProperties. + /// + /// Like @ref SystemPropertiesChecker but for bools only, with conversion built in. + /// + /// @see MakeSystemPropertiesBoolChecker for easy creation in the absence of C++17 class type parameter deduction. + template + class SystemPropertiesBoolChecker + { + public: + /// Pointer to data member type: member must be an XrBool32 + using MemberObjectPointer = XrBool32 SysPropsExtStruct::*; + + static_assert(std::is_pod{}, "Extension structs must be plain-old data"); + + /// Constructor + /// + /// @param emptyExtStruct An empty but initialized extension struct to chain on to XrSystemProperties. + /// Make sure `type` is initialized. The `next` pointer will be cleared before use. + /// @param memberToReturn A pointer to data member corresponding to the member of the extension struct to retrieve. + SystemPropertiesBoolChecker(const SysPropsExtStruct& emptyExtStruct, MemberObjectPointer memberToReturn) noexcept + : m_emptyExtStruct(emptyExtStruct), m_memberToReturn(memberToReturn) + { + m_emptyExtStruct.next = nullptr; + } + + /// Functor call operator: Will call xrGetSystemProperties with your instance and systemId, and your struct chained on. + /// Will return the value of the XrBool32 member for which you provided a pointer, converted to bool + bool operator()(XrInstance instance, XrSystemId systemId) const + { + return GetSystemPropertiesValue(m_emptyExtStruct, m_memberToReturn, instance, systemId) == XR_TRUE; + } + + private: + SysPropsExtStruct m_emptyExtStruct; + MemberObjectPointer m_memberToReturn; + }; + + /// Create a functor that you can call with XrInstance and XrSystemId to get the value of a boolean member in a struct chained to XrSystemProperties. + /// + /// Helper function to deduce the type param of @ref SystemPropertiesBoolChecker from the empty struct passed as the first argument, and convert the result from XrBool32 to bool. + /// + /// @param emptyExtStruct An empty but initialized extension struct to chain on to XrSystemProperties. Make sure `type` is initialized. The `next` pointer will be cleared before use. + /// @param memberToReturn A pointer to data member corresponding to the member of the extension struct to retrieve. + template + static inline SystemPropertiesBoolChecker + MakeSystemPropertiesBoolChecker(const SysPropsExtStruct& emptyExtStruct, XrBool32 SysPropsExtStruct::*memberToReturn) noexcept + { + return SystemPropertiesBoolChecker{emptyExtStruct, memberToReturn}; + } + +} // namespace Conformance diff --git a/src/conformance/framework/throw_helpers.h b/src/conformance/utilities/throw_helpers.h similarity index 93% rename from src/conformance/framework/throw_helpers.h rename to src/conformance/utilities/throw_helpers.h index f3be00eb..56571578 100644 --- a/src/conformance/framework/throw_helpers.h +++ b/src/conformance/utilities/throw_helpers.h @@ -17,8 +17,7 @@ #pragma once #include "utils.h" -#include "conformance_utils.h" -#include "conformance_framework.h" +#include "stringification.h" #include #include @@ -27,7 +26,7 @@ #ifdef XR_USE_PLATFORM_WIN32 // Mostly for the cautious windows.h include -#include "xr_dependencies.h" +#include "common/xr_dependencies.h" #endif #ifdef XR_USE_PLATFORM_ANDROID @@ -41,7 +40,7 @@ namespace Conformance * @defgroup cts_throw Exception-throwing helpers * @brief Helpers for when you can't use a Catch2 macro. * - * Code in the framework itself should generally use exceptions, rather than assertion macros, for thread safety. + * Code in the framework itself should generally use exceptions, rather than assertion macros, for thread safety and to avoid inflating the assertion count. * * If a helper in the framework wants to report an error, and it might be called from something other than the "main" thread, * it must throw an exception rather than using a CHECK/REQUIRE macro, as use of Catch2 test state from multiple threads is undefined. @@ -108,8 +107,8 @@ namespace Conformance return res; } - static inline XrResult CheckThrowXrResultSuccessOrLimitReached(XrResult res, const char* originator = nullptr, - const char* sourceLocation = nullptr) noexcept(false) + inline XrResult CheckThrowXrResultSuccessOrLimitReached(XrResult res, const char* originator = nullptr, + const char* sourceLocation = nullptr) noexcept(false) { if (XR_FAILED(res) && res != XR_ERROR_LIMIT_REACHED) { Throw(StringSprintf("XrResult failure (and not XR_ERROR_LIMIT_REACHED) [%s]", ResultToString(res)), originator, sourceLocation); diff --git a/src/conformance/utilities/types_and_constants.cpp b/src/conformance/utilities/types_and_constants.cpp new file mode 100644 index 00000000..a63b7243 --- /dev/null +++ b/src/conformance/utilities/types_and_constants.cpp @@ -0,0 +1,49 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include "types_and_constants.h" + +#include +#include +#include +#include + +namespace Conformance +{ + constexpr size_t HEX_DIGITS_FOR_HANDLE = 8; + + std::ostream& operator<<(std::ostream& os, NullHandleType const& /*unused*/) + { + os << "XR_NULL_HANDLE"; + return os; + } + namespace detail + { + void OutputHandle(std::ostream& os, uint64_t handle) + { + if (handle == 0) { + os << "XR_NULL_HANDLE"; + } + else { + std::ostringstream oss; + oss << "0x" << std::hex << std::setw(HEX_DIGITS_FOR_HANDLE) << std::setfill('0'); + oss << handle; + os << oss.str(); + } + } + } // namespace detail + +} // namespace Conformance diff --git a/src/conformance/framework/types_and_constants.h b/src/conformance/utilities/types_and_constants.h similarity index 93% rename from src/conformance/framework/types_and_constants.h rename to src/conformance/utilities/types_and_constants.h index 259362e5..2a927882 100644 --- a/src/conformance/framework/types_and_constants.h +++ b/src/conformance/utilities/types_and_constants.h @@ -17,14 +17,10 @@ #pragma once #include - -#include +#include #include -#include -#include -#include -#include #include +#include namespace Conformance { @@ -133,9 +129,6 @@ namespace Conformance /// @relates NullHandleType std::ostream& operator<<(std::ostream& os, NullHandleType const& /*unused*/); - template - class ScopedHandle; - /// A unique-ownership RAII helper for OpenXR handles. /// /// @tparam HandleType The handle type to wrap @@ -266,16 +259,19 @@ namespace Conformance return handle.get() != XR_NULL_HANDLE_CPP; } - struct AutoBasicInstance; - - /// Output operator for the `XrInstance` handle in a @ref AutoBasicInstance - /// @relates AutoBasicInstance - std::ostream& operator<<(std::ostream& os, AutoBasicInstance const& inst); - - struct AutoBasicSession; - - /// Output operator for the `XrSession` handle in a @ref AutoBasicSession - /// @relates AutoBasicSession - std::ostream& operator<<(std::ostream& os, AutoBasicSession const& sess); + namespace detail + { + void OutputHandle(std::ostream& os, uint64_t handle); + } + /// Outputs a formatted handle to a stream. + template + static inline void OutputHandle(std::ostream& os, T handle) + { +#if XR_PTR_SIZE == 8 + detail::OutputHandle(os, reinterpret_cast(handle)); +#else + detail::OutputHandle(os, static_cast(handle)); +#endif + } } // namespace Conformance diff --git a/src/conformance/framework/utils.cpp b/src/conformance/utilities/utils.cpp similarity index 98% rename from src/conformance/framework/utils.cpp rename to src/conformance/utilities/utils.cpp index 4b2c74c9..1f506054 100644 --- a/src/conformance/framework/utils.cpp +++ b/src/conformance/utilities/utils.cpp @@ -15,14 +15,19 @@ // limitations under the License. #include "utils.h" + +#include #include #include #include #include #include #include -#include #include +#include +#include +#include +#include #ifdef _WIN32 #include @@ -87,7 +92,7 @@ namespace Conformance // a slight skew in the distribution for most ranges. C++ random number generation requires // reconstructing distribution classes for each range used, which is onerous. // - RandEngine::RandEngine() : randEngineMutex(), seed(std::time(NULL)), engine() // Default seed + RandEngine::RandEngine() : randEngineMutex(), seed(std::time(nullptr)), engine() // Default seed { } diff --git a/src/conformance/framework/utils.h b/src/conformance/utilities/utils.h similarity index 95% rename from src/conformance/framework/utils.h rename to src/conformance/utilities/utils.h index 3b041054..e92e3d79 100644 --- a/src/conformance/framework/utils.h +++ b/src/conformance/utilities/utils.h @@ -21,12 +21,11 @@ #include #include #include -#include -#include #include #include #include #include +#include /// @addtogroup cts_framework /// @{ @@ -49,9 +48,10 @@ #define XDGW3(x) XDGW2(#x) #endif -#if defined(__GNUC__) && (__GNUC_VERSION__ >= 406) +#if defined(__GNUC__) && defined(__GNUC_VERSION__) && (__GNUC_VERSION__ >= 406) #define XRC_DISABLE_GCC_WARNING(w) _Pragma("GCC diagnostic push") _Pragma(XDGW3(w)) -#elif defined(__GNUC__) && (__GNUC_VERSION__ >= 404) // GCC 4.4 doesn't support diagnostic push, but supports disabling warnings. +#elif defined(__GNUC__) && defined(__GNUC_VERSION__) && \ + (__GNUC_VERSION__ >= 404) // GCC 4.4 doesn't support diagnostic push, but supports disabling warnings. #define XRC_DISABLE_GCC_WARNING(w) _Pragma(XDGW3(w)) #else #define XRC_DISABLE_GCC_WARNING(w) @@ -59,7 +59,7 @@ #endif // !defined(XRC_DISABLE_GCC_WARNING) #if !defined(XRC_RESTORE_GCC_WARNING) -#if defined(__GNUC__) && (__GNUC_VERSION__ >= 4006) +#if defined(__GNUC__) && defined(__GNUC_VERSION__) && (__GNUC_VERSION__ >= 4006) #define XRC_RESTORE_GCC_WARNINGS() _Pragma("GCC diagnostic pop") #else #define XRC_RESTORE_GCC_WARNING() @@ -255,7 +255,7 @@ namespace Conformance /// This is a specially crafted valid UTF8 string which has four Unicode code points, /// with the first being one byte, the second being two bytes, the third being three bytes, -/// and the fourth being four bytes. This is useful for exercizing a runtime's requirement +/// and the fourth being four bytes. This is useful for exercising a runtime's requirement /// of supporting UTF8 strings. See https://tools.ietf.org/html/rfc3629#section-3. /// #define XRC_UTF8_VALID_EXERCISE_STR "\x61\xC8\xBF\xE5\x86\x98\xF0\xAE\xAA\x85" @@ -272,7 +272,7 @@ namespace Conformance { public: RandEngine(); - RandEngine(uint64_t seed); + explicit RandEngine(uint64_t seed); /// Sets the new seed, overriding whatever seed was set by the constructor. void SetSeed(uint64_t seed); @@ -381,7 +381,7 @@ namespace Conformance StringVec& operator=(StringVec const& other); /// Construct from vector of std::string. - StringVec(std::vector const& other); + explicit StringVec(std::vector const& other); /// Assign from vector of std::string. StringVec& operator=(std::vector const& other); @@ -390,7 +390,10 @@ namespace Conformance using const_iterator = typename inner::const_iterator; /// "Conversion" operator to the contained vector of C string pointers. - operator inner const &() const + // clang-format off + operator inner const&() const + // clang-format on + // suppression required because clang-format 16 and 11 disagree on spacing before the & { return strPtrVector; } diff --git a/src/conformance/framework/vulkan_utils.h b/src/conformance/utilities/vulkan_utils.h similarity index 98% rename from src/conformance/framework/vulkan_utils.h rename to src/conformance/utilities/vulkan_utils.h index 16f089ae..99933130 100644 --- a/src/conformance/framework/vulkan_utils.h +++ b/src/conformance/utilities/vulkan_utils.h @@ -38,7 +38,7 @@ namespace Conformance { - static inline std::string vkResultString(VkResult res) + inline std::string vkResultString(VkResult res) { switch (res) { case VK_SUCCESS: @@ -113,7 +113,7 @@ namespace Conformance _(ALL_GRAPHICS) \ _(ALL_COMMANDS) - static inline std::string GetPipelineStages(VkPipelineStageFlags stages) + inline std::string GetPipelineStages(VkPipelineStageFlags stages) { std::string desc; #define MK_PIPE_STAGE_CHECK(n) \ @@ -124,7 +124,7 @@ namespace Conformance return desc; } - [[noreturn]] static inline void ThrowVkResult(VkResult res, const char* originator = nullptr, const char* sourceLocation = nullptr) + [[noreturn]] inline void ThrowVkResult(VkResult res, const char* originator = nullptr, const char* sourceLocation = nullptr) { Throw("VkResult failure " + vkResultString(res), originator, sourceLocation); } @@ -145,7 +145,7 @@ namespace Conformance while (0) #endif - static inline VkResult CheckThrowVkResult(VkResult res, const char* originator = nullptr, const char* sourceLocation = nullptr) + inline VkResult CheckThrowVkResult(VkResult res, const char* originator = nullptr, const char* sourceLocation = nullptr) { if ((res) < VK_SUCCESS) { SHOW_CHECKPOINTS(); @@ -426,7 +426,7 @@ namespace Conformance } modInfo.codeSize = code.size() * sizeof(code[0]); - modInfo.pCode = &code[0]; + modInfo.pCode = code.data(); XRC_CHECK_THROW_MSG((modInfo.codeSize > 0) && modInfo.pCode, "Invalid shader " + name); XRC_CHECK_THROW_VKCMD(vkCreateShaderModule(m_vkDevice, &modInfo, nullptr, &si.module)); diff --git a/src/conformance/utilities/xrduration_literals.h b/src/conformance/utilities/xrduration_literals.h new file mode 100644 index 00000000..fa335491 --- /dev/null +++ b/src/conformance/utilities/xrduration_literals.h @@ -0,0 +1,46 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#pragma once + +#include + +namespace Conformance +{ + /// Example usage: + /// ``` + /// XrDuration timeout = 10_xrSeconds; + /// ``` + inline constexpr XrDuration operator"" _xrSeconds(unsigned long long value) + { + return (static_cast(value) * 1000 * 1000 * 1000); // Convert seconds to XrDuration nanoseconds. + } + + /// Example usage: + /// ``` + /// XrDuration timeout = 10_xrMilliseconds; + /// ``` + inline constexpr XrDuration operator"" _xrMilliseconds(unsigned long long value) + { + return (static_cast(value) * 1000 * 1000); // Convert milliseconds to XrDuration nanoseconds. + } + + /// Example usage: + /// ``` + /// XrDuration timeout = 10_xrMicroseconds; + /// ``` + inline constexpr XrDuration operator"" _xrMicroseconds(unsigned long long value) + { + return (static_cast(value) * 1000); // Convert microseconds to XrDuration nanoseconds. + } + + /// Example usage: + /// ``` + /// XrDuration timeout = 10_xrNanoseconds; + /// ``` + inline constexpr XrDuration operator"" _xrNanoseconds(unsigned long long value) + { + return static_cast(value); // XrDuration is already in nanoseconds + } +} // namespace Conformance diff --git a/src/external/d3dx12/LICENSE b/src/external/d3dx12/LICENSE new file mode 100644 index 00000000..63447fd8 --- /dev/null +++ b/src/external/d3dx12/LICENSE @@ -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/src/external/d3dx12/README.md b/src/external/d3dx12/README.md new file mode 100644 index 00000000..1e77be6e --- /dev/null +++ b/src/external/d3dx12/README.md @@ -0,0 +1,59 @@ +# DirectX Headers + +This repository hosts the official Direct3D 12 headers. These headers are made available under the MIT license rather than the traditional Windows SDK license. + +Additionally, this repository hosts several helpers for using these headers. + +## Directory Structure + +* `/`: Build files are available here for quick integration. CMake is provided, and can be referenced either via `subdirectory()` or after installation to a system location. Meson is also available for inclusion as a subproject/wrap. +* `/include/directx`: These files are the core headers for using D3D12, plus d3dx12.h, which is a helper and does not cross the boundaries of the D3D12 API. +* `/include/wsl`: These files are provided as a shim to be able to include the D3D12 headers from a Linux build environment, without requiring the rest of the Windows SDK. +* `/include/dxguids`: This header allows an application to use `uuidof()` consistently between Windows and WSL, instead of `__uuidof()`. +* `/src/dxguids.cpp`: This cpp file can be used as a replacement for linking against `dxguid.lib` on Windows, and as a convenient translation unit to define GUIDs without multiple definitions for WSL. +* `/test`: Simple CMake/Meson projects for validating the headers can be included in a given environment + +## Use on Windows + +Note that these headers may conflict with the headers from the Windows SDK, depending on include ordering. These headers should be added to the include directory list before the SDK, and should be included before other graphics headers (e.g. `d3d11.h`) from the Windows SDK. Otherwise, the corresponding header from the Windows SDK may be included first, and will define the include guards which prevents these headers from being used. + +## Use on WSL + +Note: WSL support is not intended for general purpose application development. At this time, the only recommended usage is for frameworks wishing to provide hardware acceleration for a Linux graphics/compute API in a WSL2 virtualization environment. + +Note: WSL support is only available for 64-bit binaries. + +The headers in the `/include/wsl` directory provide alternative definitions to macros and typedefs normally found in the Windows SDK. For the most part, they should be straightforward, but there are a couple to call attention to: + +|Type|Reason| +|---|---| +|`LONG`/`ULONG`|On 64-bit Windows, a `long` is 4 bytes, but on Linux it is typically 8 bytes. The D3D12 ABI for WSL uses `long` and therefore these should be 8 bytes.| +|`WCHAR`/`WCSTR`|On Windows, a `wchar_t` is 2 bytes, but on Linux it is typically 4 bytes. The D3D12 ABI for WSL uses the native 4-byte `wchar_t`, to enable applications and the runtime to use the system C library to perform string manipulation.| + +Additionally, APIs taking `HANDLE` (`void*`) for Win32 types should instead use `reinterpret_cast(fd)` for an appropriate type of file descriptor. For `ID3D12Fence::SetEventOnCompletion` this should be an `eventfd`, and for shared resources will be an opaque fd. + +## Ways to consume + +There are various ways to consume the headers in this project: + +* Manually: Just copy the headers somewhere and point your project at them. +* CMake subproject: Add this entire project as a subdirectory of your larger project, e.g. as a git submodule, and `add_subdirectory` into it. Use the resulting `DirectX-Headers` and/or `DirectX-Guids` targets as a link dependency +* Installed CMake: After building/installing this project, it can be found through CMake's `find_package` functionality and will expose the same `DirectX-Headers` and `DirectX-Guids` targets. +* FetchContent CMake (3.11+): Fetch this library using Git and easily add it to your project. +* Meson subproject/wrap: Add this entire project as a subproject of your larger project, and use `subproject` or `dependency` to consume it. +* Pkg-config: Use Meson to build this project, and the resulting installed package can be found via pkg-config. +* vcpkg: A vcpkg port has [been added](https://github.com/microsoft/vcpkg/pull/15222). + +Contributions for new mechanisms are welcome. + +## Contributing + +This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.opensource.microsoft.com. + +When you submit a pull request, a CLA bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., status check, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA. + +This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments. + +## Trademarks + +This project may contain trademarks or logos for projects, products, or services. Authorized use of Microsoft trademarks or logos is subject to and must follow [Microsoft's Trademark & Brand Guidelines](https://www.microsoft.com/en-us/legal/intellectualproperty/trademarks/usage/general). Use of Microsoft trademarks or logos in modified versions of this project must not cause confusion or imply Microsoft sponsorship. Any use of third-party trademarks or logos are subject to those third-party's policies. \ No newline at end of file diff --git a/src/external/d3dx12/d3dx12.h b/src/external/d3dx12/d3dx12.h new file mode 100644 index 00000000..b1355989 --- /dev/null +++ b/src/external/d3dx12/d3dx12.h @@ -0,0 +1,4049 @@ +//********************************************************* +// +// Copyright (c) Microsoft Corporation. +// Licensed under the MIT License (MIT). +// +//********************************************************* + +#ifndef __D3DX12_H__ +#define __D3DX12_H__ + +#include "d3d12.h" + +#if defined( __cplusplus ) + +struct CD3DX12_DEFAULT {}; +extern const DECLSPEC_SELECTANY CD3DX12_DEFAULT D3D12_DEFAULT; + +//------------------------------------------------------------------------------------------------ +inline bool operator==( const D3D12_VIEWPORT& l, const D3D12_VIEWPORT& r ) noexcept +{ + return l.TopLeftX == r.TopLeftX && l.TopLeftY == r.TopLeftY && l.Width == r.Width && + l.Height == r.Height && l.MinDepth == r.MinDepth && l.MaxDepth == r.MaxDepth; +} + +//------------------------------------------------------------------------------------------------ +inline bool operator!=( const D3D12_VIEWPORT& l, const D3D12_VIEWPORT& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RECT : public D3D12_RECT +{ + CD3DX12_RECT() = default; + explicit CD3DX12_RECT( const D3D12_RECT& o ) noexcept : + D3D12_RECT( o ) + {} + explicit CD3DX12_RECT( + LONG Left, + LONG Top, + LONG Right, + LONG Bottom ) noexcept + { + left = Left; + top = Top; + right = Right; + bottom = Bottom; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_VIEWPORT : public D3D12_VIEWPORT +{ + CD3DX12_VIEWPORT() = default; + explicit CD3DX12_VIEWPORT( const D3D12_VIEWPORT& o ) noexcept : + D3D12_VIEWPORT( o ) + {} + explicit CD3DX12_VIEWPORT( + FLOAT topLeftX, + FLOAT topLeftY, + FLOAT width, + FLOAT height, + FLOAT minDepth = D3D12_MIN_DEPTH, + FLOAT maxDepth = D3D12_MAX_DEPTH ) noexcept + { + TopLeftX = topLeftX; + TopLeftY = topLeftY; + Width = width; + Height = height; + MinDepth = minDepth; + MaxDepth = maxDepth; + } + explicit CD3DX12_VIEWPORT( + _In_ ID3D12Resource* pResource, + UINT mipSlice = 0, + FLOAT topLeftX = 0.0f, + FLOAT topLeftY = 0.0f, + FLOAT minDepth = D3D12_MIN_DEPTH, + FLOAT maxDepth = D3D12_MAX_DEPTH ) noexcept + { + auto Desc = pResource->GetDesc(); + const UINT64 SubresourceWidth = Desc.Width >> mipSlice; + const UINT64 SubresourceHeight = Desc.Height >> mipSlice; + switch (Desc.Dimension) + { + case D3D12_RESOURCE_DIMENSION_BUFFER: + TopLeftX = topLeftX; + TopLeftY = 0.0f; + Width = float(Desc.Width) - topLeftX; + Height = 1.0f; + break; + case D3D12_RESOURCE_DIMENSION_TEXTURE1D: + TopLeftX = topLeftX; + TopLeftY = 0.0f; + Width = (SubresourceWidth ? float(SubresourceWidth) : 1.0f) - topLeftX; + Height = 1.0f; + break; + case D3D12_RESOURCE_DIMENSION_TEXTURE2D: + case D3D12_RESOURCE_DIMENSION_TEXTURE3D: + TopLeftX = topLeftX; + TopLeftY = topLeftY; + Width = (SubresourceWidth ? float(SubresourceWidth) : 1.0f) - topLeftX; + Height = (SubresourceHeight ? float(SubresourceHeight) : 1.0f) - topLeftY; + break; + default: break; + } + + MinDepth = minDepth; + MaxDepth = maxDepth; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_BOX : public D3D12_BOX +{ + CD3DX12_BOX() = default; + explicit CD3DX12_BOX( const D3D12_BOX& o ) noexcept : + D3D12_BOX( o ) + {} + explicit CD3DX12_BOX( + LONG Left, + LONG Right ) noexcept + { + left = static_cast(Left); + top = 0; + front = 0; + right = static_cast(Right); + bottom = 1; + back = 1; + } + explicit CD3DX12_BOX( + LONG Left, + LONG Top, + LONG Right, + LONG Bottom ) noexcept + { + left = static_cast(Left); + top = static_cast(Top); + front = 0; + right = static_cast(Right); + bottom = static_cast(Bottom); + back = 1; + } + explicit CD3DX12_BOX( + LONG Left, + LONG Top, + LONG Front, + LONG Right, + LONG Bottom, + LONG Back ) noexcept + { + left = static_cast(Left); + top = static_cast(Top); + front = static_cast(Front); + right = static_cast(Right); + bottom = static_cast(Bottom); + back = static_cast(Back); + } +}; +inline bool operator==( const D3D12_BOX& l, const D3D12_BOX& r ) noexcept +{ + return l.left == r.left && l.top == r.top && l.front == r.front && + l.right == r.right && l.bottom == r.bottom && l.back == r.back; +} +inline bool operator!=( const D3D12_BOX& l, const D3D12_BOX& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_DEPTH_STENCIL_DESC : public D3D12_DEPTH_STENCIL_DESC +{ + CD3DX12_DEPTH_STENCIL_DESC() = default; + explicit CD3DX12_DEPTH_STENCIL_DESC( const D3D12_DEPTH_STENCIL_DESC& o ) noexcept : + D3D12_DEPTH_STENCIL_DESC( o ) + {} + explicit CD3DX12_DEPTH_STENCIL_DESC( CD3DX12_DEFAULT ) noexcept + { + DepthEnable = TRUE; + DepthWriteMask = D3D12_DEPTH_WRITE_MASK_ALL; + DepthFunc = D3D12_COMPARISON_FUNC_LESS; + StencilEnable = FALSE; + StencilReadMask = D3D12_DEFAULT_STENCIL_READ_MASK; + StencilWriteMask = D3D12_DEFAULT_STENCIL_WRITE_MASK; + const D3D12_DEPTH_STENCILOP_DESC defaultStencilOp = + { D3D12_STENCIL_OP_KEEP, D3D12_STENCIL_OP_KEEP, D3D12_STENCIL_OP_KEEP, D3D12_COMPARISON_FUNC_ALWAYS }; + FrontFace = defaultStencilOp; + BackFace = defaultStencilOp; + } + explicit CD3DX12_DEPTH_STENCIL_DESC( + BOOL depthEnable, + D3D12_DEPTH_WRITE_MASK depthWriteMask, + D3D12_COMPARISON_FUNC depthFunc, + BOOL stencilEnable, + UINT8 stencilReadMask, + UINT8 stencilWriteMask, + D3D12_STENCIL_OP frontStencilFailOp, + D3D12_STENCIL_OP frontStencilDepthFailOp, + D3D12_STENCIL_OP frontStencilPassOp, + D3D12_COMPARISON_FUNC frontStencilFunc, + D3D12_STENCIL_OP backStencilFailOp, + D3D12_STENCIL_OP backStencilDepthFailOp, + D3D12_STENCIL_OP backStencilPassOp, + D3D12_COMPARISON_FUNC backStencilFunc ) noexcept + { + DepthEnable = depthEnable; + DepthWriteMask = depthWriteMask; + DepthFunc = depthFunc; + StencilEnable = stencilEnable; + StencilReadMask = stencilReadMask; + StencilWriteMask = stencilWriteMask; + FrontFace.StencilFailOp = frontStencilFailOp; + FrontFace.StencilDepthFailOp = frontStencilDepthFailOp; + FrontFace.StencilPassOp = frontStencilPassOp; + FrontFace.StencilFunc = frontStencilFunc; + BackFace.StencilFailOp = backStencilFailOp; + BackFace.StencilDepthFailOp = backStencilDepthFailOp; + BackFace.StencilPassOp = backStencilPassOp; + BackFace.StencilFunc = backStencilFunc; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_DEPTH_STENCIL_DESC1 : public D3D12_DEPTH_STENCIL_DESC1 +{ + CD3DX12_DEPTH_STENCIL_DESC1() = default; + explicit CD3DX12_DEPTH_STENCIL_DESC1( const D3D12_DEPTH_STENCIL_DESC1& o ) noexcept : + D3D12_DEPTH_STENCIL_DESC1( o ) + {} + explicit CD3DX12_DEPTH_STENCIL_DESC1( const D3D12_DEPTH_STENCIL_DESC& o ) noexcept + { + DepthEnable = o.DepthEnable; + DepthWriteMask = o.DepthWriteMask; + DepthFunc = o.DepthFunc; + StencilEnable = o.StencilEnable; + StencilReadMask = o.StencilReadMask; + StencilWriteMask = o.StencilWriteMask; + FrontFace.StencilFailOp = o.FrontFace.StencilFailOp; + FrontFace.StencilDepthFailOp = o.FrontFace.StencilDepthFailOp; + FrontFace.StencilPassOp = o.FrontFace.StencilPassOp; + FrontFace.StencilFunc = o.FrontFace.StencilFunc; + BackFace.StencilFailOp = o.BackFace.StencilFailOp; + BackFace.StencilDepthFailOp = o.BackFace.StencilDepthFailOp; + BackFace.StencilPassOp = o.BackFace.StencilPassOp; + BackFace.StencilFunc = o.BackFace.StencilFunc; + DepthBoundsTestEnable = FALSE; + } + explicit CD3DX12_DEPTH_STENCIL_DESC1( CD3DX12_DEFAULT ) noexcept + { + DepthEnable = TRUE; + DepthWriteMask = D3D12_DEPTH_WRITE_MASK_ALL; + DepthFunc = D3D12_COMPARISON_FUNC_LESS; + StencilEnable = FALSE; + StencilReadMask = D3D12_DEFAULT_STENCIL_READ_MASK; + StencilWriteMask = D3D12_DEFAULT_STENCIL_WRITE_MASK; + const D3D12_DEPTH_STENCILOP_DESC defaultStencilOp = + { D3D12_STENCIL_OP_KEEP, D3D12_STENCIL_OP_KEEP, D3D12_STENCIL_OP_KEEP, D3D12_COMPARISON_FUNC_ALWAYS }; + FrontFace = defaultStencilOp; + BackFace = defaultStencilOp; + DepthBoundsTestEnable = FALSE; + } + explicit CD3DX12_DEPTH_STENCIL_DESC1( + BOOL depthEnable, + D3D12_DEPTH_WRITE_MASK depthWriteMask, + D3D12_COMPARISON_FUNC depthFunc, + BOOL stencilEnable, + UINT8 stencilReadMask, + UINT8 stencilWriteMask, + D3D12_STENCIL_OP frontStencilFailOp, + D3D12_STENCIL_OP frontStencilDepthFailOp, + D3D12_STENCIL_OP frontStencilPassOp, + D3D12_COMPARISON_FUNC frontStencilFunc, + D3D12_STENCIL_OP backStencilFailOp, + D3D12_STENCIL_OP backStencilDepthFailOp, + D3D12_STENCIL_OP backStencilPassOp, + D3D12_COMPARISON_FUNC backStencilFunc, + BOOL depthBoundsTestEnable ) noexcept + { + DepthEnable = depthEnable; + DepthWriteMask = depthWriteMask; + DepthFunc = depthFunc; + StencilEnable = stencilEnable; + StencilReadMask = stencilReadMask; + StencilWriteMask = stencilWriteMask; + FrontFace.StencilFailOp = frontStencilFailOp; + FrontFace.StencilDepthFailOp = frontStencilDepthFailOp; + FrontFace.StencilPassOp = frontStencilPassOp; + FrontFace.StencilFunc = frontStencilFunc; + BackFace.StencilFailOp = backStencilFailOp; + BackFace.StencilDepthFailOp = backStencilDepthFailOp; + BackFace.StencilPassOp = backStencilPassOp; + BackFace.StencilFunc = backStencilFunc; + DepthBoundsTestEnable = depthBoundsTestEnable; + } + operator D3D12_DEPTH_STENCIL_DESC() const noexcept + { + D3D12_DEPTH_STENCIL_DESC D; + D.DepthEnable = DepthEnable; + D.DepthWriteMask = DepthWriteMask; + D.DepthFunc = DepthFunc; + D.StencilEnable = StencilEnable; + D.StencilReadMask = StencilReadMask; + D.StencilWriteMask = StencilWriteMask; + D.FrontFace.StencilFailOp = FrontFace.StencilFailOp; + D.FrontFace.StencilDepthFailOp = FrontFace.StencilDepthFailOp; + D.FrontFace.StencilPassOp = FrontFace.StencilPassOp; + D.FrontFace.StencilFunc = FrontFace.StencilFunc; + D.BackFace.StencilFailOp = BackFace.StencilFailOp; + D.BackFace.StencilDepthFailOp = BackFace.StencilDepthFailOp; + D.BackFace.StencilPassOp = BackFace.StencilPassOp; + D.BackFace.StencilFunc = BackFace.StencilFunc; + return D; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_BLEND_DESC : public D3D12_BLEND_DESC +{ + CD3DX12_BLEND_DESC() = default; + explicit CD3DX12_BLEND_DESC( const D3D12_BLEND_DESC& o ) noexcept : + D3D12_BLEND_DESC( o ) + {} + explicit CD3DX12_BLEND_DESC( CD3DX12_DEFAULT ) noexcept + { + AlphaToCoverageEnable = FALSE; + IndependentBlendEnable = FALSE; + const D3D12_RENDER_TARGET_BLEND_DESC defaultRenderTargetBlendDesc = + { + FALSE,FALSE, + D3D12_BLEND_ONE, D3D12_BLEND_ZERO, D3D12_BLEND_OP_ADD, + D3D12_BLEND_ONE, D3D12_BLEND_ZERO, D3D12_BLEND_OP_ADD, + D3D12_LOGIC_OP_NOOP, + D3D12_COLOR_WRITE_ENABLE_ALL, + }; + for (UINT i = 0; i < D3D12_SIMULTANEOUS_RENDER_TARGET_COUNT; ++i) + RenderTarget[ i ] = defaultRenderTargetBlendDesc; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RASTERIZER_DESC : public D3D12_RASTERIZER_DESC +{ + CD3DX12_RASTERIZER_DESC() = default; + explicit CD3DX12_RASTERIZER_DESC( const D3D12_RASTERIZER_DESC& o ) noexcept : + D3D12_RASTERIZER_DESC( o ) + {} + explicit CD3DX12_RASTERIZER_DESC( CD3DX12_DEFAULT ) noexcept + { + FillMode = D3D12_FILL_MODE_SOLID; + CullMode = D3D12_CULL_MODE_BACK; + FrontCounterClockwise = FALSE; + DepthBias = D3D12_DEFAULT_DEPTH_BIAS; + DepthBiasClamp = D3D12_DEFAULT_DEPTH_BIAS_CLAMP; + SlopeScaledDepthBias = D3D12_DEFAULT_SLOPE_SCALED_DEPTH_BIAS; + DepthClipEnable = TRUE; + MultisampleEnable = FALSE; + AntialiasedLineEnable = FALSE; + ForcedSampleCount = 0; + ConservativeRaster = D3D12_CONSERVATIVE_RASTERIZATION_MODE_OFF; + } + explicit CD3DX12_RASTERIZER_DESC( + D3D12_FILL_MODE fillMode, + D3D12_CULL_MODE cullMode, + BOOL frontCounterClockwise, + INT depthBias, + FLOAT depthBiasClamp, + FLOAT slopeScaledDepthBias, + BOOL depthClipEnable, + BOOL multisampleEnable, + BOOL antialiasedLineEnable, + UINT forcedSampleCount, + D3D12_CONSERVATIVE_RASTERIZATION_MODE conservativeRaster) noexcept + { + FillMode = fillMode; + CullMode = cullMode; + FrontCounterClockwise = frontCounterClockwise; + DepthBias = depthBias; + DepthBiasClamp = depthBiasClamp; + SlopeScaledDepthBias = slopeScaledDepthBias; + DepthClipEnable = depthClipEnable; + MultisampleEnable = multisampleEnable; + AntialiasedLineEnable = antialiasedLineEnable; + ForcedSampleCount = forcedSampleCount; + ConservativeRaster = conservativeRaster; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RESOURCE_ALLOCATION_INFO : public D3D12_RESOURCE_ALLOCATION_INFO +{ + CD3DX12_RESOURCE_ALLOCATION_INFO() = default; + explicit CD3DX12_RESOURCE_ALLOCATION_INFO( const D3D12_RESOURCE_ALLOCATION_INFO& o ) noexcept : + D3D12_RESOURCE_ALLOCATION_INFO( o ) + {} + CD3DX12_RESOURCE_ALLOCATION_INFO( + UINT64 size, + UINT64 alignment ) noexcept + { + SizeInBytes = size; + Alignment = alignment; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_HEAP_PROPERTIES : public D3D12_HEAP_PROPERTIES +{ + CD3DX12_HEAP_PROPERTIES() = default; + explicit CD3DX12_HEAP_PROPERTIES(const D3D12_HEAP_PROPERTIES &o) noexcept : + D3D12_HEAP_PROPERTIES(o) + {} + CD3DX12_HEAP_PROPERTIES( + D3D12_CPU_PAGE_PROPERTY cpuPageProperty, + D3D12_MEMORY_POOL memoryPoolPreference, + UINT creationNodeMask = 1, + UINT nodeMask = 1 ) noexcept + { + Type = D3D12_HEAP_TYPE_CUSTOM; + CPUPageProperty = cpuPageProperty; + MemoryPoolPreference = memoryPoolPreference; + CreationNodeMask = creationNodeMask; + VisibleNodeMask = nodeMask; + } + explicit CD3DX12_HEAP_PROPERTIES( + D3D12_HEAP_TYPE type, + UINT creationNodeMask = 1, + UINT nodeMask = 1 ) noexcept + { + Type = type; + CPUPageProperty = D3D12_CPU_PAGE_PROPERTY_UNKNOWN; + MemoryPoolPreference = D3D12_MEMORY_POOL_UNKNOWN; + CreationNodeMask = creationNodeMask; + VisibleNodeMask = nodeMask; + } + bool IsCPUAccessible() const noexcept + { + return Type == D3D12_HEAP_TYPE_UPLOAD || Type == D3D12_HEAP_TYPE_READBACK || (Type == D3D12_HEAP_TYPE_CUSTOM && + (CPUPageProperty == D3D12_CPU_PAGE_PROPERTY_WRITE_COMBINE || CPUPageProperty == D3D12_CPU_PAGE_PROPERTY_WRITE_BACK)); + } +}; +inline bool operator==( const D3D12_HEAP_PROPERTIES& l, const D3D12_HEAP_PROPERTIES& r ) noexcept +{ + return l.Type == r.Type && l.CPUPageProperty == r.CPUPageProperty && + l.MemoryPoolPreference == r.MemoryPoolPreference && + l.CreationNodeMask == r.CreationNodeMask && + l.VisibleNodeMask == r.VisibleNodeMask; +} +inline bool operator!=( const D3D12_HEAP_PROPERTIES& l, const D3D12_HEAP_PROPERTIES& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_HEAP_DESC : public D3D12_HEAP_DESC +{ + CD3DX12_HEAP_DESC() = default; + explicit CD3DX12_HEAP_DESC(const D3D12_HEAP_DESC &o) noexcept : + D3D12_HEAP_DESC(o) + {} + CD3DX12_HEAP_DESC( + UINT64 size, + D3D12_HEAP_PROPERTIES properties, + UINT64 alignment = 0, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = size; + Properties = properties; + Alignment = alignment; + Flags = flags; + } + CD3DX12_HEAP_DESC( + UINT64 size, + D3D12_HEAP_TYPE type, + UINT64 alignment = 0, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = size; + Properties = CD3DX12_HEAP_PROPERTIES( type ); + Alignment = alignment; + Flags = flags; + } + CD3DX12_HEAP_DESC( + UINT64 size, + D3D12_CPU_PAGE_PROPERTY cpuPageProperty, + D3D12_MEMORY_POOL memoryPoolPreference, + UINT64 alignment = 0, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = size; + Properties = CD3DX12_HEAP_PROPERTIES( cpuPageProperty, memoryPoolPreference ); + Alignment = alignment; + Flags = flags; + } + CD3DX12_HEAP_DESC( + const D3D12_RESOURCE_ALLOCATION_INFO& resAllocInfo, + D3D12_HEAP_PROPERTIES properties, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = resAllocInfo.SizeInBytes; + Properties = properties; + Alignment = resAllocInfo.Alignment; + Flags = flags; + } + CD3DX12_HEAP_DESC( + const D3D12_RESOURCE_ALLOCATION_INFO& resAllocInfo, + D3D12_HEAP_TYPE type, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = resAllocInfo.SizeInBytes; + Properties = CD3DX12_HEAP_PROPERTIES( type ); + Alignment = resAllocInfo.Alignment; + Flags = flags; + } + CD3DX12_HEAP_DESC( + const D3D12_RESOURCE_ALLOCATION_INFO& resAllocInfo, + D3D12_CPU_PAGE_PROPERTY cpuPageProperty, + D3D12_MEMORY_POOL memoryPoolPreference, + D3D12_HEAP_FLAGS flags = D3D12_HEAP_FLAG_NONE ) noexcept + { + SizeInBytes = resAllocInfo.SizeInBytes; + Properties = CD3DX12_HEAP_PROPERTIES( cpuPageProperty, memoryPoolPreference ); + Alignment = resAllocInfo.Alignment; + Flags = flags; + } + bool IsCPUAccessible() const noexcept + { return static_cast< const CD3DX12_HEAP_PROPERTIES* >( &Properties )->IsCPUAccessible(); } +}; +inline bool operator==( const D3D12_HEAP_DESC& l, const D3D12_HEAP_DESC& r ) noexcept +{ + return l.SizeInBytes == r.SizeInBytes && + l.Properties == r.Properties && + l.Alignment == r.Alignment && + l.Flags == r.Flags; +} +inline bool operator!=( const D3D12_HEAP_DESC& l, const D3D12_HEAP_DESC& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_CLEAR_VALUE : public D3D12_CLEAR_VALUE +{ + CD3DX12_CLEAR_VALUE() = default; + explicit CD3DX12_CLEAR_VALUE(const D3D12_CLEAR_VALUE &o) noexcept : + D3D12_CLEAR_VALUE(o) + {} + CD3DX12_CLEAR_VALUE( + DXGI_FORMAT format, + const FLOAT color[4] ) noexcept + { + Format = format; + memcpy( Color, color, sizeof( Color ) ); + } + CD3DX12_CLEAR_VALUE( + DXGI_FORMAT format, + FLOAT depth, + UINT8 stencil ) noexcept + { + Format = format; + memset( &Color, 0, sizeof( Color ) ); + /* Use memcpy to preserve NAN values */ + memcpy( &DepthStencil.Depth, &depth, sizeof( depth ) ); + DepthStencil.Stencil = stencil; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RANGE : public D3D12_RANGE +{ + CD3DX12_RANGE() = default; + explicit CD3DX12_RANGE(const D3D12_RANGE &o) noexcept : + D3D12_RANGE(o) + {} + CD3DX12_RANGE( + SIZE_T begin, + SIZE_T end ) noexcept + { + Begin = begin; + End = end; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RANGE_UINT64 : public D3D12_RANGE_UINT64 +{ + CD3DX12_RANGE_UINT64() = default; + explicit CD3DX12_RANGE_UINT64(const D3D12_RANGE_UINT64 &o) noexcept : + D3D12_RANGE_UINT64(o) + {} + CD3DX12_RANGE_UINT64( + UINT64 begin, + UINT64 end ) noexcept + { + Begin = begin; + End = end; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_SUBRESOURCE_RANGE_UINT64 : public D3D12_SUBRESOURCE_RANGE_UINT64 +{ + CD3DX12_SUBRESOURCE_RANGE_UINT64() = default; + explicit CD3DX12_SUBRESOURCE_RANGE_UINT64(const D3D12_SUBRESOURCE_RANGE_UINT64 &o) noexcept : + D3D12_SUBRESOURCE_RANGE_UINT64(o) + {} + CD3DX12_SUBRESOURCE_RANGE_UINT64( + UINT subresource, + const D3D12_RANGE_UINT64& range ) noexcept + { + Subresource = subresource; + Range = range; + } + CD3DX12_SUBRESOURCE_RANGE_UINT64( + UINT subresource, + UINT64 begin, + UINT64 end ) noexcept + { + Subresource = subresource; + Range.Begin = begin; + Range.End = end; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_SHADER_BYTECODE : public D3D12_SHADER_BYTECODE +{ + CD3DX12_SHADER_BYTECODE() = default; + explicit CD3DX12_SHADER_BYTECODE(const D3D12_SHADER_BYTECODE &o) noexcept : + D3D12_SHADER_BYTECODE(o) + {} + CD3DX12_SHADER_BYTECODE( + _In_ ID3DBlob* pShaderBlob ) noexcept + { + pShaderBytecode = pShaderBlob->GetBufferPointer(); + BytecodeLength = pShaderBlob->GetBufferSize(); + } + CD3DX12_SHADER_BYTECODE( + const void* _pShaderBytecode, + SIZE_T bytecodeLength ) noexcept + { + pShaderBytecode = _pShaderBytecode; + BytecodeLength = bytecodeLength; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_TILED_RESOURCE_COORDINATE : public D3D12_TILED_RESOURCE_COORDINATE +{ + CD3DX12_TILED_RESOURCE_COORDINATE() = default; + explicit CD3DX12_TILED_RESOURCE_COORDINATE(const D3D12_TILED_RESOURCE_COORDINATE &o) noexcept : + D3D12_TILED_RESOURCE_COORDINATE(o) + {} + CD3DX12_TILED_RESOURCE_COORDINATE( + UINT x, + UINT y, + UINT z, + UINT subresource ) noexcept + { + X = x; + Y = y; + Z = z; + Subresource = subresource; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_TILE_REGION_SIZE : public D3D12_TILE_REGION_SIZE +{ + CD3DX12_TILE_REGION_SIZE() = default; + explicit CD3DX12_TILE_REGION_SIZE(const D3D12_TILE_REGION_SIZE &o) noexcept : + D3D12_TILE_REGION_SIZE(o) + {} + CD3DX12_TILE_REGION_SIZE( + UINT numTiles, + BOOL useBox, + UINT width, + UINT16 height, + UINT16 depth ) noexcept + { + NumTiles = numTiles; + UseBox = useBox; + Width = width; + Height = height; + Depth = depth; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_SUBRESOURCE_TILING : public D3D12_SUBRESOURCE_TILING +{ + CD3DX12_SUBRESOURCE_TILING() = default; + explicit CD3DX12_SUBRESOURCE_TILING(const D3D12_SUBRESOURCE_TILING &o) noexcept : + D3D12_SUBRESOURCE_TILING(o) + {} + CD3DX12_SUBRESOURCE_TILING( + UINT widthInTiles, + UINT16 heightInTiles, + UINT16 depthInTiles, + UINT startTileIndexInOverallResource ) noexcept + { + WidthInTiles = widthInTiles; + HeightInTiles = heightInTiles; + DepthInTiles = depthInTiles; + StartTileIndexInOverallResource = startTileIndexInOverallResource; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_TILE_SHAPE : public D3D12_TILE_SHAPE +{ + CD3DX12_TILE_SHAPE() = default; + explicit CD3DX12_TILE_SHAPE(const D3D12_TILE_SHAPE &o) noexcept : + D3D12_TILE_SHAPE(o) + {} + CD3DX12_TILE_SHAPE( + UINT widthInTexels, + UINT heightInTexels, + UINT depthInTexels ) noexcept + { + WidthInTexels = widthInTexels; + HeightInTexels = heightInTexels; + DepthInTexels = depthInTexels; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RESOURCE_BARRIER : public D3D12_RESOURCE_BARRIER +{ + CD3DX12_RESOURCE_BARRIER() = default; + explicit CD3DX12_RESOURCE_BARRIER(const D3D12_RESOURCE_BARRIER &o) noexcept : + D3D12_RESOURCE_BARRIER(o) + {} + static inline CD3DX12_RESOURCE_BARRIER Transition( + _In_ ID3D12Resource* pResource, + D3D12_RESOURCE_STATES stateBefore, + D3D12_RESOURCE_STATES stateAfter, + UINT subresource = D3D12_RESOURCE_BARRIER_ALL_SUBRESOURCES, + D3D12_RESOURCE_BARRIER_FLAGS flags = D3D12_RESOURCE_BARRIER_FLAG_NONE) noexcept + { + CD3DX12_RESOURCE_BARRIER result = {}; + D3D12_RESOURCE_BARRIER &barrier = result; + result.Type = D3D12_RESOURCE_BARRIER_TYPE_TRANSITION; + result.Flags = flags; + barrier.Transition.pResource = pResource; + barrier.Transition.StateBefore = stateBefore; + barrier.Transition.StateAfter = stateAfter; + barrier.Transition.Subresource = subresource; + return result; + } + static inline CD3DX12_RESOURCE_BARRIER Aliasing( + _In_ ID3D12Resource* pResourceBefore, + _In_ ID3D12Resource* pResourceAfter) noexcept + { + CD3DX12_RESOURCE_BARRIER result = {}; + D3D12_RESOURCE_BARRIER &barrier = result; + result.Type = D3D12_RESOURCE_BARRIER_TYPE_ALIASING; + barrier.Aliasing.pResourceBefore = pResourceBefore; + barrier.Aliasing.pResourceAfter = pResourceAfter; + return result; + } + static inline CD3DX12_RESOURCE_BARRIER UAV( + _In_ ID3D12Resource* pResource) noexcept + { + CD3DX12_RESOURCE_BARRIER result = {}; + D3D12_RESOURCE_BARRIER &barrier = result; + result.Type = D3D12_RESOURCE_BARRIER_TYPE_UAV; + barrier.UAV.pResource = pResource; + return result; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_PACKED_MIP_INFO : public D3D12_PACKED_MIP_INFO +{ + CD3DX12_PACKED_MIP_INFO() = default; + explicit CD3DX12_PACKED_MIP_INFO(const D3D12_PACKED_MIP_INFO &o) noexcept : + D3D12_PACKED_MIP_INFO(o) + {} + CD3DX12_PACKED_MIP_INFO( + UINT8 numStandardMips, + UINT8 numPackedMips, + UINT numTilesForPackedMips, + UINT startTileIndexInOverallResource ) noexcept + { + NumStandardMips = numStandardMips; + NumPackedMips = numPackedMips; + NumTilesForPackedMips = numTilesForPackedMips; + StartTileIndexInOverallResource = startTileIndexInOverallResource; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_SUBRESOURCE_FOOTPRINT : public D3D12_SUBRESOURCE_FOOTPRINT +{ + CD3DX12_SUBRESOURCE_FOOTPRINT() = default; + explicit CD3DX12_SUBRESOURCE_FOOTPRINT(const D3D12_SUBRESOURCE_FOOTPRINT &o) noexcept : + D3D12_SUBRESOURCE_FOOTPRINT(o) + {} + CD3DX12_SUBRESOURCE_FOOTPRINT( + DXGI_FORMAT format, + UINT width, + UINT height, + UINT depth, + UINT rowPitch ) noexcept + { + Format = format; + Width = width; + Height = height; + Depth = depth; + RowPitch = rowPitch; + } + explicit CD3DX12_SUBRESOURCE_FOOTPRINT( + const D3D12_RESOURCE_DESC& resDesc, + UINT rowPitch ) noexcept + { + Format = resDesc.Format; + Width = UINT( resDesc.Width ); + Height = resDesc.Height; + Depth = (resDesc.Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE3D ? resDesc.DepthOrArraySize : 1); + RowPitch = rowPitch; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_TEXTURE_COPY_LOCATION : public D3D12_TEXTURE_COPY_LOCATION +{ + CD3DX12_TEXTURE_COPY_LOCATION() = default; + explicit CD3DX12_TEXTURE_COPY_LOCATION(const D3D12_TEXTURE_COPY_LOCATION &o) noexcept : + D3D12_TEXTURE_COPY_LOCATION(o) + {} + CD3DX12_TEXTURE_COPY_LOCATION(_In_ ID3D12Resource* pRes) noexcept + { + pResource = pRes; + Type = D3D12_TEXTURE_COPY_TYPE_SUBRESOURCE_INDEX; + PlacedFootprint = {}; + } + CD3DX12_TEXTURE_COPY_LOCATION(_In_ ID3D12Resource* pRes, D3D12_PLACED_SUBRESOURCE_FOOTPRINT const& Footprint) noexcept + { + pResource = pRes; + Type = D3D12_TEXTURE_COPY_TYPE_PLACED_FOOTPRINT; + PlacedFootprint = Footprint; + } + CD3DX12_TEXTURE_COPY_LOCATION(_In_ ID3D12Resource* pRes, UINT Sub) noexcept + { + pResource = pRes; + Type = D3D12_TEXTURE_COPY_TYPE_SUBRESOURCE_INDEX; + PlacedFootprint = {}; + SubresourceIndex = Sub; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_DESCRIPTOR_RANGE : public D3D12_DESCRIPTOR_RANGE +{ + CD3DX12_DESCRIPTOR_RANGE() = default; + explicit CD3DX12_DESCRIPTOR_RANGE(const D3D12_DESCRIPTOR_RANGE &o) noexcept : + D3D12_DESCRIPTOR_RANGE(o) + {} + CD3DX12_DESCRIPTOR_RANGE( + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + Init(rangeType, numDescriptors, baseShaderRegister, registerSpace, offsetInDescriptorsFromTableStart); + } + + inline void Init( + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + Init(*this, rangeType, numDescriptors, baseShaderRegister, registerSpace, offsetInDescriptorsFromTableStart); + } + + static inline void Init( + _Out_ D3D12_DESCRIPTOR_RANGE &range, + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + range.RangeType = rangeType; + range.NumDescriptors = numDescriptors; + range.BaseShaderRegister = baseShaderRegister; + range.RegisterSpace = registerSpace; + range.OffsetInDescriptorsFromTableStart = offsetInDescriptorsFromTableStart; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_DESCRIPTOR_TABLE : public D3D12_ROOT_DESCRIPTOR_TABLE +{ + CD3DX12_ROOT_DESCRIPTOR_TABLE() = default; + explicit CD3DX12_ROOT_DESCRIPTOR_TABLE(const D3D12_ROOT_DESCRIPTOR_TABLE &o) noexcept : + D3D12_ROOT_DESCRIPTOR_TABLE(o) + {} + CD3DX12_ROOT_DESCRIPTOR_TABLE( + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE* _pDescriptorRanges) noexcept + { + Init(numDescriptorRanges, _pDescriptorRanges); + } + + inline void Init( + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE* _pDescriptorRanges) noexcept + { + Init(*this, numDescriptorRanges, _pDescriptorRanges); + } + + static inline void Init( + _Out_ D3D12_ROOT_DESCRIPTOR_TABLE &rootDescriptorTable, + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE* _pDescriptorRanges) noexcept + { + rootDescriptorTable.NumDescriptorRanges = numDescriptorRanges; + rootDescriptorTable.pDescriptorRanges = _pDescriptorRanges; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_CONSTANTS : public D3D12_ROOT_CONSTANTS +{ + CD3DX12_ROOT_CONSTANTS() = default; + explicit CD3DX12_ROOT_CONSTANTS(const D3D12_ROOT_CONSTANTS &o) noexcept : + D3D12_ROOT_CONSTANTS(o) + {} + CD3DX12_ROOT_CONSTANTS( + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0) noexcept + { + Init(num32BitValues, shaderRegister, registerSpace); + } + + inline void Init( + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0) noexcept + { + Init(*this, num32BitValues, shaderRegister, registerSpace); + } + + static inline void Init( + _Out_ D3D12_ROOT_CONSTANTS &rootConstants, + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0) noexcept + { + rootConstants.Num32BitValues = num32BitValues; + rootConstants.ShaderRegister = shaderRegister; + rootConstants.RegisterSpace = registerSpace; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_DESCRIPTOR : public D3D12_ROOT_DESCRIPTOR +{ + CD3DX12_ROOT_DESCRIPTOR() = default; + explicit CD3DX12_ROOT_DESCRIPTOR(const D3D12_ROOT_DESCRIPTOR &o) noexcept : + D3D12_ROOT_DESCRIPTOR(o) + {} + CD3DX12_ROOT_DESCRIPTOR( + UINT shaderRegister, + UINT registerSpace = 0) noexcept + { + Init(shaderRegister, registerSpace); + } + + inline void Init( + UINT shaderRegister, + UINT registerSpace = 0) noexcept + { + Init(*this, shaderRegister, registerSpace); + } + + static inline void Init(_Out_ D3D12_ROOT_DESCRIPTOR &table, UINT shaderRegister, UINT registerSpace = 0) noexcept + { + table.ShaderRegister = shaderRegister; + table.RegisterSpace = registerSpace; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_PARAMETER : public D3D12_ROOT_PARAMETER +{ + CD3DX12_ROOT_PARAMETER() = default; + explicit CD3DX12_ROOT_PARAMETER(const D3D12_ROOT_PARAMETER &o) noexcept : + D3D12_ROOT_PARAMETER(o) + {} + + static inline void InitAsDescriptorTable( + _Out_ D3D12_ROOT_PARAMETER &rootParam, + UINT numDescriptorRanges, + _In_reads_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE* pDescriptorRanges, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR_TABLE::Init(rootParam.DescriptorTable, numDescriptorRanges, pDescriptorRanges); + } + + static inline void InitAsConstants( + _Out_ D3D12_ROOT_PARAMETER &rootParam, + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_32BIT_CONSTANTS; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_CONSTANTS::Init(rootParam.Constants, num32BitValues, shaderRegister, registerSpace); + } + + static inline void InitAsConstantBufferView( + _Out_ D3D12_ROOT_PARAMETER &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_CBV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR::Init(rootParam.Descriptor, shaderRegister, registerSpace); + } + + static inline void InitAsShaderResourceView( + _Out_ D3D12_ROOT_PARAMETER &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_SRV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR::Init(rootParam.Descriptor, shaderRegister, registerSpace); + } + + static inline void InitAsUnorderedAccessView( + _Out_ D3D12_ROOT_PARAMETER &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_UAV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR::Init(rootParam.Descriptor, shaderRegister, registerSpace); + } + + inline void InitAsDescriptorTable( + UINT numDescriptorRanges, + _In_reads_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE* pDescriptorRanges, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsDescriptorTable(*this, numDescriptorRanges, pDescriptorRanges, visibility); + } + + inline void InitAsConstants( + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsConstants(*this, num32BitValues, shaderRegister, registerSpace, visibility); + } + + inline void InitAsConstantBufferView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsConstantBufferView(*this, shaderRegister, registerSpace, visibility); + } + + inline void InitAsShaderResourceView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsShaderResourceView(*this, shaderRegister, registerSpace, visibility); + } + + inline void InitAsUnorderedAccessView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsUnorderedAccessView(*this, shaderRegister, registerSpace, visibility); + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_STATIC_SAMPLER_DESC : public D3D12_STATIC_SAMPLER_DESC +{ + CD3DX12_STATIC_SAMPLER_DESC() = default; + explicit CD3DX12_STATIC_SAMPLER_DESC(const D3D12_STATIC_SAMPLER_DESC &o) noexcept : + D3D12_STATIC_SAMPLER_DESC(o) + {} + CD3DX12_STATIC_SAMPLER_DESC( + UINT shaderRegister, + D3D12_FILTER filter = D3D12_FILTER_ANISOTROPIC, + D3D12_TEXTURE_ADDRESS_MODE addressU = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressV = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressW = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + FLOAT mipLODBias = 0, + UINT maxAnisotropy = 16, + D3D12_COMPARISON_FUNC comparisonFunc = D3D12_COMPARISON_FUNC_LESS_EQUAL, + D3D12_STATIC_BORDER_COLOR borderColor = D3D12_STATIC_BORDER_COLOR_OPAQUE_WHITE, + FLOAT minLOD = 0.f, + FLOAT maxLOD = D3D12_FLOAT32_MAX, + D3D12_SHADER_VISIBILITY shaderVisibility = D3D12_SHADER_VISIBILITY_ALL, + UINT registerSpace = 0) noexcept + { + Init( + shaderRegister, + filter, + addressU, + addressV, + addressW, + mipLODBias, + maxAnisotropy, + comparisonFunc, + borderColor, + minLOD, + maxLOD, + shaderVisibility, + registerSpace); + } + + static inline void Init( + _Out_ D3D12_STATIC_SAMPLER_DESC &samplerDesc, + UINT shaderRegister, + D3D12_FILTER filter = D3D12_FILTER_ANISOTROPIC, + D3D12_TEXTURE_ADDRESS_MODE addressU = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressV = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressW = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + FLOAT mipLODBias = 0, + UINT maxAnisotropy = 16, + D3D12_COMPARISON_FUNC comparisonFunc = D3D12_COMPARISON_FUNC_LESS_EQUAL, + D3D12_STATIC_BORDER_COLOR borderColor = D3D12_STATIC_BORDER_COLOR_OPAQUE_WHITE, + FLOAT minLOD = 0.f, + FLOAT maxLOD = D3D12_FLOAT32_MAX, + D3D12_SHADER_VISIBILITY shaderVisibility = D3D12_SHADER_VISIBILITY_ALL, + UINT registerSpace = 0) noexcept + { + samplerDesc.ShaderRegister = shaderRegister; + samplerDesc.Filter = filter; + samplerDesc.AddressU = addressU; + samplerDesc.AddressV = addressV; + samplerDesc.AddressW = addressW; + samplerDesc.MipLODBias = mipLODBias; + samplerDesc.MaxAnisotropy = maxAnisotropy; + samplerDesc.ComparisonFunc = comparisonFunc; + samplerDesc.BorderColor = borderColor; + samplerDesc.MinLOD = minLOD; + samplerDesc.MaxLOD = maxLOD; + samplerDesc.ShaderVisibility = shaderVisibility; + samplerDesc.RegisterSpace = registerSpace; + } + inline void Init( + UINT shaderRegister, + D3D12_FILTER filter = D3D12_FILTER_ANISOTROPIC, + D3D12_TEXTURE_ADDRESS_MODE addressU = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressV = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + D3D12_TEXTURE_ADDRESS_MODE addressW = D3D12_TEXTURE_ADDRESS_MODE_WRAP, + FLOAT mipLODBias = 0, + UINT maxAnisotropy = 16, + D3D12_COMPARISON_FUNC comparisonFunc = D3D12_COMPARISON_FUNC_LESS_EQUAL, + D3D12_STATIC_BORDER_COLOR borderColor = D3D12_STATIC_BORDER_COLOR_OPAQUE_WHITE, + FLOAT minLOD = 0.f, + FLOAT maxLOD = D3D12_FLOAT32_MAX, + D3D12_SHADER_VISIBILITY shaderVisibility = D3D12_SHADER_VISIBILITY_ALL, + UINT registerSpace = 0) noexcept + { + Init( + *this, + shaderRegister, + filter, + addressU, + addressV, + addressW, + mipLODBias, + maxAnisotropy, + comparisonFunc, + borderColor, + minLOD, + maxLOD, + shaderVisibility, + registerSpace); + } + +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_SIGNATURE_DESC : public D3D12_ROOT_SIGNATURE_DESC +{ + CD3DX12_ROOT_SIGNATURE_DESC() = default; + explicit CD3DX12_ROOT_SIGNATURE_DESC(const D3D12_ROOT_SIGNATURE_DESC &o) noexcept : + D3D12_ROOT_SIGNATURE_DESC(o) + {} + CD3DX12_ROOT_SIGNATURE_DESC( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init(numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + CD3DX12_ROOT_SIGNATURE_DESC(CD3DX12_DEFAULT) noexcept + { + Init(0, nullptr, 0, nullptr, D3D12_ROOT_SIGNATURE_FLAG_NONE); + } + + inline void Init( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init(*this, numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + + static inline void Init( + _Out_ D3D12_ROOT_SIGNATURE_DESC &desc, + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + desc.NumParameters = numParameters; + desc.pParameters = _pParameters; + desc.NumStaticSamplers = numStaticSamplers; + desc.pStaticSamplers = _pStaticSamplers; + desc.Flags = flags; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_DESCRIPTOR_RANGE1 : public D3D12_DESCRIPTOR_RANGE1 +{ + CD3DX12_DESCRIPTOR_RANGE1() = default; + explicit CD3DX12_DESCRIPTOR_RANGE1(const D3D12_DESCRIPTOR_RANGE1 &o) noexcept : + D3D12_DESCRIPTOR_RANGE1(o) + {} + CD3DX12_DESCRIPTOR_RANGE1( + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + D3D12_DESCRIPTOR_RANGE_FLAGS flags = D3D12_DESCRIPTOR_RANGE_FLAG_NONE, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + Init(rangeType, numDescriptors, baseShaderRegister, registerSpace, flags, offsetInDescriptorsFromTableStart); + } + + inline void Init( + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + D3D12_DESCRIPTOR_RANGE_FLAGS flags = D3D12_DESCRIPTOR_RANGE_FLAG_NONE, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + Init(*this, rangeType, numDescriptors, baseShaderRegister, registerSpace, flags, offsetInDescriptorsFromTableStart); + } + + static inline void Init( + _Out_ D3D12_DESCRIPTOR_RANGE1 &range, + D3D12_DESCRIPTOR_RANGE_TYPE rangeType, + UINT numDescriptors, + UINT baseShaderRegister, + UINT registerSpace = 0, + D3D12_DESCRIPTOR_RANGE_FLAGS flags = D3D12_DESCRIPTOR_RANGE_FLAG_NONE, + UINT offsetInDescriptorsFromTableStart = + D3D12_DESCRIPTOR_RANGE_OFFSET_APPEND) noexcept + { + range.RangeType = rangeType; + range.NumDescriptors = numDescriptors; + range.BaseShaderRegister = baseShaderRegister; + range.RegisterSpace = registerSpace; + range.Flags = flags; + range.OffsetInDescriptorsFromTableStart = offsetInDescriptorsFromTableStart; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_DESCRIPTOR_TABLE1 : public D3D12_ROOT_DESCRIPTOR_TABLE1 +{ + CD3DX12_ROOT_DESCRIPTOR_TABLE1() = default; + explicit CD3DX12_ROOT_DESCRIPTOR_TABLE1(const D3D12_ROOT_DESCRIPTOR_TABLE1 &o) noexcept : + D3D12_ROOT_DESCRIPTOR_TABLE1(o) + {} + CD3DX12_ROOT_DESCRIPTOR_TABLE1( + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE1* _pDescriptorRanges) noexcept + { + Init(numDescriptorRanges, _pDescriptorRanges); + } + + inline void Init( + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE1* _pDescriptorRanges) noexcept + { + Init(*this, numDescriptorRanges, _pDescriptorRanges); + } + + static inline void Init( + _Out_ D3D12_ROOT_DESCRIPTOR_TABLE1 &rootDescriptorTable, + UINT numDescriptorRanges, + _In_reads_opt_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE1* _pDescriptorRanges) noexcept + { + rootDescriptorTable.NumDescriptorRanges = numDescriptorRanges; + rootDescriptorTable.pDescriptorRanges = _pDescriptorRanges; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_DESCRIPTOR1 : public D3D12_ROOT_DESCRIPTOR1 +{ + CD3DX12_ROOT_DESCRIPTOR1() = default; + explicit CD3DX12_ROOT_DESCRIPTOR1(const D3D12_ROOT_DESCRIPTOR1 &o) noexcept : + D3D12_ROOT_DESCRIPTOR1(o) + {} + CD3DX12_ROOT_DESCRIPTOR1( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE) noexcept + { + Init(shaderRegister, registerSpace, flags); + } + + inline void Init( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE) noexcept + { + Init(*this, shaderRegister, registerSpace, flags); + } + + static inline void Init( + _Out_ D3D12_ROOT_DESCRIPTOR1 &table, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE) noexcept + { + table.ShaderRegister = shaderRegister; + table.RegisterSpace = registerSpace; + table.Flags = flags; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_ROOT_PARAMETER1 : public D3D12_ROOT_PARAMETER1 +{ + CD3DX12_ROOT_PARAMETER1() = default; + explicit CD3DX12_ROOT_PARAMETER1(const D3D12_ROOT_PARAMETER1 &o) noexcept : + D3D12_ROOT_PARAMETER1(o) + {} + + static inline void InitAsDescriptorTable( + _Out_ D3D12_ROOT_PARAMETER1 &rootParam, + UINT numDescriptorRanges, + _In_reads_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE1* pDescriptorRanges, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR_TABLE1::Init(rootParam.DescriptorTable, numDescriptorRanges, pDescriptorRanges); + } + + static inline void InitAsConstants( + _Out_ D3D12_ROOT_PARAMETER1 &rootParam, + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_32BIT_CONSTANTS; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_CONSTANTS::Init(rootParam.Constants, num32BitValues, shaderRegister, registerSpace); + } + + static inline void InitAsConstantBufferView( + _Out_ D3D12_ROOT_PARAMETER1 &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_CBV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR1::Init(rootParam.Descriptor, shaderRegister, registerSpace, flags); + } + + static inline void InitAsShaderResourceView( + _Out_ D3D12_ROOT_PARAMETER1 &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_SRV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR1::Init(rootParam.Descriptor, shaderRegister, registerSpace, flags); + } + + static inline void InitAsUnorderedAccessView( + _Out_ D3D12_ROOT_PARAMETER1 &rootParam, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + rootParam.ParameterType = D3D12_ROOT_PARAMETER_TYPE_UAV; + rootParam.ShaderVisibility = visibility; + CD3DX12_ROOT_DESCRIPTOR1::Init(rootParam.Descriptor, shaderRegister, registerSpace, flags); + } + + inline void InitAsDescriptorTable( + UINT numDescriptorRanges, + _In_reads_(numDescriptorRanges) const D3D12_DESCRIPTOR_RANGE1* pDescriptorRanges, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsDescriptorTable(*this, numDescriptorRanges, pDescriptorRanges, visibility); + } + + inline void InitAsConstants( + UINT num32BitValues, + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsConstants(*this, num32BitValues, shaderRegister, registerSpace, visibility); + } + + inline void InitAsConstantBufferView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsConstantBufferView(*this, shaderRegister, registerSpace, flags, visibility); + } + + inline void InitAsShaderResourceView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsShaderResourceView(*this, shaderRegister, registerSpace, flags, visibility); + } + + inline void InitAsUnorderedAccessView( + UINT shaderRegister, + UINT registerSpace = 0, + D3D12_ROOT_DESCRIPTOR_FLAGS flags = D3D12_ROOT_DESCRIPTOR_FLAG_NONE, + D3D12_SHADER_VISIBILITY visibility = D3D12_SHADER_VISIBILITY_ALL) noexcept + { + InitAsUnorderedAccessView(*this, shaderRegister, registerSpace, flags, visibility); + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC : public D3D12_VERSIONED_ROOT_SIGNATURE_DESC +{ + CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC() = default; + explicit CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC(const D3D12_VERSIONED_ROOT_SIGNATURE_DESC &o) noexcept : + D3D12_VERSIONED_ROOT_SIGNATURE_DESC(o) + {} + explicit CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC(const D3D12_ROOT_SIGNATURE_DESC &o) noexcept + { + Version = D3D_ROOT_SIGNATURE_VERSION_1_0; + Desc_1_0 = o; + } + explicit CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC(const D3D12_ROOT_SIGNATURE_DESC1 &o) noexcept + { + Version = D3D_ROOT_SIGNATURE_VERSION_1_1; + Desc_1_1 = o; + } + CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init_1_0(numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER1* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init_1_1(numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + CD3DX12_VERSIONED_ROOT_SIGNATURE_DESC(CD3DX12_DEFAULT) noexcept + { + Init_1_1(0, nullptr, 0, nullptr, D3D12_ROOT_SIGNATURE_FLAG_NONE); + } + + inline void Init_1_0( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init_1_0(*this, numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + + static inline void Init_1_0( + _Out_ D3D12_VERSIONED_ROOT_SIGNATURE_DESC &desc, + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + desc.Version = D3D_ROOT_SIGNATURE_VERSION_1_0; + desc.Desc_1_0.NumParameters = numParameters; + desc.Desc_1_0.pParameters = _pParameters; + desc.Desc_1_0.NumStaticSamplers = numStaticSamplers; + desc.Desc_1_0.pStaticSamplers = _pStaticSamplers; + desc.Desc_1_0.Flags = flags; + } + + inline void Init_1_1( + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER1* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + Init_1_1(*this, numParameters, _pParameters, numStaticSamplers, _pStaticSamplers, flags); + } + + static inline void Init_1_1( + _Out_ D3D12_VERSIONED_ROOT_SIGNATURE_DESC &desc, + UINT numParameters, + _In_reads_opt_(numParameters) const D3D12_ROOT_PARAMETER1* _pParameters, + UINT numStaticSamplers = 0, + _In_reads_opt_(numStaticSamplers) const D3D12_STATIC_SAMPLER_DESC* _pStaticSamplers = nullptr, + D3D12_ROOT_SIGNATURE_FLAGS flags = D3D12_ROOT_SIGNATURE_FLAG_NONE) noexcept + { + desc.Version = D3D_ROOT_SIGNATURE_VERSION_1_1; + desc.Desc_1_1.NumParameters = numParameters; + desc.Desc_1_1.pParameters = _pParameters; + desc.Desc_1_1.NumStaticSamplers = numStaticSamplers; + desc.Desc_1_1.pStaticSamplers = _pStaticSamplers; + desc.Desc_1_1.Flags = flags; + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_CPU_DESCRIPTOR_HANDLE : public D3D12_CPU_DESCRIPTOR_HANDLE +{ + CD3DX12_CPU_DESCRIPTOR_HANDLE() = default; + explicit CD3DX12_CPU_DESCRIPTOR_HANDLE(const D3D12_CPU_DESCRIPTOR_HANDLE &o) noexcept : + D3D12_CPU_DESCRIPTOR_HANDLE(o) + {} + CD3DX12_CPU_DESCRIPTOR_HANDLE(CD3DX12_DEFAULT) noexcept { ptr = 0; } + CD3DX12_CPU_DESCRIPTOR_HANDLE(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE &other, INT offsetScaledByIncrementSize) noexcept + { + InitOffsetted(other, offsetScaledByIncrementSize); + } + CD3DX12_CPU_DESCRIPTOR_HANDLE(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE &other, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + InitOffsetted(other, offsetInDescriptors, descriptorIncrementSize); + } + CD3DX12_CPU_DESCRIPTOR_HANDLE& Offset(INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + ptr = SIZE_T(INT64(ptr) + INT64(offsetInDescriptors) * INT64(descriptorIncrementSize)); + return *this; + } + CD3DX12_CPU_DESCRIPTOR_HANDLE& Offset(INT offsetScaledByIncrementSize) noexcept + { + ptr = SIZE_T(INT64(ptr) + INT64(offsetScaledByIncrementSize)); + return *this; + } + bool operator==(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE& other) const noexcept + { + return (ptr == other.ptr); + } + bool operator!=(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE& other) const noexcept + { + return (ptr != other.ptr); + } + CD3DX12_CPU_DESCRIPTOR_HANDLE &operator=(const D3D12_CPU_DESCRIPTOR_HANDLE &other) noexcept + { + ptr = other.ptr; + return *this; + } + + inline void InitOffsetted(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE &base, INT offsetScaledByIncrementSize) noexcept + { + InitOffsetted(*this, base, offsetScaledByIncrementSize); + } + + inline void InitOffsetted(_In_ const D3D12_CPU_DESCRIPTOR_HANDLE &base, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + InitOffsetted(*this, base, offsetInDescriptors, descriptorIncrementSize); + } + + static inline void InitOffsetted(_Out_ D3D12_CPU_DESCRIPTOR_HANDLE &handle, _In_ const D3D12_CPU_DESCRIPTOR_HANDLE &base, INT offsetScaledByIncrementSize) noexcept + { + handle.ptr = SIZE_T(INT64(base.ptr) + INT64(offsetScaledByIncrementSize)); + } + + static inline void InitOffsetted(_Out_ D3D12_CPU_DESCRIPTOR_HANDLE &handle, _In_ const D3D12_CPU_DESCRIPTOR_HANDLE &base, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + handle.ptr = SIZE_T(INT64(base.ptr) + INT64(offsetInDescriptors) * INT64(descriptorIncrementSize)); + } +}; + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_GPU_DESCRIPTOR_HANDLE : public D3D12_GPU_DESCRIPTOR_HANDLE +{ + CD3DX12_GPU_DESCRIPTOR_HANDLE() = default; + explicit CD3DX12_GPU_DESCRIPTOR_HANDLE(const D3D12_GPU_DESCRIPTOR_HANDLE &o) noexcept : + D3D12_GPU_DESCRIPTOR_HANDLE(o) + {} + CD3DX12_GPU_DESCRIPTOR_HANDLE(CD3DX12_DEFAULT) noexcept { ptr = 0; } + CD3DX12_GPU_DESCRIPTOR_HANDLE(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE &other, INT offsetScaledByIncrementSize) noexcept + { + InitOffsetted(other, offsetScaledByIncrementSize); + } + CD3DX12_GPU_DESCRIPTOR_HANDLE(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE &other, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + InitOffsetted(other, offsetInDescriptors, descriptorIncrementSize); + } + CD3DX12_GPU_DESCRIPTOR_HANDLE& Offset(INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + ptr = UINT64(INT64(ptr) + INT64(offsetInDescriptors) * INT64(descriptorIncrementSize)); + return *this; + } + CD3DX12_GPU_DESCRIPTOR_HANDLE& Offset(INT offsetScaledByIncrementSize) noexcept + { + ptr = UINT64(INT64(ptr) + INT64(offsetScaledByIncrementSize)); + return *this; + } + inline bool operator==(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE& other) const noexcept + { + return (ptr == other.ptr); + } + inline bool operator!=(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE& other) const noexcept + { + return (ptr != other.ptr); + } + CD3DX12_GPU_DESCRIPTOR_HANDLE &operator=(const D3D12_GPU_DESCRIPTOR_HANDLE &other) noexcept + { + ptr = other.ptr; + return *this; + } + + inline void InitOffsetted(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE &base, INT offsetScaledByIncrementSize) noexcept + { + InitOffsetted(*this, base, offsetScaledByIncrementSize); + } + + inline void InitOffsetted(_In_ const D3D12_GPU_DESCRIPTOR_HANDLE &base, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + InitOffsetted(*this, base, offsetInDescriptors, descriptorIncrementSize); + } + + static inline void InitOffsetted(_Out_ D3D12_GPU_DESCRIPTOR_HANDLE &handle, _In_ const D3D12_GPU_DESCRIPTOR_HANDLE &base, INT offsetScaledByIncrementSize) noexcept + { + handle.ptr = UINT64(INT64(base.ptr) + INT64(offsetScaledByIncrementSize)); + } + + static inline void InitOffsetted(_Out_ D3D12_GPU_DESCRIPTOR_HANDLE &handle, _In_ const D3D12_GPU_DESCRIPTOR_HANDLE &base, INT offsetInDescriptors, UINT descriptorIncrementSize) noexcept + { + handle.ptr = UINT64(INT64(base.ptr) + INT64(offsetInDescriptors) * INT64(descriptorIncrementSize)); + } +}; + +//------------------------------------------------------------------------------------------------ +inline constexpr UINT D3D12CalcSubresource( UINT MipSlice, UINT ArraySlice, UINT PlaneSlice, UINT MipLevels, UINT ArraySize ) noexcept +{ + return MipSlice + ArraySlice * MipLevels + PlaneSlice * MipLevels * ArraySize; +} + +//------------------------------------------------------------------------------------------------ +template +inline void D3D12DecomposeSubresource( UINT Subresource, UINT MipLevels, UINT ArraySize, _Out_ T& MipSlice, _Out_ U& ArraySlice, _Out_ V& PlaneSlice ) noexcept +{ + MipSlice = static_cast(Subresource % MipLevels); + ArraySlice = static_cast((Subresource / MipLevels) % ArraySize); + PlaneSlice = static_cast(Subresource / (MipLevels * ArraySize)); +} + +//------------------------------------------------------------------------------------------------ +inline UINT8 D3D12GetFormatPlaneCount( + _In_ ID3D12Device* pDevice, + DXGI_FORMAT Format + ) noexcept +{ + D3D12_FEATURE_DATA_FORMAT_INFO formatInfo = { Format, 0 }; + if (FAILED(pDevice->CheckFeatureSupport(D3D12_FEATURE_FORMAT_INFO, &formatInfo, sizeof(formatInfo)))) + { + return 0; + } + return formatInfo.PlaneCount; +} + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RESOURCE_DESC : public D3D12_RESOURCE_DESC +{ + CD3DX12_RESOURCE_DESC() = default; + explicit CD3DX12_RESOURCE_DESC( const D3D12_RESOURCE_DESC& o ) noexcept : + D3D12_RESOURCE_DESC( o ) + {} + CD3DX12_RESOURCE_DESC( + D3D12_RESOURCE_DIMENSION dimension, + UINT64 alignment, + UINT64 width, + UINT height, + UINT16 depthOrArraySize, + UINT16 mipLevels, + DXGI_FORMAT format, + UINT sampleCount, + UINT sampleQuality, + D3D12_TEXTURE_LAYOUT layout, + D3D12_RESOURCE_FLAGS flags ) noexcept + { + Dimension = dimension; + Alignment = alignment; + Width = width; + Height = height; + DepthOrArraySize = depthOrArraySize; + MipLevels = mipLevels; + Format = format; + SampleDesc.Count = sampleCount; + SampleDesc.Quality = sampleQuality; + Layout = layout; + Flags = flags; + } + static inline CD3DX12_RESOURCE_DESC Buffer( + const D3D12_RESOURCE_ALLOCATION_INFO& resAllocInfo, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE ) noexcept + { + return CD3DX12_RESOURCE_DESC( D3D12_RESOURCE_DIMENSION_BUFFER, resAllocInfo.Alignment, resAllocInfo.SizeInBytes, + 1, 1, 1, DXGI_FORMAT_UNKNOWN, 1, 0, D3D12_TEXTURE_LAYOUT_ROW_MAJOR, flags ); + } + static inline CD3DX12_RESOURCE_DESC Buffer( + UINT64 width, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC( D3D12_RESOURCE_DIMENSION_BUFFER, alignment, width, 1, 1, 1, + DXGI_FORMAT_UNKNOWN, 1, 0, D3D12_TEXTURE_LAYOUT_ROW_MAJOR, flags ); + } + static inline CD3DX12_RESOURCE_DESC Tex1D( + DXGI_FORMAT format, + UINT64 width, + UINT16 arraySize = 1, + UINT16 mipLevels = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC( D3D12_RESOURCE_DIMENSION_TEXTURE1D, alignment, width, 1, arraySize, + mipLevels, format, 1, 0, layout, flags ); + } + static inline CD3DX12_RESOURCE_DESC Tex2D( + DXGI_FORMAT format, + UINT64 width, + UINT height, + UINT16 arraySize = 1, + UINT16 mipLevels = 0, + UINT sampleCount = 1, + UINT sampleQuality = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC( D3D12_RESOURCE_DIMENSION_TEXTURE2D, alignment, width, height, arraySize, + mipLevels, format, sampleCount, sampleQuality, layout, flags ); + } + static inline CD3DX12_RESOURCE_DESC Tex3D( + DXGI_FORMAT format, + UINT64 width, + UINT height, + UINT16 depth, + UINT16 mipLevels = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC( D3D12_RESOURCE_DIMENSION_TEXTURE3D, alignment, width, height, depth, + mipLevels, format, 1, 0, layout, flags ); + } + inline UINT16 Depth() const noexcept + { return (Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE3D ? DepthOrArraySize : 1); } + inline UINT16 ArraySize() const noexcept + { return (Dimension != D3D12_RESOURCE_DIMENSION_TEXTURE3D ? DepthOrArraySize : 1); } + inline UINT8 PlaneCount(_In_ ID3D12Device* pDevice) const noexcept + { return D3D12GetFormatPlaneCount(pDevice, Format); } + inline UINT Subresources(_In_ ID3D12Device* pDevice) const noexcept + { return MipLevels * ArraySize() * PlaneCount(pDevice); } + inline UINT CalcSubresource(UINT MipSlice, UINT ArraySlice, UINT PlaneSlice) noexcept + { return D3D12CalcSubresource(MipSlice, ArraySlice, PlaneSlice, MipLevels, ArraySize()); } +}; +inline bool operator==( const D3D12_RESOURCE_DESC& l, const D3D12_RESOURCE_DESC& r ) noexcept +{ + return l.Dimension == r.Dimension && + l.Alignment == r.Alignment && + l.Width == r.Width && + l.Height == r.Height && + l.DepthOrArraySize == r.DepthOrArraySize && + l.MipLevels == r.MipLevels && + l.Format == r.Format && + l.SampleDesc.Count == r.SampleDesc.Count && + l.SampleDesc.Quality == r.SampleDesc.Quality && + l.Layout == r.Layout && + l.Flags == r.Flags; +} +inline bool operator!=( const D3D12_RESOURCE_DESC& l, const D3D12_RESOURCE_DESC& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RESOURCE_DESC1 : public D3D12_RESOURCE_DESC1 +{ + CD3DX12_RESOURCE_DESC1() = default; + explicit CD3DX12_RESOURCE_DESC1( const D3D12_RESOURCE_DESC1& o ) noexcept : + D3D12_RESOURCE_DESC1( o ) + {} + CD3DX12_RESOURCE_DESC1( + D3D12_RESOURCE_DIMENSION dimension, + UINT64 alignment, + UINT64 width, + UINT height, + UINT16 depthOrArraySize, + UINT16 mipLevels, + DXGI_FORMAT format, + UINT sampleCount, + UINT sampleQuality, + D3D12_TEXTURE_LAYOUT layout, + D3D12_RESOURCE_FLAGS flags, + UINT samplerFeedbackMipRegionWidth = 0, + UINT samplerFeedbackMipRegionHeight = 0, + UINT samplerFeedbackMipRegionDepth = 0) noexcept + { + Dimension = dimension; + Alignment = alignment; + Width = width; + Height = height; + DepthOrArraySize = depthOrArraySize; + MipLevels = mipLevels; + Format = format; + SampleDesc.Count = sampleCount; + SampleDesc.Quality = sampleQuality; + Layout = layout; + Flags = flags; + SamplerFeedbackMipRegion.Width = samplerFeedbackMipRegionWidth; + SamplerFeedbackMipRegion.Height = samplerFeedbackMipRegionHeight; + SamplerFeedbackMipRegion.Depth = samplerFeedbackMipRegionDepth; + } + static inline CD3DX12_RESOURCE_DESC1 Buffer( + const D3D12_RESOURCE_ALLOCATION_INFO& resAllocInfo, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE ) noexcept + { + return CD3DX12_RESOURCE_DESC1( D3D12_RESOURCE_DIMENSION_BUFFER, resAllocInfo.Alignment, resAllocInfo.SizeInBytes, + 1, 1, 1, DXGI_FORMAT_UNKNOWN, 1, 0, D3D12_TEXTURE_LAYOUT_ROW_MAJOR, flags, 0, 0, 0 ); + } + static inline CD3DX12_RESOURCE_DESC1 Buffer( + UINT64 width, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC1( D3D12_RESOURCE_DIMENSION_BUFFER, alignment, width, 1, 1, 1, + DXGI_FORMAT_UNKNOWN, 1, 0, D3D12_TEXTURE_LAYOUT_ROW_MAJOR, flags, 0, 0, 0 ); + } + static inline CD3DX12_RESOURCE_DESC1 Tex1D( + DXGI_FORMAT format, + UINT64 width, + UINT16 arraySize = 1, + UINT16 mipLevels = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC1( D3D12_RESOURCE_DIMENSION_TEXTURE1D, alignment, width, 1, arraySize, + mipLevels, format, 1, 0, layout, flags, 0, 0, 0 ); + } + static inline CD3DX12_RESOURCE_DESC1 Tex2D( + DXGI_FORMAT format, + UINT64 width, + UINT height, + UINT16 arraySize = 1, + UINT16 mipLevels = 0, + UINT sampleCount = 1, + UINT sampleQuality = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0, + UINT samplerFeedbackMipRegionWidth = 0, + UINT samplerFeedbackMipRegionHeight = 0, + UINT samplerFeedbackMipRegionDepth = 0) noexcept + { + return CD3DX12_RESOURCE_DESC1( D3D12_RESOURCE_DIMENSION_TEXTURE2D, alignment, width, height, arraySize, + mipLevels, format, sampleCount, sampleQuality, layout, flags, samplerFeedbackMipRegionWidth, + samplerFeedbackMipRegionHeight, samplerFeedbackMipRegionDepth ); + } + static inline CD3DX12_RESOURCE_DESC1 Tex3D( + DXGI_FORMAT format, + UINT64 width, + UINT height, + UINT16 depth, + UINT16 mipLevels = 0, + D3D12_RESOURCE_FLAGS flags = D3D12_RESOURCE_FLAG_NONE, + D3D12_TEXTURE_LAYOUT layout = D3D12_TEXTURE_LAYOUT_UNKNOWN, + UINT64 alignment = 0 ) noexcept + { + return CD3DX12_RESOURCE_DESC1( D3D12_RESOURCE_DIMENSION_TEXTURE3D, alignment, width, height, depth, + mipLevels, format, 1, 0, layout, flags, 0, 0, 0 ); + } + inline UINT16 Depth() const noexcept + { return (Dimension == D3D12_RESOURCE_DIMENSION_TEXTURE3D ? DepthOrArraySize : 1); } + inline UINT16 ArraySize() const noexcept + { return (Dimension != D3D12_RESOURCE_DIMENSION_TEXTURE3D ? DepthOrArraySize : 1); } + inline UINT8 PlaneCount(_In_ ID3D12Device* pDevice) const noexcept + { return D3D12GetFormatPlaneCount(pDevice, Format); } + inline UINT Subresources(_In_ ID3D12Device* pDevice) const noexcept + { return MipLevels * ArraySize() * PlaneCount(pDevice); } + inline UINT CalcSubresource(UINT MipSlice, UINT ArraySlice, UINT PlaneSlice) noexcept + { return D3D12CalcSubresource(MipSlice, ArraySlice, PlaneSlice, MipLevels, ArraySize()); } +}; +inline bool operator==( const D3D12_RESOURCE_DESC1& l, const D3D12_RESOURCE_DESC1& r ) noexcept +{ + return l.Dimension == r.Dimension && + l.Alignment == r.Alignment && + l.Width == r.Width && + l.Height == r.Height && + l.DepthOrArraySize == r.DepthOrArraySize && + l.MipLevels == r.MipLevels && + l.Format == r.Format && + l.SampleDesc.Count == r.SampleDesc.Count && + l.SampleDesc.Quality == r.SampleDesc.Quality && + l.Layout == r.Layout && + l.Flags == r.Flags && + l.SamplerFeedbackMipRegion.Width == r.SamplerFeedbackMipRegion.Width && + l.SamplerFeedbackMipRegion.Height == r.SamplerFeedbackMipRegion.Height && + l.SamplerFeedbackMipRegion.Depth == r.SamplerFeedbackMipRegion.Depth; +} +inline bool operator!=( const D3D12_RESOURCE_DESC1& l, const D3D12_RESOURCE_DESC1& r ) noexcept +{ return !( l == r ); } + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_VIEW_INSTANCING_DESC : public D3D12_VIEW_INSTANCING_DESC +{ + CD3DX12_VIEW_INSTANCING_DESC() = default; + explicit CD3DX12_VIEW_INSTANCING_DESC( const D3D12_VIEW_INSTANCING_DESC& o ) noexcept : + D3D12_VIEW_INSTANCING_DESC( o ) + {} + explicit CD3DX12_VIEW_INSTANCING_DESC( CD3DX12_DEFAULT ) noexcept + { + ViewInstanceCount = 0; + pViewInstanceLocations = nullptr; + Flags = D3D12_VIEW_INSTANCING_FLAG_NONE; + } + explicit CD3DX12_VIEW_INSTANCING_DESC( + UINT InViewInstanceCount, + const D3D12_VIEW_INSTANCE_LOCATION* InViewInstanceLocations, + D3D12_VIEW_INSTANCING_FLAGS InFlags) noexcept + { + ViewInstanceCount = InViewInstanceCount; + pViewInstanceLocations = InViewInstanceLocations; + Flags = InFlags; + } +}; + +//------------------------------------------------------------------------------------------------ +// Row-by-row memcpy +inline void MemcpySubresource( + _In_ const D3D12_MEMCPY_DEST* pDest, + _In_ const D3D12_SUBRESOURCE_DATA* pSrc, + SIZE_T RowSizeInBytes, + UINT NumRows, + UINT NumSlices) noexcept +{ + for (UINT z = 0; z < NumSlices; ++z) + { + auto pDestSlice = static_cast(pDest->pData) + pDest->SlicePitch * z; + auto pSrcSlice = static_cast(pSrc->pData) + pSrc->SlicePitch * LONG_PTR(z); + for (UINT y = 0; y < NumRows; ++y) + { + memcpy(pDestSlice + pDest->RowPitch * y, + pSrcSlice + pSrc->RowPitch * LONG_PTR(y), + RowSizeInBytes); + } + } +} + +//------------------------------------------------------------------------------------------------ +// Row-by-row memcpy +inline void MemcpySubresource( + _In_ const D3D12_MEMCPY_DEST* pDest, + _In_ const void* pResourceData, + _In_ const D3D12_SUBRESOURCE_INFO* pSrc, + SIZE_T RowSizeInBytes, + UINT NumRows, + UINT NumSlices) noexcept +{ + for (UINT z = 0; z < NumSlices; ++z) + { + auto pDestSlice = static_cast(pDest->pData) + pDest->SlicePitch * z; + auto pSrcSlice = (static_cast(pResourceData) + pSrc->Offset) + pSrc->DepthPitch * ULONG_PTR(z); + for (UINT y = 0; y < NumRows; ++y) + { + memcpy(pDestSlice + pDest->RowPitch * y, + pSrcSlice + pSrc->RowPitch * ULONG_PTR(y), + RowSizeInBytes); + } + } +} + +//------------------------------------------------------------------------------------------------ +// Returns required size of a buffer to be used for data upload +inline UINT64 GetRequiredIntermediateSize( + _In_ ID3D12Resource* pDestinationResource, + _In_range_(0,D3D12_REQ_SUBRESOURCES) UINT FirstSubresource, + _In_range_(0,D3D12_REQ_SUBRESOURCES-FirstSubresource) UINT NumSubresources) noexcept +{ + auto Desc = pDestinationResource->GetDesc(); + UINT64 RequiredSize = 0; + + ID3D12Device* pDevice = nullptr; + pDestinationResource->GetDevice(IID_ID3D12Device, reinterpret_cast(&pDevice)); + pDevice->GetCopyableFootprints(&Desc, FirstSubresource, NumSubresources, 0, nullptr, nullptr, nullptr, &RequiredSize); + pDevice->Release(); + + return RequiredSize; +} + +//------------------------------------------------------------------------------------------------ +// All arrays must be populated (e.g. by calling GetCopyableFootprints) +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + _In_range_(0,D3D12_REQ_SUBRESOURCES) UINT FirstSubresource, + _In_range_(0,D3D12_REQ_SUBRESOURCES-FirstSubresource) UINT NumSubresources, + UINT64 RequiredSize, + _In_reads_(NumSubresources) const D3D12_PLACED_SUBRESOURCE_FOOTPRINT* pLayouts, + _In_reads_(NumSubresources) const UINT* pNumRows, + _In_reads_(NumSubresources) const UINT64* pRowSizesInBytes, + _In_reads_(NumSubresources) const D3D12_SUBRESOURCE_DATA* pSrcData) noexcept +{ + // Minor validation + auto IntermediateDesc = pIntermediate->GetDesc(); + auto DestinationDesc = pDestinationResource->GetDesc(); + if (IntermediateDesc.Dimension != D3D12_RESOURCE_DIMENSION_BUFFER || + IntermediateDesc.Width < RequiredSize + pLayouts[0].Offset || + RequiredSize > SIZE_T(-1) || + (DestinationDesc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER && + (FirstSubresource != 0 || NumSubresources != 1))) + { + return 0; + } + + BYTE* pData; + HRESULT hr = pIntermediate->Map(0, nullptr, reinterpret_cast(&pData)); + if (FAILED(hr)) + { + return 0; + } + + for (UINT i = 0; i < NumSubresources; ++i) + { + if (pRowSizesInBytes[i] > SIZE_T(-1)) return 0; + D3D12_MEMCPY_DEST DestData = { pData + pLayouts[i].Offset, pLayouts[i].Footprint.RowPitch, SIZE_T(pLayouts[i].Footprint.RowPitch) * SIZE_T(pNumRows[i]) }; + MemcpySubresource(&DestData, &pSrcData[i], static_cast(pRowSizesInBytes[i]), pNumRows[i], pLayouts[i].Footprint.Depth); + } + pIntermediate->Unmap(0, nullptr); + + if (DestinationDesc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER) + { + pCmdList->CopyBufferRegion( + pDestinationResource, 0, pIntermediate, pLayouts[0].Offset, pLayouts[0].Footprint.Width); + } + else + { + for (UINT i = 0; i < NumSubresources; ++i) + { + CD3DX12_TEXTURE_COPY_LOCATION Dst(pDestinationResource, i + FirstSubresource); + CD3DX12_TEXTURE_COPY_LOCATION Src(pIntermediate, pLayouts[i]); + pCmdList->CopyTextureRegion(&Dst, 0, 0, 0, &Src, nullptr); + } + } + return RequiredSize; +} + +//------------------------------------------------------------------------------------------------ +// All arrays must be populated (e.g. by calling GetCopyableFootprints) +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + _In_range_(0,D3D12_REQ_SUBRESOURCES) UINT FirstSubresource, + _In_range_(0,D3D12_REQ_SUBRESOURCES-FirstSubresource) UINT NumSubresources, + UINT64 RequiredSize, + _In_reads_(NumSubresources) const D3D12_PLACED_SUBRESOURCE_FOOTPRINT* pLayouts, + _In_reads_(NumSubresources) const UINT* pNumRows, + _In_reads_(NumSubresources) const UINT64* pRowSizesInBytes, + _In_ const void* pResourceData, + _In_reads_(NumSubresources) const D3D12_SUBRESOURCE_INFO* pSrcData) noexcept +{ + // Minor validation + auto IntermediateDesc = pIntermediate->GetDesc(); + auto DestinationDesc = pDestinationResource->GetDesc(); + if (IntermediateDesc.Dimension != D3D12_RESOURCE_DIMENSION_BUFFER || + IntermediateDesc.Width < RequiredSize + pLayouts[0].Offset || + RequiredSize > SIZE_T(-1) || + (DestinationDesc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER && + (FirstSubresource != 0 || NumSubresources != 1))) + { + return 0; + } + + BYTE* pData; + HRESULT hr = pIntermediate->Map(0, nullptr, reinterpret_cast(&pData)); + if (FAILED(hr)) + { + return 0; + } + + for (UINT i = 0; i < NumSubresources; ++i) + { + if (pRowSizesInBytes[i] > SIZE_T(-1)) return 0; + D3D12_MEMCPY_DEST DestData = { pData + pLayouts[i].Offset, pLayouts[i].Footprint.RowPitch, SIZE_T(pLayouts[i].Footprint.RowPitch) * SIZE_T(pNumRows[i]) }; + MemcpySubresource(&DestData, pResourceData, &pSrcData[i], static_cast(pRowSizesInBytes[i]), pNumRows[i], pLayouts[i].Footprint.Depth); + } + pIntermediate->Unmap(0, nullptr); + + if (DestinationDesc.Dimension == D3D12_RESOURCE_DIMENSION_BUFFER) + { + pCmdList->CopyBufferRegion( + pDestinationResource, 0, pIntermediate, pLayouts[0].Offset, pLayouts[0].Footprint.Width); + } + else + { + for (UINT i = 0; i < NumSubresources; ++i) + { + CD3DX12_TEXTURE_COPY_LOCATION Dst(pDestinationResource, i + FirstSubresource); + CD3DX12_TEXTURE_COPY_LOCATION Src(pIntermediate, pLayouts[i]); + pCmdList->CopyTextureRegion(&Dst, 0, 0, 0, &Src, nullptr); + } + } + return RequiredSize; +} + +//------------------------------------------------------------------------------------------------ +// Heap-allocating UpdateSubresources implementation +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + UINT64 IntermediateOffset, + _In_range_(0,D3D12_REQ_SUBRESOURCES) UINT FirstSubresource, + _In_range_(0,D3D12_REQ_SUBRESOURCES-FirstSubresource) UINT NumSubresources, + _In_reads_(NumSubresources) const D3D12_SUBRESOURCE_DATA* pSrcData) noexcept +{ + UINT64 RequiredSize = 0; + auto MemToAlloc = static_cast(sizeof(D3D12_PLACED_SUBRESOURCE_FOOTPRINT) + sizeof(UINT) + sizeof(UINT64)) * NumSubresources; + if (MemToAlloc > SIZE_MAX) + { + return 0; + } + void* pMem = HeapAlloc(GetProcessHeap(), 0, static_cast(MemToAlloc)); + if (pMem == nullptr) + { + return 0; + } + auto pLayouts = static_cast(pMem); + auto pRowSizesInBytes = reinterpret_cast(pLayouts + NumSubresources); + auto pNumRows = reinterpret_cast(pRowSizesInBytes + NumSubresources); + + auto Desc = pDestinationResource->GetDesc(); + ID3D12Device* pDevice = nullptr; + pDestinationResource->GetDevice(IID_ID3D12Device, reinterpret_cast(&pDevice)); + pDevice->GetCopyableFootprints(&Desc, FirstSubresource, NumSubresources, IntermediateOffset, pLayouts, pNumRows, pRowSizesInBytes, &RequiredSize); + pDevice->Release(); + + UINT64 Result = UpdateSubresources(pCmdList, pDestinationResource, pIntermediate, FirstSubresource, NumSubresources, RequiredSize, pLayouts, pNumRows, pRowSizesInBytes, pSrcData); + HeapFree(GetProcessHeap(), 0, pMem); + return Result; +} + +//------------------------------------------------------------------------------------------------ +// Heap-allocating UpdateSubresources implementation +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + UINT64 IntermediateOffset, + _In_range_(0,D3D12_REQ_SUBRESOURCES) UINT FirstSubresource, + _In_range_(0,D3D12_REQ_SUBRESOURCES-FirstSubresource) UINT NumSubresources, + _In_ const void* pResourceData, + _In_reads_(NumSubresources) D3D12_SUBRESOURCE_INFO* pSrcData) noexcept +{ + UINT64 RequiredSize = 0; + auto MemToAlloc = static_cast(sizeof(D3D12_PLACED_SUBRESOURCE_FOOTPRINT) + sizeof(UINT) + sizeof(UINT64)) * NumSubresources; + if (MemToAlloc > SIZE_MAX) + { + return 0; + } + void* pMem = HeapAlloc(GetProcessHeap(), 0, static_cast(MemToAlloc)); + if (pMem == nullptr) + { + return 0; + } + auto pLayouts = reinterpret_cast(pMem); + auto pRowSizesInBytes = reinterpret_cast(pLayouts + NumSubresources); + auto pNumRows = reinterpret_cast(pRowSizesInBytes + NumSubresources); + + auto Desc = pDestinationResource->GetDesc(); + ID3D12Device* pDevice = nullptr; + pDestinationResource->GetDevice(IID_ID3D12Device, reinterpret_cast(&pDevice)); + pDevice->GetCopyableFootprints(&Desc, FirstSubresource, NumSubresources, IntermediateOffset, pLayouts, pNumRows, pRowSizesInBytes, &RequiredSize); + pDevice->Release(); + + UINT64 Result = UpdateSubresources(pCmdList, pDestinationResource, pIntermediate, FirstSubresource, NumSubresources, RequiredSize, pLayouts, pNumRows, pRowSizesInBytes, pResourceData, pSrcData); + HeapFree(GetProcessHeap(), 0, pMem); + return Result; +} + +//------------------------------------------------------------------------------------------------ +// Stack-allocating UpdateSubresources implementation +template +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + UINT64 IntermediateOffset, + _In_range_(0,MaxSubresources) UINT FirstSubresource, + _In_range_(1,MaxSubresources-FirstSubresource) UINT NumSubresources, + _In_reads_(NumSubresources) const D3D12_SUBRESOURCE_DATA* pSrcData) noexcept +{ + UINT64 RequiredSize = 0; + D3D12_PLACED_SUBRESOURCE_FOOTPRINT Layouts[MaxSubresources]; + UINT NumRows[MaxSubresources]; + UINT64 RowSizesInBytes[MaxSubresources]; + + auto Desc = pDestinationResource->GetDesc(); + ID3D12Device* pDevice = nullptr; + pDestinationResource->GetDevice(IID_ID3D12Device, reinterpret_cast(&pDevice)); + pDevice->GetCopyableFootprints(&Desc, FirstSubresource, NumSubresources, IntermediateOffset, Layouts, NumRows, RowSizesInBytes, &RequiredSize); + pDevice->Release(); + + return UpdateSubresources(pCmdList, pDestinationResource, pIntermediate, FirstSubresource, NumSubresources, RequiredSize, Layouts, NumRows, RowSizesInBytes, pSrcData); +} + +//------------------------------------------------------------------------------------------------ +// Stack-allocating UpdateSubresources implementation +template +inline UINT64 UpdateSubresources( + _In_ ID3D12GraphicsCommandList* pCmdList, + _In_ ID3D12Resource* pDestinationResource, + _In_ ID3D12Resource* pIntermediate, + UINT64 IntermediateOffset, + _In_range_(0,MaxSubresources) UINT FirstSubresource, + _In_range_(1,MaxSubresources-FirstSubresource) UINT NumSubresources, + _In_ const void* pResourceData, + _In_reads_(NumSubresources) D3D12_SUBRESOURCE_INFO* pSrcData) noexcept +{ + UINT64 RequiredSize = 0; + D3D12_PLACED_SUBRESOURCE_FOOTPRINT Layouts[MaxSubresources]; + UINT NumRows[MaxSubresources]; + UINT64 RowSizesInBytes[MaxSubresources]; + + auto Desc = pDestinationResource->GetDesc(); + ID3D12Device* pDevice = nullptr; + pDestinationResource->GetDevice(IID_ID3D12Device, reinterpret_cast(&pDevice)); + pDevice->GetCopyableFootprints(&Desc, FirstSubresource, NumSubresources, IntermediateOffset, Layouts, NumRows, RowSizesInBytes, &RequiredSize); + pDevice->Release(); + + return UpdateSubresources(pCmdList, pDestinationResource, pIntermediate, FirstSubresource, NumSubresources, RequiredSize, Layouts, NumRows, RowSizesInBytes, pResourceData, pSrcData); +} + +//------------------------------------------------------------------------------------------------ +inline constexpr bool D3D12IsLayoutOpaque( D3D12_TEXTURE_LAYOUT Layout ) noexcept +{ return Layout == D3D12_TEXTURE_LAYOUT_UNKNOWN || Layout == D3D12_TEXTURE_LAYOUT_64KB_UNDEFINED_SWIZZLE; } + +//------------------------------------------------------------------------------------------------ +template +inline ID3D12CommandList * const * CommandListCast(t_CommandListType * const * pp) noexcept +{ + // This cast is useful for passing strongly typed command list pointers into + // ExecuteCommandLists. + // This cast is valid as long as the const-ness is respected. D3D12 APIs do + // respect the const-ness of their arguments. + return reinterpret_cast(pp); +} + +//------------------------------------------------------------------------------------------------ +// D3D12 exports a new method for serializing root signatures in the Windows 10 Anniversary Update. +// To help enable root signature 1.1 features when they are available and not require maintaining +// two code paths for building root signatures, this helper method reconstructs a 1.0 signature when +// 1.1 is not supported. +inline HRESULT D3DX12SerializeVersionedRootSignature( + _In_ const D3D12_VERSIONED_ROOT_SIGNATURE_DESC* pRootSignatureDesc, + D3D_ROOT_SIGNATURE_VERSION MaxVersion, + _Outptr_ ID3DBlob** ppBlob, + _Always_(_Outptr_opt_result_maybenull_) ID3DBlob** ppErrorBlob) noexcept +{ + if (ppErrorBlob != nullptr) + { + *ppErrorBlob = nullptr; + } + + switch (MaxVersion) + { + case D3D_ROOT_SIGNATURE_VERSION_1_0: + switch (pRootSignatureDesc->Version) + { + case D3D_ROOT_SIGNATURE_VERSION_1_0: + return D3D12SerializeRootSignature(&pRootSignatureDesc->Desc_1_0, D3D_ROOT_SIGNATURE_VERSION_1, ppBlob, ppErrorBlob); + + case D3D_ROOT_SIGNATURE_VERSION_1_1: + { + HRESULT hr = S_OK; + const D3D12_ROOT_SIGNATURE_DESC1& desc_1_1 = pRootSignatureDesc->Desc_1_1; + + const SIZE_T ParametersSize = sizeof(D3D12_ROOT_PARAMETER) * desc_1_1.NumParameters; + void* pParameters = (ParametersSize > 0) ? HeapAlloc(GetProcessHeap(), 0, ParametersSize) : nullptr; + if (ParametersSize > 0 && pParameters == nullptr) + { + hr = E_OUTOFMEMORY; + } + auto pParameters_1_0 = static_cast(pParameters); + + if (SUCCEEDED(hr)) + { + for (UINT n = 0; n < desc_1_1.NumParameters; n++) + { + __analysis_assume(ParametersSize == sizeof(D3D12_ROOT_PARAMETER) * desc_1_1.NumParameters); + pParameters_1_0[n].ParameterType = desc_1_1.pParameters[n].ParameterType; + pParameters_1_0[n].ShaderVisibility = desc_1_1.pParameters[n].ShaderVisibility; + + switch (desc_1_1.pParameters[n].ParameterType) + { + case D3D12_ROOT_PARAMETER_TYPE_32BIT_CONSTANTS: + pParameters_1_0[n].Constants.Num32BitValues = desc_1_1.pParameters[n].Constants.Num32BitValues; + pParameters_1_0[n].Constants.RegisterSpace = desc_1_1.pParameters[n].Constants.RegisterSpace; + pParameters_1_0[n].Constants.ShaderRegister = desc_1_1.pParameters[n].Constants.ShaderRegister; + break; + + case D3D12_ROOT_PARAMETER_TYPE_CBV: + case D3D12_ROOT_PARAMETER_TYPE_SRV: + case D3D12_ROOT_PARAMETER_TYPE_UAV: + pParameters_1_0[n].Descriptor.RegisterSpace = desc_1_1.pParameters[n].Descriptor.RegisterSpace; + pParameters_1_0[n].Descriptor.ShaderRegister = desc_1_1.pParameters[n].Descriptor.ShaderRegister; + break; + + case D3D12_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE: + const D3D12_ROOT_DESCRIPTOR_TABLE1& table_1_1 = desc_1_1.pParameters[n].DescriptorTable; + + const SIZE_T DescriptorRangesSize = sizeof(D3D12_DESCRIPTOR_RANGE) * table_1_1.NumDescriptorRanges; + void* pDescriptorRanges = (DescriptorRangesSize > 0 && SUCCEEDED(hr)) ? HeapAlloc(GetProcessHeap(), 0, DescriptorRangesSize) : nullptr; + if (DescriptorRangesSize > 0 && pDescriptorRanges == nullptr) + { + hr = E_OUTOFMEMORY; + } + auto pDescriptorRanges_1_0 = static_cast(pDescriptorRanges); + + if (SUCCEEDED(hr)) + { + for (UINT x = 0; x < table_1_1.NumDescriptorRanges; x++) + { + __analysis_assume(DescriptorRangesSize == sizeof(D3D12_DESCRIPTOR_RANGE) * table_1_1.NumDescriptorRanges); + pDescriptorRanges_1_0[x].BaseShaderRegister = table_1_1.pDescriptorRanges[x].BaseShaderRegister; + pDescriptorRanges_1_0[x].NumDescriptors = table_1_1.pDescriptorRanges[x].NumDescriptors; + pDescriptorRanges_1_0[x].OffsetInDescriptorsFromTableStart = table_1_1.pDescriptorRanges[x].OffsetInDescriptorsFromTableStart; + pDescriptorRanges_1_0[x].RangeType = table_1_1.pDescriptorRanges[x].RangeType; + pDescriptorRanges_1_0[x].RegisterSpace = table_1_1.pDescriptorRanges[x].RegisterSpace; + } + } + + D3D12_ROOT_DESCRIPTOR_TABLE& table_1_0 = pParameters_1_0[n].DescriptorTable; + table_1_0.NumDescriptorRanges = table_1_1.NumDescriptorRanges; + table_1_0.pDescriptorRanges = pDescriptorRanges_1_0; + } + } + } + + if (SUCCEEDED(hr)) + { + CD3DX12_ROOT_SIGNATURE_DESC desc_1_0(desc_1_1.NumParameters, pParameters_1_0, desc_1_1.NumStaticSamplers, desc_1_1.pStaticSamplers, desc_1_1.Flags); + hr = D3D12SerializeRootSignature(&desc_1_0, D3D_ROOT_SIGNATURE_VERSION_1, ppBlob, ppErrorBlob); + } + + if (pParameters) + { + for (UINT n = 0; n < desc_1_1.NumParameters; n++) + { + if (desc_1_1.pParameters[n].ParameterType == D3D12_ROOT_PARAMETER_TYPE_DESCRIPTOR_TABLE) + { + auto pDescriptorRanges_1_0 = pParameters_1_0[n].DescriptorTable.pDescriptorRanges; + HeapFree(GetProcessHeap(), 0, reinterpret_cast(const_cast(pDescriptorRanges_1_0))); + } + } + HeapFree(GetProcessHeap(), 0, pParameters); + } + return hr; + } + } + break; + + case D3D_ROOT_SIGNATURE_VERSION_1_1: + return D3D12SerializeVersionedRootSignature(pRootSignatureDesc, ppBlob, ppErrorBlob); + } + + return E_INVALIDARG; +} + +//------------------------------------------------------------------------------------------------ +struct CD3DX12_RT_FORMAT_ARRAY : public D3D12_RT_FORMAT_ARRAY +{ + CD3DX12_RT_FORMAT_ARRAY() = default; + explicit CD3DX12_RT_FORMAT_ARRAY(const D3D12_RT_FORMAT_ARRAY& o) noexcept + : D3D12_RT_FORMAT_ARRAY(o) + {} + explicit CD3DX12_RT_FORMAT_ARRAY(_In_reads_(NumFormats) const DXGI_FORMAT* pFormats, UINT NumFormats) noexcept + { + NumRenderTargets = NumFormats; + memcpy(RTFormats, pFormats, sizeof(RTFormats)); + // assumes ARRAY_SIZE(pFormats) == ARRAY_SIZE(RTFormats) + } +}; + +//------------------------------------------------------------------------------------------------ +// Pipeline State Stream Helpers +//------------------------------------------------------------------------------------------------ + +//------------------------------------------------------------------------------------------------ +// Stream Subobjects, i.e. elements of a stream + +struct DefaultSampleMask { operator UINT() noexcept { return UINT_MAX; } }; +struct DefaultSampleDesc { operator DXGI_SAMPLE_DESC() noexcept { return DXGI_SAMPLE_DESC{1, 0}; } }; + +#pragma warning(push) +#pragma warning(disable : 4324) +template +class alignas(void*) CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT +{ +private: + D3D12_PIPELINE_STATE_SUBOBJECT_TYPE _Type; + InnerStructType _Inner; +public: + CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT() noexcept : _Type(Type), _Inner(DefaultArg()) {} + CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT(InnerStructType const& i) noexcept : _Type(Type), _Inner(i) {} + CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT& operator=(InnerStructType const& i) noexcept { _Type = Type; _Inner = i; return *this; } + operator InnerStructType const&() const noexcept { return _Inner; } + operator InnerStructType&() noexcept { return _Inner; } + InnerStructType* operator&() noexcept { return &_Inner; } + InnerStructType const* operator&() const noexcept { return &_Inner; } +}; +#pragma warning(pop) +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_PIPELINE_STATE_FLAGS, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_FLAGS> CD3DX12_PIPELINE_STATE_STREAM_FLAGS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< UINT, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_NODE_MASK> CD3DX12_PIPELINE_STATE_STREAM_NODE_MASK; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< ID3D12RootSignature*, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_ROOT_SIGNATURE> CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_INPUT_LAYOUT_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_INPUT_LAYOUT> CD3DX12_PIPELINE_STATE_STREAM_INPUT_LAYOUT; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_INDEX_BUFFER_STRIP_CUT_VALUE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_IB_STRIP_CUT_VALUE> CD3DX12_PIPELINE_STATE_STREAM_IB_STRIP_CUT_VALUE; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_PRIMITIVE_TOPOLOGY_TYPE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_PRIMITIVE_TOPOLOGY> CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VS> CD3DX12_PIPELINE_STATE_STREAM_VS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_GS> CD3DX12_PIPELINE_STATE_STREAM_GS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_STREAM_OUTPUT_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_STREAM_OUTPUT> CD3DX12_PIPELINE_STATE_STREAM_STREAM_OUTPUT; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_HS> CD3DX12_PIPELINE_STATE_STREAM_HS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DS> CD3DX12_PIPELINE_STATE_STREAM_DS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_PS> CD3DX12_PIPELINE_STATE_STREAM_PS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_AS> CD3DX12_PIPELINE_STATE_STREAM_AS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MS> CD3DX12_PIPELINE_STATE_STREAM_MS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CS> CD3DX12_PIPELINE_STATE_STREAM_CS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_BLEND_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_BLEND, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_DEPTH_STENCIL_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_DEPTH_STENCIL_DESC1, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL1, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL1; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< DXGI_FORMAT, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL_FORMAT> CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL_FORMAT; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_RASTERIZER_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_RASTERIZER, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_RASTERIZER; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_RT_FORMAT_ARRAY, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_RENDER_TARGET_FORMATS> CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< DXGI_SAMPLE_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_DESC, DefaultSampleDesc> CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< UINT, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_MASK, DefaultSampleMask> CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_CACHED_PIPELINE_STATE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CACHED_PSO> CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO; +typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_VIEW_INSTANCING_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VIEW_INSTANCING, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_VIEW_INSTANCING; + +//------------------------------------------------------------------------------------------------ +// Stream Parser Helpers + +struct ID3DX12PipelineParserCallbacks +{ + // Subobject Callbacks + virtual void FlagsCb(D3D12_PIPELINE_STATE_FLAGS) {} + virtual void NodeMaskCb(UINT) {} + virtual void RootSignatureCb(ID3D12RootSignature*) {} + virtual void InputLayoutCb(const D3D12_INPUT_LAYOUT_DESC&) {} + virtual void IBStripCutValueCb(D3D12_INDEX_BUFFER_STRIP_CUT_VALUE) {} + virtual void PrimitiveTopologyTypeCb(D3D12_PRIMITIVE_TOPOLOGY_TYPE) {} + virtual void VSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void GSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void StreamOutputCb(const D3D12_STREAM_OUTPUT_DESC&) {} + virtual void HSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void DSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void PSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void CSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void ASCb(const D3D12_SHADER_BYTECODE&) {} + virtual void MSCb(const D3D12_SHADER_BYTECODE&) {} + virtual void BlendStateCb(const D3D12_BLEND_DESC&) {} + virtual void DepthStencilStateCb(const D3D12_DEPTH_STENCIL_DESC&) {} + virtual void DepthStencilState1Cb(const D3D12_DEPTH_STENCIL_DESC1&) {} + virtual void DSVFormatCb(DXGI_FORMAT) {} + virtual void RasterizerStateCb(const D3D12_RASTERIZER_DESC&) {} + virtual void RTVFormatsCb(const D3D12_RT_FORMAT_ARRAY&) {} + virtual void SampleDescCb(const DXGI_SAMPLE_DESC&) {} + virtual void SampleMaskCb(UINT) {} + virtual void ViewInstancingCb(const D3D12_VIEW_INSTANCING_DESC&) {} + virtual void CachedPSOCb(const D3D12_CACHED_PIPELINE_STATE&) {} + + // Error Callbacks + virtual void ErrorBadInputParameter(UINT /*ParameterIndex*/) {} + virtual void ErrorDuplicateSubobject(D3D12_PIPELINE_STATE_SUBOBJECT_TYPE /*DuplicateType*/) {} + virtual void ErrorUnknownSubobject(UINT /*UnknownTypeValue*/) {} + + virtual ~ID3DX12PipelineParserCallbacks() = default; +}; + +struct D3DX12_MESH_SHADER_PIPELINE_STATE_DESC +{ + ID3D12RootSignature* pRootSignature; + D3D12_SHADER_BYTECODE AS; + D3D12_SHADER_BYTECODE MS; + D3D12_SHADER_BYTECODE PS; + D3D12_BLEND_DESC BlendState; + UINT SampleMask; + D3D12_RASTERIZER_DESC RasterizerState; + D3D12_DEPTH_STENCIL_DESC DepthStencilState; + D3D12_PRIMITIVE_TOPOLOGY_TYPE PrimitiveTopologyType; + UINT NumRenderTargets; + DXGI_FORMAT RTVFormats[ D3D12_SIMULTANEOUS_RENDER_TARGET_COUNT ]; + DXGI_FORMAT DSVFormat; + DXGI_SAMPLE_DESC SampleDesc; + UINT NodeMask; + D3D12_CACHED_PIPELINE_STATE CachedPSO; + D3D12_PIPELINE_STATE_FLAGS Flags; +}; + +// CD3DX12_PIPELINE_STATE_STREAM2 Works on OS Build 19041+ (where there is a new mesh shader pipeline). +// Use CD3DX12_PIPELINE_STATE_STREAM1 for OS Build 16299+ (where there is a new view instancing subobject). +// Use CD3DX12_PIPELINE_STATE_STREAM for OS Build 15063+ support. +struct CD3DX12_PIPELINE_STATE_STREAM2 +{ + CD3DX12_PIPELINE_STATE_STREAM2() = default; + // Mesh and amplification shaders must be set manually, since they do not have representation in D3D12_GRAPHICS_PIPELINE_STATE_DESC + CD3DX12_PIPELINE_STATE_STREAM2(const D3D12_GRAPHICS_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , InputLayout(Desc.InputLayout) + , IBStripCutValue(Desc.IBStripCutValue) + , PrimitiveTopologyType(Desc.PrimitiveTopologyType) + , VS(Desc.VS) + , GS(Desc.GS) + , StreamOutput(Desc.StreamOutput) + , HS(Desc.HS) + , DS(Desc.DS) + , PS(Desc.PS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) + {} + CD3DX12_PIPELINE_STATE_STREAM2(const D3DX12_MESH_SHADER_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , PrimitiveTopologyType(Desc.PrimitiveTopologyType) + , PS(Desc.PS) + , AS(Desc.AS) + , MS(Desc.MS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) + {} + CD3DX12_PIPELINE_STATE_STREAM2(const D3D12_COMPUTE_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , CS(CD3DX12_SHADER_BYTECODE(Desc.CS)) + , CachedPSO(Desc.CachedPSO) + { + static_cast(DepthStencilState).DepthEnable = false; + } + CD3DX12_PIPELINE_STATE_STREAM_FLAGS Flags; + CD3DX12_PIPELINE_STATE_STREAM_NODE_MASK NodeMask; + CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE pRootSignature; + CD3DX12_PIPELINE_STATE_STREAM_INPUT_LAYOUT InputLayout; + CD3DX12_PIPELINE_STATE_STREAM_IB_STRIP_CUT_VALUE IBStripCutValue; + CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY PrimitiveTopologyType; + CD3DX12_PIPELINE_STATE_STREAM_VS VS; + CD3DX12_PIPELINE_STATE_STREAM_GS GS; + CD3DX12_PIPELINE_STATE_STREAM_STREAM_OUTPUT StreamOutput; + CD3DX12_PIPELINE_STATE_STREAM_HS HS; + CD3DX12_PIPELINE_STATE_STREAM_DS DS; + CD3DX12_PIPELINE_STATE_STREAM_PS PS; + CD3DX12_PIPELINE_STATE_STREAM_AS AS; + CD3DX12_PIPELINE_STATE_STREAM_MS MS; + CD3DX12_PIPELINE_STATE_STREAM_CS CS; + CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC BlendState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL1 DepthStencilState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL_FORMAT DSVFormat; + CD3DX12_PIPELINE_STATE_STREAM_RASTERIZER RasterizerState; + CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS RTVFormats; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC SampleDesc; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK SampleMask; + CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO CachedPSO; + CD3DX12_PIPELINE_STATE_STREAM_VIEW_INSTANCING ViewInstancingDesc; + D3D12_GRAPHICS_PIPELINE_STATE_DESC GraphicsDescV0() const noexcept + { + D3D12_GRAPHICS_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.InputLayout = this->InputLayout; + D.IBStripCutValue = this->IBStripCutValue; + D.PrimitiveTopologyType = this->PrimitiveTopologyType; + D.VS = this->VS; + D.GS = this->GS; + D.StreamOutput = this->StreamOutput; + D.HS = this->HS; + D.DS = this->DS; + D.PS = this->PS; + D.BlendState = this->BlendState; + D.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(D3D12_DEPTH_STENCIL_DESC1(this->DepthStencilState)); + D.DSVFormat = this->DSVFormat; + D.RasterizerState = this->RasterizerState; + D.NumRenderTargets = D3D12_RT_FORMAT_ARRAY(this->RTVFormats).NumRenderTargets; + memcpy(D.RTVFormats, D3D12_RT_FORMAT_ARRAY(this->RTVFormats).RTFormats, sizeof(D.RTVFormats)); + D.SampleDesc = this->SampleDesc; + D.SampleMask = this->SampleMask; + D.CachedPSO = this->CachedPSO; + return D; + } + D3D12_COMPUTE_PIPELINE_STATE_DESC ComputeDescV0() const noexcept + { + D3D12_COMPUTE_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.CS = this->CS; + D.CachedPSO = this->CachedPSO; + return D; + } +}; + +// CD3DX12_PIPELINE_STATE_STREAM1 Works on OS Build 16299+ (where there is a new view instancing subobject). +// Use CD3DX12_PIPELINE_STATE_STREAM for OS Build 15063+ support. +struct CD3DX12_PIPELINE_STATE_STREAM1 +{ + CD3DX12_PIPELINE_STATE_STREAM1() = default; + // Mesh and amplification shaders must be set manually, since they do not have representation in D3D12_GRAPHICS_PIPELINE_STATE_DESC + CD3DX12_PIPELINE_STATE_STREAM1(const D3D12_GRAPHICS_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , InputLayout(Desc.InputLayout) + , IBStripCutValue(Desc.IBStripCutValue) + , PrimitiveTopologyType(Desc.PrimitiveTopologyType) + , VS(Desc.VS) + , GS(Desc.GS) + , StreamOutput(Desc.StreamOutput) + , HS(Desc.HS) + , DS(Desc.DS) + , PS(Desc.PS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) + {} + CD3DX12_PIPELINE_STATE_STREAM1(const D3DX12_MESH_SHADER_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , PrimitiveTopologyType(Desc.PrimitiveTopologyType) + , PS(Desc.PS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) + {} + CD3DX12_PIPELINE_STATE_STREAM1(const D3D12_COMPUTE_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , CS(CD3DX12_SHADER_BYTECODE(Desc.CS)) + , CachedPSO(Desc.CachedPSO) + { + static_cast(DepthStencilState).DepthEnable = false; + } + CD3DX12_PIPELINE_STATE_STREAM_FLAGS Flags; + CD3DX12_PIPELINE_STATE_STREAM_NODE_MASK NodeMask; + CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE pRootSignature; + CD3DX12_PIPELINE_STATE_STREAM_INPUT_LAYOUT InputLayout; + CD3DX12_PIPELINE_STATE_STREAM_IB_STRIP_CUT_VALUE IBStripCutValue; + CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY PrimitiveTopologyType; + CD3DX12_PIPELINE_STATE_STREAM_VS VS; + CD3DX12_PIPELINE_STATE_STREAM_GS GS; + CD3DX12_PIPELINE_STATE_STREAM_STREAM_OUTPUT StreamOutput; + CD3DX12_PIPELINE_STATE_STREAM_HS HS; + CD3DX12_PIPELINE_STATE_STREAM_DS DS; + CD3DX12_PIPELINE_STATE_STREAM_PS PS; + CD3DX12_PIPELINE_STATE_STREAM_CS CS; + CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC BlendState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL1 DepthStencilState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL_FORMAT DSVFormat; + CD3DX12_PIPELINE_STATE_STREAM_RASTERIZER RasterizerState; + CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS RTVFormats; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC SampleDesc; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK SampleMask; + CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO CachedPSO; + CD3DX12_PIPELINE_STATE_STREAM_VIEW_INSTANCING ViewInstancingDesc; + D3D12_GRAPHICS_PIPELINE_STATE_DESC GraphicsDescV0() const noexcept + { + D3D12_GRAPHICS_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.InputLayout = this->InputLayout; + D.IBStripCutValue = this->IBStripCutValue; + D.PrimitiveTopologyType = this->PrimitiveTopologyType; + D.VS = this->VS; + D.GS = this->GS; + D.StreamOutput = this->StreamOutput; + D.HS = this->HS; + D.DS = this->DS; + D.PS = this->PS; + D.BlendState = this->BlendState; + D.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(D3D12_DEPTH_STENCIL_DESC1(this->DepthStencilState)); + D.DSVFormat = this->DSVFormat; + D.RasterizerState = this->RasterizerState; + D.NumRenderTargets = D3D12_RT_FORMAT_ARRAY(this->RTVFormats).NumRenderTargets; + memcpy(D.RTVFormats, D3D12_RT_FORMAT_ARRAY(this->RTVFormats).RTFormats, sizeof(D.RTVFormats)); + D.SampleDesc = this->SampleDesc; + D.SampleMask = this->SampleMask; + D.CachedPSO = this->CachedPSO; + return D; + } + D3D12_COMPUTE_PIPELINE_STATE_DESC ComputeDescV0() const noexcept + { + D3D12_COMPUTE_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.CS = this->CS; + D.CachedPSO = this->CachedPSO; + return D; + } +}; + + +struct CD3DX12_PIPELINE_MESH_STATE_STREAM +{ + CD3DX12_PIPELINE_MESH_STATE_STREAM() = default; + CD3DX12_PIPELINE_MESH_STATE_STREAM(const D3DX12_MESH_SHADER_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , PS(Desc.PS) + , AS(Desc.AS) + , MS(Desc.MS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) + {} + CD3DX12_PIPELINE_STATE_STREAM_FLAGS Flags; + CD3DX12_PIPELINE_STATE_STREAM_NODE_MASK NodeMask; + CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE pRootSignature; + CD3DX12_PIPELINE_STATE_STREAM_PS PS; + CD3DX12_PIPELINE_STATE_STREAM_AS AS; + CD3DX12_PIPELINE_STATE_STREAM_MS MS; + CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC BlendState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL1 DepthStencilState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL_FORMAT DSVFormat; + CD3DX12_PIPELINE_STATE_STREAM_RASTERIZER RasterizerState; + CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS RTVFormats; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC SampleDesc; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK SampleMask; + CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO CachedPSO; + CD3DX12_PIPELINE_STATE_STREAM_VIEW_INSTANCING ViewInstancingDesc; + D3DX12_MESH_SHADER_PIPELINE_STATE_DESC MeshShaderDescV0() const noexcept + { + D3DX12_MESH_SHADER_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.PS = this->PS; + D.AS = this->AS; + D.MS = this->MS; + D.BlendState = this->BlendState; + D.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(D3D12_DEPTH_STENCIL_DESC1(this->DepthStencilState)); + D.DSVFormat = this->DSVFormat; + D.RasterizerState = this->RasterizerState; + D.NumRenderTargets = D3D12_RT_FORMAT_ARRAY(this->RTVFormats).NumRenderTargets; + memcpy(D.RTVFormats, D3D12_RT_FORMAT_ARRAY(this->RTVFormats).RTFormats, sizeof(D.RTVFormats)); + D.SampleDesc = this->SampleDesc; + D.SampleMask = this->SampleMask; + D.CachedPSO = this->CachedPSO; + return D; + } +}; + +// CD3DX12_PIPELINE_STATE_STREAM works on OS Build 15063+ but does not support new subobject(s) added in OS Build 16299+. +// See CD3DX12_PIPELINE_STATE_STREAM1 for instance. +struct CD3DX12_PIPELINE_STATE_STREAM +{ + CD3DX12_PIPELINE_STATE_STREAM() = default; + CD3DX12_PIPELINE_STATE_STREAM(const D3D12_GRAPHICS_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , InputLayout(Desc.InputLayout) + , IBStripCutValue(Desc.IBStripCutValue) + , PrimitiveTopologyType(Desc.PrimitiveTopologyType) + , VS(Desc.VS) + , GS(Desc.GS) + , StreamOutput(Desc.StreamOutput) + , HS(Desc.HS) + , DS(Desc.DS) + , PS(Desc.PS) + , BlendState(CD3DX12_BLEND_DESC(Desc.BlendState)) + , DepthStencilState(CD3DX12_DEPTH_STENCIL_DESC1(Desc.DepthStencilState)) + , DSVFormat(Desc.DSVFormat) + , RasterizerState(CD3DX12_RASTERIZER_DESC(Desc.RasterizerState)) + , RTVFormats(CD3DX12_RT_FORMAT_ARRAY(Desc.RTVFormats, Desc.NumRenderTargets)) + , SampleDesc(Desc.SampleDesc) + , SampleMask(Desc.SampleMask) + , CachedPSO(Desc.CachedPSO) + {} + CD3DX12_PIPELINE_STATE_STREAM(const D3D12_COMPUTE_PIPELINE_STATE_DESC& Desc) noexcept + : Flags(Desc.Flags) + , NodeMask(Desc.NodeMask) + , pRootSignature(Desc.pRootSignature) + , CS(CD3DX12_SHADER_BYTECODE(Desc.CS)) + , CachedPSO(Desc.CachedPSO) + {} + CD3DX12_PIPELINE_STATE_STREAM_FLAGS Flags; + CD3DX12_PIPELINE_STATE_STREAM_NODE_MASK NodeMask; + CD3DX12_PIPELINE_STATE_STREAM_ROOT_SIGNATURE pRootSignature; + CD3DX12_PIPELINE_STATE_STREAM_INPUT_LAYOUT InputLayout; + CD3DX12_PIPELINE_STATE_STREAM_IB_STRIP_CUT_VALUE IBStripCutValue; + CD3DX12_PIPELINE_STATE_STREAM_PRIMITIVE_TOPOLOGY PrimitiveTopologyType; + CD3DX12_PIPELINE_STATE_STREAM_VS VS; + CD3DX12_PIPELINE_STATE_STREAM_GS GS; + CD3DX12_PIPELINE_STATE_STREAM_STREAM_OUTPUT StreamOutput; + CD3DX12_PIPELINE_STATE_STREAM_HS HS; + CD3DX12_PIPELINE_STATE_STREAM_DS DS; + CD3DX12_PIPELINE_STATE_STREAM_PS PS; + CD3DX12_PIPELINE_STATE_STREAM_CS CS; + CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC BlendState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL1 DepthStencilState; + CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL_FORMAT DSVFormat; + CD3DX12_PIPELINE_STATE_STREAM_RASTERIZER RasterizerState; + CD3DX12_PIPELINE_STATE_STREAM_RENDER_TARGET_FORMATS RTVFormats; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC SampleDesc; + CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK SampleMask; + CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO CachedPSO; + D3D12_GRAPHICS_PIPELINE_STATE_DESC GraphicsDescV0() const noexcept + { + D3D12_GRAPHICS_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.InputLayout = this->InputLayout; + D.IBStripCutValue = this->IBStripCutValue; + D.PrimitiveTopologyType = this->PrimitiveTopologyType; + D.VS = this->VS; + D.GS = this->GS; + D.StreamOutput = this->StreamOutput; + D.HS = this->HS; + D.DS = this->DS; + D.PS = this->PS; + D.BlendState = this->BlendState; + D.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(D3D12_DEPTH_STENCIL_DESC1(this->DepthStencilState)); + D.DSVFormat = this->DSVFormat; + D.RasterizerState = this->RasterizerState; + D.NumRenderTargets = D3D12_RT_FORMAT_ARRAY(this->RTVFormats).NumRenderTargets; + memcpy(D.RTVFormats, D3D12_RT_FORMAT_ARRAY(this->RTVFormats).RTFormats, sizeof(D.RTVFormats)); + D.SampleDesc = this->SampleDesc; + D.SampleMask = this->SampleMask; + D.CachedPSO = this->CachedPSO; + return D; + } + D3D12_COMPUTE_PIPELINE_STATE_DESC ComputeDescV0() const noexcept + { + D3D12_COMPUTE_PIPELINE_STATE_DESC D; + D.Flags = this->Flags; + D.NodeMask = this->NodeMask; + D.pRootSignature = this->pRootSignature; + D.CS = this->CS; + D.CachedPSO = this->CachedPSO; + return D; + } +}; + +struct CD3DX12_PIPELINE_STATE_STREAM2_PARSE_HELPER : public ID3DX12PipelineParserCallbacks +{ + CD3DX12_PIPELINE_STATE_STREAM2 PipelineStream; + CD3DX12_PIPELINE_STATE_STREAM2_PARSE_HELPER() noexcept + : SeenDSS(false) + { + // Adjust defaults to account for absent members. + PipelineStream.PrimitiveTopologyType = D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE; + + // Depth disabled if no DSV format specified. + static_cast(PipelineStream.DepthStencilState).DepthEnable = false; + } + + // ID3DX12PipelineParserCallbacks + void FlagsCb(D3D12_PIPELINE_STATE_FLAGS Flags) override {PipelineStream.Flags = Flags;} + void NodeMaskCb(UINT NodeMask) override {PipelineStream.NodeMask = NodeMask;} + void RootSignatureCb(ID3D12RootSignature* pRootSignature) override {PipelineStream.pRootSignature = pRootSignature;} + void InputLayoutCb(const D3D12_INPUT_LAYOUT_DESC& InputLayout) override {PipelineStream.InputLayout = InputLayout;} + void IBStripCutValueCb(D3D12_INDEX_BUFFER_STRIP_CUT_VALUE IBStripCutValue) override {PipelineStream.IBStripCutValue = IBStripCutValue;} + void PrimitiveTopologyTypeCb(D3D12_PRIMITIVE_TOPOLOGY_TYPE PrimitiveTopologyType) override {PipelineStream.PrimitiveTopologyType = PrimitiveTopologyType;} + void VSCb(const D3D12_SHADER_BYTECODE& VS) override {PipelineStream.VS = VS;} + void GSCb(const D3D12_SHADER_BYTECODE& GS) override {PipelineStream.GS = GS;} + void StreamOutputCb(const D3D12_STREAM_OUTPUT_DESC& StreamOutput) override {PipelineStream.StreamOutput = StreamOutput;} + void HSCb(const D3D12_SHADER_BYTECODE& HS) override {PipelineStream.HS = HS;} + void DSCb(const D3D12_SHADER_BYTECODE& DS) override {PipelineStream.DS = DS;} + void PSCb(const D3D12_SHADER_BYTECODE& PS) override {PipelineStream.PS = PS;} + void CSCb(const D3D12_SHADER_BYTECODE& CS) override {PipelineStream.CS = CS;} + void ASCb(const D3D12_SHADER_BYTECODE& AS) override {PipelineStream.AS = AS;} + void MSCb(const D3D12_SHADER_BYTECODE& MS) override {PipelineStream.MS = MS;} + void BlendStateCb(const D3D12_BLEND_DESC& BlendState) override {PipelineStream.BlendState = CD3DX12_BLEND_DESC(BlendState);} + void DepthStencilStateCb(const D3D12_DEPTH_STENCIL_DESC& DepthStencilState) override + { + PipelineStream.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(DepthStencilState); + SeenDSS = true; + } + void DepthStencilState1Cb(const D3D12_DEPTH_STENCIL_DESC1& DepthStencilState) override + { + PipelineStream.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(DepthStencilState); + SeenDSS = true; + } + void DSVFormatCb(DXGI_FORMAT DSVFormat) override + { + PipelineStream.DSVFormat = DSVFormat; + if (!SeenDSS && DSVFormat != DXGI_FORMAT_UNKNOWN) + { + // Re-enable depth for the default state. + static_cast(PipelineStream.DepthStencilState).DepthEnable = true; + } + } + void RasterizerStateCb(const D3D12_RASTERIZER_DESC& RasterizerState) override {PipelineStream.RasterizerState = CD3DX12_RASTERIZER_DESC(RasterizerState);} + void RTVFormatsCb(const D3D12_RT_FORMAT_ARRAY& RTVFormats) override {PipelineStream.RTVFormats = RTVFormats;} + void SampleDescCb(const DXGI_SAMPLE_DESC& SampleDesc) override {PipelineStream.SampleDesc = SampleDesc;} + void SampleMaskCb(UINT SampleMask) override {PipelineStream.SampleMask = SampleMask;} + void ViewInstancingCb(const D3D12_VIEW_INSTANCING_DESC& ViewInstancingDesc) override {PipelineStream.ViewInstancingDesc = CD3DX12_VIEW_INSTANCING_DESC(ViewInstancingDesc);} + void CachedPSOCb(const D3D12_CACHED_PIPELINE_STATE& CachedPSO) override {PipelineStream.CachedPSO = CachedPSO;} + +private: + bool SeenDSS; +}; + + +struct CD3DX12_PIPELINE_STATE_STREAM_PARSE_HELPER : public ID3DX12PipelineParserCallbacks +{ + CD3DX12_PIPELINE_STATE_STREAM1 PipelineStream; + CD3DX12_PIPELINE_STATE_STREAM_PARSE_HELPER() noexcept + : SeenDSS(false) + { + // Adjust defaults to account for absent members. + PipelineStream.PrimitiveTopologyType = D3D12_PRIMITIVE_TOPOLOGY_TYPE_TRIANGLE; + + // Depth disabled if no DSV format specified. + static_cast(PipelineStream.DepthStencilState).DepthEnable = false; + } + + // ID3DX12PipelineParserCallbacks + void FlagsCb(D3D12_PIPELINE_STATE_FLAGS Flags) override {PipelineStream.Flags = Flags;} + void NodeMaskCb(UINT NodeMask) override {PipelineStream.NodeMask = NodeMask;} + void RootSignatureCb(ID3D12RootSignature* pRootSignature) override {PipelineStream.pRootSignature = pRootSignature;} + void InputLayoutCb(const D3D12_INPUT_LAYOUT_DESC& InputLayout) override {PipelineStream.InputLayout = InputLayout;} + void IBStripCutValueCb(D3D12_INDEX_BUFFER_STRIP_CUT_VALUE IBStripCutValue) override {PipelineStream.IBStripCutValue = IBStripCutValue;} + void PrimitiveTopologyTypeCb(D3D12_PRIMITIVE_TOPOLOGY_TYPE PrimitiveTopologyType) override {PipelineStream.PrimitiveTopologyType = PrimitiveTopologyType;} + void VSCb(const D3D12_SHADER_BYTECODE& VS) override {PipelineStream.VS = VS;} + void GSCb(const D3D12_SHADER_BYTECODE& GS) override {PipelineStream.GS = GS;} + void StreamOutputCb(const D3D12_STREAM_OUTPUT_DESC& StreamOutput) override {PipelineStream.StreamOutput = StreamOutput;} + void HSCb(const D3D12_SHADER_BYTECODE& HS) override {PipelineStream.HS = HS;} + void DSCb(const D3D12_SHADER_BYTECODE& DS) override {PipelineStream.DS = DS;} + void PSCb(const D3D12_SHADER_BYTECODE& PS) override {PipelineStream.PS = PS;} + void CSCb(const D3D12_SHADER_BYTECODE& CS) override {PipelineStream.CS = CS;} + void BlendStateCb(const D3D12_BLEND_DESC& BlendState) override {PipelineStream.BlendState = CD3DX12_BLEND_DESC(BlendState);} + void DepthStencilStateCb(const D3D12_DEPTH_STENCIL_DESC& DepthStencilState) override + { + PipelineStream.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(DepthStencilState); + SeenDSS = true; + } + void DepthStencilState1Cb(const D3D12_DEPTH_STENCIL_DESC1& DepthStencilState) override + { + PipelineStream.DepthStencilState = CD3DX12_DEPTH_STENCIL_DESC1(DepthStencilState); + SeenDSS = true; + } + void DSVFormatCb(DXGI_FORMAT DSVFormat) override + { + PipelineStream.DSVFormat = DSVFormat; + if (!SeenDSS && DSVFormat != DXGI_FORMAT_UNKNOWN) + { + // Re-enable depth for the default state. + static_cast(PipelineStream.DepthStencilState).DepthEnable = true; + } + } + void RasterizerStateCb(const D3D12_RASTERIZER_DESC& RasterizerState) override {PipelineStream.RasterizerState = CD3DX12_RASTERIZER_DESC(RasterizerState);} + void RTVFormatsCb(const D3D12_RT_FORMAT_ARRAY& RTVFormats) override {PipelineStream.RTVFormats = RTVFormats;} + void SampleDescCb(const DXGI_SAMPLE_DESC& SampleDesc) override {PipelineStream.SampleDesc = SampleDesc;} + void SampleMaskCb(UINT SampleMask) override {PipelineStream.SampleMask = SampleMask;} + void ViewInstancingCb(const D3D12_VIEW_INSTANCING_DESC& ViewInstancingDesc) override {PipelineStream.ViewInstancingDesc = CD3DX12_VIEW_INSTANCING_DESC(ViewInstancingDesc);} + void CachedPSOCb(const D3D12_CACHED_PIPELINE_STATE& CachedPSO) override {PipelineStream.CachedPSO = CachedPSO;} + +private: + bool SeenDSS; +}; + +inline D3D12_PIPELINE_STATE_SUBOBJECT_TYPE D3DX12GetBaseSubobjectType(D3D12_PIPELINE_STATE_SUBOBJECT_TYPE SubobjectType) noexcept +{ + switch (SubobjectType) + { + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL1: + return D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL; + default: + return SubobjectType; + } +} + +inline HRESULT D3DX12ParsePipelineStream(const D3D12_PIPELINE_STATE_STREAM_DESC& Desc, ID3DX12PipelineParserCallbacks* pCallbacks) +{ + if (pCallbacks == nullptr) + { + return E_INVALIDARG; + } + + if (Desc.SizeInBytes == 0 || Desc.pPipelineStateSubobjectStream == nullptr) + { + pCallbacks->ErrorBadInputParameter(1); // first parameter issue + return E_INVALIDARG; + } + + bool SubobjectSeen[D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MAX_VALID] = {}; + for (SIZE_T CurOffset = 0, SizeOfSubobject = 0; CurOffset < Desc.SizeInBytes; CurOffset += SizeOfSubobject) + { + BYTE* pStream = static_cast(Desc.pPipelineStateSubobjectStream)+CurOffset; + auto SubobjectType = *reinterpret_cast(pStream); + if (SubobjectType < 0 || SubobjectType >= D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MAX_VALID) + { + pCallbacks->ErrorUnknownSubobject(SubobjectType); + return E_INVALIDARG; + } + if (SubobjectSeen[D3DX12GetBaseSubobjectType(SubobjectType)]) + { + pCallbacks->ErrorDuplicateSubobject(SubobjectType); + return E_INVALIDARG; // disallow subobject duplicates in a stream + } + SubobjectSeen[SubobjectType] = true; + switch (SubobjectType) + { + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_ROOT_SIGNATURE: + pCallbacks->RootSignatureCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::pRootSignature); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VS: + pCallbacks->VSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::VS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_PS: + pCallbacks->PSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::PS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DS: + pCallbacks->DSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::DS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_HS: + pCallbacks->HSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::HS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_GS: + pCallbacks->GSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::GS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CS: + pCallbacks->CSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::CS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_AS: + pCallbacks->ASCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM2::AS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MS: + pCallbacks->MSCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM2::MS); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_STREAM_OUTPUT: + pCallbacks->StreamOutputCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::StreamOutput); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_BLEND: + pCallbacks->BlendStateCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::BlendState); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_MASK: + pCallbacks->SampleMaskCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::SampleMask); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_RASTERIZER: + pCallbacks->RasterizerStateCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::RasterizerState); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL: + pCallbacks->DepthStencilStateCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL1: + pCallbacks->DepthStencilState1Cb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::DepthStencilState); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_INPUT_LAYOUT: + pCallbacks->InputLayoutCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::InputLayout); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_IB_STRIP_CUT_VALUE: + pCallbacks->IBStripCutValueCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::IBStripCutValue); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_PRIMITIVE_TOPOLOGY: + pCallbacks->PrimitiveTopologyTypeCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::PrimitiveTopologyType); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_RENDER_TARGET_FORMATS: + pCallbacks->RTVFormatsCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::RTVFormats); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL_FORMAT: + pCallbacks->DSVFormatCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::DSVFormat); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_DESC: + pCallbacks->SampleDescCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::SampleDesc); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_NODE_MASK: + pCallbacks->NodeMaskCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::NodeMask); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CACHED_PSO: + pCallbacks->CachedPSOCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::CachedPSO); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_FLAGS: + pCallbacks->FlagsCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::Flags); + break; + case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VIEW_INSTANCING: + pCallbacks->ViewInstancingCb(*reinterpret_cast(pStream)); + SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM1::ViewInstancingDesc); + break; + default: + pCallbacks->ErrorUnknownSubobject(SubobjectType); + return E_INVALIDARG; + } + } + + return S_OK; +} + +//------------------------------------------------------------------------------------------------ +inline bool operator==( const D3D12_CLEAR_VALUE &a, const D3D12_CLEAR_VALUE &b) noexcept +{ + if (a.Format != b.Format) return false; + if (a.Format == DXGI_FORMAT_D24_UNORM_S8_UINT + || a.Format == DXGI_FORMAT_D16_UNORM + || a.Format == DXGI_FORMAT_D32_FLOAT + || a.Format == DXGI_FORMAT_D32_FLOAT_S8X24_UINT) + { + return (a.DepthStencil.Depth == b.DepthStencil.Depth) && + (a.DepthStencil.Stencil == b.DepthStencil.Stencil); + } else { + return (a.Color[0] == b.Color[0]) && + (a.Color[1] == b.Color[1]) && + (a.Color[2] == b.Color[2]) && + (a.Color[3] == b.Color[3]); + } +} +inline bool operator==( const D3D12_RENDER_PASS_BEGINNING_ACCESS_CLEAR_PARAMETERS &a, const D3D12_RENDER_PASS_BEGINNING_ACCESS_CLEAR_PARAMETERS &b) noexcept +{ + return a.ClearValue == b.ClearValue; +} +inline bool operator==( const D3D12_RENDER_PASS_ENDING_ACCESS_RESOLVE_PARAMETERS &a, const D3D12_RENDER_PASS_ENDING_ACCESS_RESOLVE_PARAMETERS &b) noexcept +{ + if (a.pSrcResource != b.pSrcResource) return false; + if (a.pDstResource != b.pDstResource) return false; + if (a.SubresourceCount != b.SubresourceCount) return false; + if (a.Format != b.Format) return false; + if (a.ResolveMode != b.ResolveMode) return false; + if (a.PreserveResolveSource != b.PreserveResolveSource) return false; + return true; +} +inline bool operator==( const D3D12_RENDER_PASS_BEGINNING_ACCESS &a, const D3D12_RENDER_PASS_BEGINNING_ACCESS &b) noexcept +{ + if (a.Type != b.Type) return false; + if (a.Type == D3D12_RENDER_PASS_BEGINNING_ACCESS_TYPE_CLEAR && !(a.Clear == b.Clear)) return false; + return true; +} +inline bool operator==( const D3D12_RENDER_PASS_ENDING_ACCESS &a, const D3D12_RENDER_PASS_ENDING_ACCESS &b) noexcept +{ + if (a.Type != b.Type) return false; + if (a.Type == D3D12_RENDER_PASS_ENDING_ACCESS_TYPE_RESOLVE && !(a.Resolve == b.Resolve)) return false; + return true; +} +inline bool operator==( const D3D12_RENDER_PASS_RENDER_TARGET_DESC &a, const D3D12_RENDER_PASS_RENDER_TARGET_DESC &b) noexcept +{ + if (a.cpuDescriptor.ptr != b.cpuDescriptor.ptr) return false; + if (!(a.BeginningAccess == b.BeginningAccess)) return false; + if (!(a.EndingAccess == b.EndingAccess)) return false; + return true; +} +inline bool operator==( const D3D12_RENDER_PASS_DEPTH_STENCIL_DESC &a, const D3D12_RENDER_PASS_DEPTH_STENCIL_DESC &b) noexcept +{ + if (a.cpuDescriptor.ptr != b.cpuDescriptor.ptr) return false; + if (!(a.DepthBeginningAccess == b.DepthBeginningAccess)) return false; + if (!(a.StencilBeginningAccess == b.StencilBeginningAccess)) return false; + if (!(a.DepthEndingAccess == b.DepthEndingAccess)) return false; + if (!(a.StencilEndingAccess == b.StencilEndingAccess)) return false; + return true; +} + + +#ifndef D3DX12_NO_STATE_OBJECT_HELPERS + +//================================================================================================ +// D3DX12 State Object Creation Helpers +// +// Helper classes for creating new style state objects out of an arbitrary set of subobjects. +// Uses STL +// +// Start by instantiating CD3DX12_STATE_OBJECT_DESC (see it's public methods). +// One of its methods is CreateSubobject(), which has a comment showing a couple of options for +// defining subobjects using the helper classes for each subobject (CD3DX12_DXIL_LIBRARY_SUBOBJECT +// etc.). The subobject helpers each have methods specific to the subobject for configuring it's +// contents. +// +//================================================================================================ +#include +#include +#include +#include +#ifndef D3DX12_USE_ATL +#include +#define D3DX12_COM_PTR Microsoft::WRL::ComPtr +#define D3DX12_COM_PTR_GET(x) x.Get() +#define D3DX12_COM_PTR_ADDRESSOF(x) x.GetAddressOf() +#else +#include +#define D3DX12_COM_PTR ATL::CComPtr +#define D3DX12_COM_PTR_GET(x) x.p +#define D3DX12_COM_PTR_ADDRESSOF(x) &x.p +#endif + +//------------------------------------------------------------------------------------------------ +class CD3DX12_STATE_OBJECT_DESC +{ +public: + CD3DX12_STATE_OBJECT_DESC() noexcept + { + Init(D3D12_STATE_OBJECT_TYPE_COLLECTION); + } + CD3DX12_STATE_OBJECT_DESC(D3D12_STATE_OBJECT_TYPE Type) noexcept + { + Init(Type); + } + void SetStateObjectType(D3D12_STATE_OBJECT_TYPE Type) noexcept { m_Desc.Type = Type; } + operator const D3D12_STATE_OBJECT_DESC&() + { + // Do final preparation work + m_RepointedAssociations.clear(); + m_SubobjectArray.clear(); + m_SubobjectArray.reserve(m_Desc.NumSubobjects); + // Flatten subobjects into an array (each flattened subobject still has a + // member that's a pointer to it's desc that's not flattened) + for (auto Iter = m_SubobjectList.begin(); + Iter != m_SubobjectList.end(); Iter++) + { + m_SubobjectArray.push_back(*Iter); + // Store new location in array so we can redirect pointers contained in subobjects + Iter->pSubobjectArrayLocation = &m_SubobjectArray.back(); + } + // For subobjects with pointer fields, create a new copy of those subobject definitions + // with fixed pointers + for (UINT i = 0; i < m_Desc.NumSubobjects; i++) + { + if (m_SubobjectArray[i].Type == D3D12_STATE_SUBOBJECT_TYPE_SUBOBJECT_TO_EXPORTS_ASSOCIATION) + { + auto pOriginalSubobjectAssociation = + static_cast(m_SubobjectArray[i].pDesc); + D3D12_SUBOBJECT_TO_EXPORTS_ASSOCIATION Repointed = *pOriginalSubobjectAssociation; + auto pWrapper = + static_cast(pOriginalSubobjectAssociation->pSubobjectToAssociate); + Repointed.pSubobjectToAssociate = pWrapper->pSubobjectArrayLocation; + m_RepointedAssociations.push_back(Repointed); + m_SubobjectArray[i].pDesc = &m_RepointedAssociations.back(); + } + } + // Below: using ugly way to get pointer in case .data() is not defined + m_Desc.pSubobjects = m_Desc.NumSubobjects ? &m_SubobjectArray[0] : nullptr; + return m_Desc; + } + operator const D3D12_STATE_OBJECT_DESC*() + { + // Cast calls the above final preparation work + return &static_cast(*this); + } + + // CreateSubobject creates a sububject helper (e.g. CD3DX12_HIT_GROUP_SUBOBJECT) + // whose lifetime is owned by this class. + // e.g. + // + // CD3DX12_STATE_OBJECT_DESC Collection1(D3D12_STATE_OBJECT_TYPE_COLLECTION); + // auto Lib0 = Collection1.CreateSubobject(); + // Lib0->SetDXILLibrary(&pMyAppDxilLibs[0]); + // Lib0->DefineExport(L"rayGenShader0"); // in practice these export listings might be + // // data/engine driven + // etc. + // + // Alternatively, users can instantiate sububject helpers explicitly, such as via local + // variables instead, passing the state object desc that should point to it into the helper + // constructor (or call mySubobjectHelper.AddToStateObject(Collection1)). + // In this alternative scenario, the user must keep the subobject alive as long as the state + // object it is associated with is alive, else it's pointer references will be stale. + // e.g. + // + // CD3DX12_STATE_OBJECT_DESC RaytracingState2(D3D12_STATE_OBJECT_TYPE_RAYTRACING_PIPELINE); + // CD3DX12_DXIL_LIBRARY_SUBOBJECT LibA(RaytracingState2); + // LibA.SetDXILLibrary(&pMyAppDxilLibs[4]); // not manually specifying exports + // // - meaning all exports in the libraries + // // are exported + // etc. + + template + T* CreateSubobject() + { + T* pSubobject = new T(*this); + m_OwnedSubobjectHelpers.emplace_back(pSubobject); + return pSubobject; + } + +private: + D3D12_STATE_SUBOBJECT* TrackSubobject(D3D12_STATE_SUBOBJECT_TYPE Type, void* pDesc) + { + SUBOBJECT_WRAPPER Subobject; + Subobject.pSubobjectArrayLocation = nullptr; + Subobject.Type = Type; + Subobject.pDesc = pDesc; + m_SubobjectList.push_back(Subobject); + m_Desc.NumSubobjects++; + return &m_SubobjectList.back(); + } + void Init(D3D12_STATE_OBJECT_TYPE Type) noexcept + { + SetStateObjectType(Type); + m_Desc.pSubobjects = nullptr; + m_Desc.NumSubobjects = 0; + m_SubobjectList.clear(); + m_SubobjectArray.clear(); + m_RepointedAssociations.clear(); + } + typedef struct SUBOBJECT_WRAPPER : public D3D12_STATE_SUBOBJECT + { + D3D12_STATE_SUBOBJECT* pSubobjectArrayLocation; // new location when flattened into array + // for repointing pointers in subobjects + } SUBOBJECT_WRAPPER; + D3D12_STATE_OBJECT_DESC m_Desc; + std::list m_SubobjectList; // Pointers to list nodes handed out so + // these can be edited live + std::vector m_SubobjectArray; // Built at the end, copying list contents + + std::list + m_RepointedAssociations; // subobject type that contains pointers to other subobjects, + // repointed to flattened array + + class StringContainer + { + public: + LPCWSTR LocalCopy(LPCWSTR string, bool bSingleString = false) + { + if (string) + { + if (bSingleString) + { + m_Strings.clear(); + m_Strings.push_back(string); + } + else + { + m_Strings.push_back(string); + } + return m_Strings.back().c_str(); + } + else + { + return nullptr; + } + } + void clear() noexcept { m_Strings.clear(); } + private: + std::list m_Strings; + }; + + class SUBOBJECT_HELPER_BASE + { + public: + SUBOBJECT_HELPER_BASE() noexcept { Init(); } + virtual ~SUBOBJECT_HELPER_BASE() = default; + virtual D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept = 0; + void AddToStateObject(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + m_pSubobject = ContainingStateObject.TrackSubobject(Type(), Data()); + } + protected: + virtual void* Data() noexcept = 0; + void Init() noexcept { m_pSubobject = nullptr; } + D3D12_STATE_SUBOBJECT* m_pSubobject; + }; + +#if(__cplusplus >= 201103L) + std::list> m_OwnedSubobjectHelpers; +#else + class OWNED_HELPER + { + public: + OWNED_HELPER(const SUBOBJECT_HELPER_BASE* pHelper) noexcept { m_pHelper = pHelper; } + ~OWNED_HELPER() { delete m_pHelper; } + const SUBOBJECT_HELPER_BASE* m_pHelper; + }; + + std::list m_OwnedSubobjectHelpers; +#endif + + friend class CD3DX12_DXIL_LIBRARY_SUBOBJECT; + friend class CD3DX12_EXISTING_COLLECTION_SUBOBJECT; + friend class CD3DX12_SUBOBJECT_TO_EXPORTS_ASSOCIATION_SUBOBJECT; + friend class CD3DX12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION; + friend class CD3DX12_HIT_GROUP_SUBOBJECT; + friend class CD3DX12_RAYTRACING_SHADER_CONFIG_SUBOBJECT; + friend class CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT; + friend class CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT; + friend class CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT; + friend class CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT; + friend class CD3DX12_STATE_OBJECT_CONFIG_SUBOBJECT; + friend class CD3DX12_NODE_MASK_SUBOBJECT; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_DXIL_LIBRARY_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_DXIL_LIBRARY_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_DXIL_LIBRARY_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetDXILLibrary(const D3D12_SHADER_BYTECODE* pCode) noexcept + { + static const D3D12_SHADER_BYTECODE Default = {}; + m_Desc.DXILLibrary = pCode ? *pCode : Default; + } + void DefineExport( + LPCWSTR Name, + LPCWSTR ExportToRename = nullptr, + D3D12_EXPORT_FLAGS Flags = D3D12_EXPORT_FLAG_NONE) + { + D3D12_EXPORT_DESC Export; + Export.Name = m_Strings.LocalCopy(Name); + Export.ExportToRename = m_Strings.LocalCopy(ExportToRename); + Export.Flags = Flags; + m_Exports.push_back(Export); + m_Desc.pExports = &m_Exports[0]; // using ugly way to get pointer in case .data() is not defined + m_Desc.NumExports = static_cast(m_Exports.size()); + } + template + void DefineExports(LPCWSTR(&Exports)[N]) + { + for (UINT i = 0; i < N; i++) + { + DefineExport(Exports[i]); + } + } + void DefineExports(const LPCWSTR* Exports, UINT N) + { + for (UINT i = 0; i < N; i++) + { + DefineExport(Exports[i]); + } + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_DXIL_LIBRARY; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_DXIL_LIBRARY_DESC&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + m_Strings.clear(); + m_Exports.clear(); + } + void* Data() noexcept override { return &m_Desc; } + D3D12_DXIL_LIBRARY_DESC m_Desc; + CD3DX12_STATE_OBJECT_DESC::StringContainer m_Strings; + std::vector m_Exports; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_EXISTING_COLLECTION_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_EXISTING_COLLECTION_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_EXISTING_COLLECTION_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetExistingCollection(ID3D12StateObject*pExistingCollection) noexcept + { + m_Desc.pExistingCollection = pExistingCollection; + m_CollectionRef = pExistingCollection; + } + void DefineExport( + LPCWSTR Name, + LPCWSTR ExportToRename = nullptr, + D3D12_EXPORT_FLAGS Flags = D3D12_EXPORT_FLAG_NONE) + { + D3D12_EXPORT_DESC Export; + Export.Name = m_Strings.LocalCopy(Name); + Export.ExportToRename = m_Strings.LocalCopy(ExportToRename); + Export.Flags = Flags; + m_Exports.push_back(Export); + m_Desc.pExports = &m_Exports[0]; // using ugly way to get pointer in case .data() is not defined + m_Desc.NumExports = static_cast(m_Exports.size()); + } + template + void DefineExports(LPCWSTR(&Exports)[N]) + { + for (UINT i = 0; i < N; i++) + { + DefineExport(Exports[i]); + } + } + void DefineExports(const LPCWSTR* Exports, UINT N) + { + for (UINT i = 0; i < N; i++) + { + DefineExport(Exports[i]); + } + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_EXISTING_COLLECTION; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_EXISTING_COLLECTION_DESC&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + m_CollectionRef = nullptr; + m_Strings.clear(); + m_Exports.clear(); + } + void* Data() noexcept override { return &m_Desc; } + D3D12_EXISTING_COLLECTION_DESC m_Desc; + D3DX12_COM_PTR m_CollectionRef; + CD3DX12_STATE_OBJECT_DESC::StringContainer m_Strings; + std::vector m_Exports; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_SUBOBJECT_TO_EXPORTS_ASSOCIATION_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_SUBOBJECT_TO_EXPORTS_ASSOCIATION_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_SUBOBJECT_TO_EXPORTS_ASSOCIATION_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetSubobjectToAssociate(const D3D12_STATE_SUBOBJECT& SubobjectToAssociate) noexcept + { + m_Desc.pSubobjectToAssociate = &SubobjectToAssociate; + } + void AddExport(LPCWSTR Export) + { + m_Desc.NumExports++; + m_Exports.push_back(m_Strings.LocalCopy(Export)); + m_Desc.pExports = &m_Exports[0]; // using ugly way to get pointer in case .data() is not defined + } + template + void AddExports(LPCWSTR (&Exports)[N]) + { + for (UINT i = 0; i < N; i++) + { + AddExport(Exports[i]); + } + } + void AddExports(const LPCWSTR* Exports, UINT N) + { + for (UINT i = 0; i < N; i++) + { + AddExport(Exports[i]); + } + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_SUBOBJECT_TO_EXPORTS_ASSOCIATION; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_SUBOBJECT_TO_EXPORTS_ASSOCIATION&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + m_Strings.clear(); + m_Exports.clear(); + } + void* Data() noexcept override { return &m_Desc; } + D3D12_SUBOBJECT_TO_EXPORTS_ASSOCIATION m_Desc; + CD3DX12_STATE_OBJECT_DESC::StringContainer m_Strings; + std::vector m_Exports; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION() noexcept + { + Init(); + } + CD3DX12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetSubobjectNameToAssociate(LPCWSTR SubobjectToAssociate) + { + m_Desc.SubobjectToAssociate = m_SubobjectName.LocalCopy(SubobjectToAssociate, true); + } + void AddExport(LPCWSTR Export) + { + m_Desc.NumExports++; + m_Exports.push_back(m_Strings.LocalCopy(Export)); + m_Desc.pExports = &m_Exports[0]; // using ugly way to get pointer in case .data() is not defined + } + template + void AddExports(LPCWSTR (&Exports)[N]) + { + for (UINT i = 0; i < N; i++) + { + AddExport(Exports[i]); + } + } + void AddExports(const LPCWSTR* Exports, UINT N) + { + for (UINT i = 0; i < N; i++) + { + AddExport(Exports[i]); + } + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + m_Strings.clear(); + m_SubobjectName.clear(); + m_Exports.clear(); + } + void* Data() noexcept override { return &m_Desc; } + D3D12_DXIL_SUBOBJECT_TO_EXPORTS_ASSOCIATION m_Desc; + CD3DX12_STATE_OBJECT_DESC::StringContainer m_Strings; + CD3DX12_STATE_OBJECT_DESC::StringContainer m_SubobjectName; + std::vector m_Exports; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_HIT_GROUP_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_HIT_GROUP_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_HIT_GROUP_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetHitGroupExport(LPCWSTR exportName) + { + m_Desc.HitGroupExport = m_Strings[0].LocalCopy(exportName, true); + } + void SetHitGroupType(D3D12_HIT_GROUP_TYPE Type) noexcept { m_Desc.Type = Type; } + void SetAnyHitShaderImport(LPCWSTR importName) + { + m_Desc.AnyHitShaderImport = m_Strings[1].LocalCopy(importName, true); + } + void SetClosestHitShaderImport(LPCWSTR importName) + { + m_Desc.ClosestHitShaderImport = m_Strings[2].LocalCopy(importName, true); + } + void SetIntersectionShaderImport(LPCWSTR importName) + { + m_Desc.IntersectionShaderImport = m_Strings[3].LocalCopy(importName, true); + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_HIT_GROUP; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_HIT_GROUP_DESC&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + for (UINT i = 0; i < m_NumStrings; i++) + { + m_Strings[i].clear(); + } + } + void* Data() noexcept override { return &m_Desc; } + D3D12_HIT_GROUP_DESC m_Desc; + static const UINT m_NumStrings = 4; + CD3DX12_STATE_OBJECT_DESC::StringContainer + m_Strings[m_NumStrings]; // one string for every entrypoint name +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_RAYTRACING_SHADER_CONFIG_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_RAYTRACING_SHADER_CONFIG_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_RAYTRACING_SHADER_CONFIG_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void Config(UINT MaxPayloadSizeInBytes, UINT MaxAttributeSizeInBytes) noexcept + { + m_Desc.MaxPayloadSizeInBytes = MaxPayloadSizeInBytes; + m_Desc.MaxAttributeSizeInBytes = MaxAttributeSizeInBytes; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_RAYTRACING_SHADER_CONFIG; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_RAYTRACING_SHADER_CONFIG&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + } + void* Data() noexcept override { return &m_Desc; } + D3D12_RAYTRACING_SHADER_CONFIG m_Desc; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void Config(UINT MaxTraceRecursionDepth) noexcept + { + m_Desc.MaxTraceRecursionDepth = MaxTraceRecursionDepth; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_RAYTRACING_PIPELINE_CONFIG; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_RAYTRACING_PIPELINE_CONFIG&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + } + void* Data() noexcept override { return &m_Desc; } + D3D12_RAYTRACING_PIPELINE_CONFIG m_Desc; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void Config(UINT MaxTraceRecursionDepth, D3D12_RAYTRACING_PIPELINE_FLAGS Flags) noexcept + { + m_Desc.MaxTraceRecursionDepth = MaxTraceRecursionDepth; + m_Desc.Flags = Flags; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_RAYTRACING_PIPELINE_CONFIG1; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_RAYTRACING_PIPELINE_CONFIG1&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + } + void* Data() noexcept override { return &m_Desc; } + D3D12_RAYTRACING_PIPELINE_CONFIG1 m_Desc; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetRootSignature(ID3D12RootSignature* pRootSig) noexcept + { + m_pRootSig = pRootSig; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_GLOBAL_ROOT_SIGNATURE; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator ID3D12RootSignature*() const noexcept { return D3DX12_COM_PTR_GET(m_pRootSig); } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_pRootSig = nullptr; + } + void* Data() noexcept override { return D3DX12_COM_PTR_ADDRESSOF(m_pRootSig); } + D3DX12_COM_PTR m_pRootSig; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetRootSignature(ID3D12RootSignature* pRootSig) noexcept + { + m_pRootSig = pRootSig; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_LOCAL_ROOT_SIGNATURE; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator ID3D12RootSignature*() const noexcept { return D3DX12_COM_PTR_GET(m_pRootSig); } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_pRootSig = nullptr; + } + void* Data() noexcept override { return D3DX12_COM_PTR_ADDRESSOF(m_pRootSig); } + D3DX12_COM_PTR m_pRootSig; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_STATE_OBJECT_CONFIG_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_STATE_OBJECT_CONFIG_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_STATE_OBJECT_CONFIG_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetFlags(D3D12_STATE_OBJECT_FLAGS Flags) noexcept + { + m_Desc.Flags = Flags; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_STATE_OBJECT_CONFIG; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_STATE_OBJECT_CONFIG&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + } + void* Data() noexcept override { return &m_Desc; } + D3D12_STATE_OBJECT_CONFIG m_Desc; +}; + +//------------------------------------------------------------------------------------------------ +class CD3DX12_NODE_MASK_SUBOBJECT + : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE +{ +public: + CD3DX12_NODE_MASK_SUBOBJECT() noexcept + { + Init(); + } + CD3DX12_NODE_MASK_SUBOBJECT(CD3DX12_STATE_OBJECT_DESC& ContainingStateObject) + { + Init(); + AddToStateObject(ContainingStateObject); + } + void SetNodeMask(UINT NodeMask) noexcept + { + m_Desc.NodeMask = NodeMask; + } + D3D12_STATE_SUBOBJECT_TYPE Type() const noexcept override + { + return D3D12_STATE_SUBOBJECT_TYPE_NODE_MASK; + } + operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } + operator const D3D12_NODE_MASK&() const noexcept { return m_Desc; } +private: + void Init() noexcept + { + SUBOBJECT_HELPER_BASE::Init(); + m_Desc = {}; + } + void* Data() noexcept override { return &m_Desc; } + D3D12_NODE_MASK m_Desc; +}; + +#undef D3DX12_COM_PTR +#undef D3DX12_COM_PTR_GET +#undef D3DX12_COM_PTR_ADDRESSOF +#endif // #ifndef D3DX12_NO_STATE_OBJECT_HELPERS + +#endif // defined( __cplusplus ) + +#endif //__D3DX12_H__ + + diff --git a/src/external/mikktspace/README.md b/src/external/mikktspace/README.md new file mode 100644 index 00000000..9fda1559 --- /dev/null +++ b/src/external/mikktspace/README.md @@ -0,0 +1,4 @@ +# MikkTSpace +A common standard for tangent space used in baking tools to produce normal maps. + +More information can be found at http://www.mikktspace.com/. diff --git a/src/external/mikktspace/mikktspace.c b/src/external/mikktspace/mikktspace.c new file mode 100644 index 00000000..0342ae01 --- /dev/null +++ b/src/external/mikktspace/mikktspace.c @@ -0,0 +1,1899 @@ +/** \file mikktspace/mikktspace.c + * \ingroup mikktspace + */ +/** + * Copyright (C) 2011 by Morten S. Mikkelsen + * + * This software is provided 'as-is', without any express or implied + * warranty. In no event will the authors be held liable for any damages + * arising from the use of this software. + * + * Permission is granted to anyone to use this software for any purpose, + * including commercial applications, and to alter it and redistribute it + * freely, subject to the following restrictions: + * + * 1. The origin of this software must not be misrepresented; you must not + * claim that you wrote the original software. If you use this software + * in a product, an acknowledgment in the product documentation would be + * appreciated but is not required. + * 2. Altered source versions must be plainly marked as such, and must not be + * misrepresented as being the original software. + * 3. This notice may not be removed or altered from any source distribution. + */ + +#include +#include +#include +#include +#include +#include + +#include "mikktspace.h" + +#define TFALSE 0 +#define TTRUE 1 + +#ifndef M_PI +#define M_PI 3.1415926535897932384626433832795 +#endif + +#define INTERNAL_RND_SORT_SEED 39871946 + +// internal structure +typedef struct { + float x, y, z; +} SVec3; + +static tbool veq( const SVec3 v1, const SVec3 v2 ) +{ + return (v1.x == v2.x) && (v1.y == v2.y) && (v1.z == v2.z); +} + +static SVec3 vadd( const SVec3 v1, const SVec3 v2 ) +{ + SVec3 vRes; + + vRes.x = v1.x + v2.x; + vRes.y = v1.y + v2.y; + vRes.z = v1.z + v2.z; + + return vRes; +} + + +static SVec3 vsub( const SVec3 v1, const SVec3 v2 ) +{ + SVec3 vRes; + + vRes.x = v1.x - v2.x; + vRes.y = v1.y - v2.y; + vRes.z = v1.z - v2.z; + + return vRes; +} + +static SVec3 vscale(const float fS, const SVec3 v) +{ + SVec3 vRes; + + vRes.x = fS * v.x; + vRes.y = fS * v.y; + vRes.z = fS * v.z; + + return vRes; +} + +static float LengthSquared( const SVec3 v ) +{ + return v.x*v.x + v.y*v.y + v.z*v.z; +} + +static float Length( const SVec3 v ) +{ + return sqrtf(LengthSquared(v)); +} + +static SVec3 Normalize( const SVec3 v ) +{ + return vscale(1 / Length(v), v); +} + +static float vdot( const SVec3 v1, const SVec3 v2) +{ + return v1.x*v2.x + v1.y*v2.y + v1.z*v2.z; +} + + +static tbool NotZero(const float fX) +{ + // could possibly use FLT_EPSILON instead + return fabsf(fX) > FLT_MIN; +} + +static tbool VNotZero(const SVec3 v) +{ + // might change this to an epsilon based test + return NotZero(v.x) || NotZero(v.y) || NotZero(v.z); +} + + + +typedef struct { + int iNrFaces; + int * pTriMembers; +} SSubGroup; + +typedef struct { + int iNrFaces; + int * pFaceIndices; + int iVertexRepresentitive; + tbool bOrientPreservering; +} SGroup; + +// +#define MARK_DEGENERATE 1 +#define QUAD_ONE_DEGEN_TRI 2 +#define GROUP_WITH_ANY 4 +#define ORIENT_PRESERVING 8 + + + +typedef struct { + int FaceNeighbors[3]; + SGroup * AssignedGroup[3]; + + // normalized first order face derivatives + SVec3 vOs, vOt; + float fMagS, fMagT; // original magnitudes + + // determines if the current and the next triangle are a quad. + int iOrgFaceNumber; + int iFlag, iTSpacesOffs; + unsigned char vert_num[4]; +} STriInfo; + +typedef struct { + SVec3 vOs; + float fMagS; + SVec3 vOt; + float fMagT; + int iCounter; // this is to average back into quads. + tbool bOrient; +} STSpace; + +static int GenerateInitialVerticesIndexList(STriInfo pTriInfos[], int piTriList_out[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn); +static void GenerateSharedVerticesIndexList(int piTriList_in_and_out[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn); +static void InitTriInfo(STriInfo pTriInfos[], const int piTriListIn[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn); +static int Build4RuleGroups(STriInfo pTriInfos[], SGroup pGroups[], int piGroupTrianglesBuffer[], const int piTriListIn[], const int iNrTrianglesIn); +static tbool GenerateTSpaces(STSpace psTspace[], const STriInfo pTriInfos[], const SGroup pGroups[], + const int iNrActiveGroups, const int piTriListIn[], const float fThresCos, + const SMikkTSpaceContext * pContext); + +static int MakeIndex(const int iFace, const int iVert) +{ + assert(iVert>=0 && iVert<4 && iFace>=0); + return (iFace<<2) | (iVert&0x3); +} + +static void IndexToData(int * piFace, int * piVert, const int iIndexIn) +{ + piVert[0] = iIndexIn&0x3; + piFace[0] = iIndexIn>>2; +} + +static STSpace AvgTSpace(const STSpace * pTS0, const STSpace * pTS1) +{ + STSpace ts_res; + + // this if is important. Due to floating point precision + // averaging when ts0==ts1 will cause a slight difference + // which results in tangent space splits later on + if (pTS0->fMagS==pTS1->fMagS && pTS0->fMagT==pTS1->fMagT && + veq(pTS0->vOs,pTS1->vOs) && veq(pTS0->vOt, pTS1->vOt)) + { + ts_res.fMagS = pTS0->fMagS; + ts_res.fMagT = pTS0->fMagT; + ts_res.vOs = pTS0->vOs; + ts_res.vOt = pTS0->vOt; + } + else + { + ts_res.fMagS = 0.5f*(pTS0->fMagS+pTS1->fMagS); + ts_res.fMagT = 0.5f*(pTS0->fMagT+pTS1->fMagT); + ts_res.vOs = vadd(pTS0->vOs,pTS1->vOs); + ts_res.vOt = vadd(pTS0->vOt,pTS1->vOt); + if ( VNotZero(ts_res.vOs) ) ts_res.vOs = Normalize(ts_res.vOs); + if ( VNotZero(ts_res.vOt) ) ts_res.vOt = Normalize(ts_res.vOt); + } + + return ts_res; +} + + + +static SVec3 GetPosition(const SMikkTSpaceContext * pContext, const int index); +static SVec3 GetNormal(const SMikkTSpaceContext * pContext, const int index); +static SVec3 GetTexCoord(const SMikkTSpaceContext * pContext, const int index); + + +// degen triangles +static void DegenPrologue(STriInfo pTriInfos[], int piTriList_out[], const int iNrTrianglesIn, const int iTotTris); +static void DegenEpilogue(STSpace psTspace[], STriInfo pTriInfos[], int piTriListIn[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn, const int iTotTris); + + +tbool genTangSpaceDefault(const SMikkTSpaceContext * pContext) +{ + return genTangSpace(pContext, 180.0f); +} + +tbool genTangSpace(const SMikkTSpaceContext * pContext, const float fAngularThreshold) +{ + // count nr_triangles + int * piTriListIn = NULL, * piGroupTrianglesBuffer = NULL; + STriInfo * pTriInfos = NULL; + SGroup * pGroups = NULL; + STSpace * psTspace = NULL; + int iNrTrianglesIn = 0, f=0, t=0, i=0; + int iNrTSPaces = 0, iTotTris = 0, iDegenTriangles = 0, iNrMaxGroups = 0; + int iNrActiveGroups = 0, index = 0; + const int iNrFaces = pContext->m_pInterface->m_getNumFaces(pContext); + tbool bRes = TFALSE; + const float fThresCos = (float) cos((fAngularThreshold*(float)M_PI)/180.0f); + + // verify all call-backs have been set + if ( pContext->m_pInterface->m_getNumFaces==NULL || + pContext->m_pInterface->m_getNumVerticesOfFace==NULL || + pContext->m_pInterface->m_getPosition==NULL || + pContext->m_pInterface->m_getNormal==NULL || + pContext->m_pInterface->m_getTexCoord==NULL ) + return TFALSE; + + // count triangles on supported faces + for (f=0; fm_pInterface->m_getNumVerticesOfFace(pContext, f); + if (verts==3) ++iNrTrianglesIn; + else if (verts==4) iNrTrianglesIn += 2; + } + if (iNrTrianglesIn<=0) return TFALSE; + + // allocate memory for an index list + piTriListIn = (int *) malloc(sizeof(int)*3*iNrTrianglesIn); + pTriInfos = (STriInfo *) malloc(sizeof(STriInfo)*iNrTrianglesIn); + if (piTriListIn==NULL || pTriInfos==NULL) + { + if (piTriListIn!=NULL) free(piTriListIn); + if (pTriInfos!=NULL) free(pTriInfos); + return TFALSE; + } + + // make an initial triangle --> face index list + iNrTSPaces = GenerateInitialVerticesIndexList(pTriInfos, piTriListIn, pContext, iNrTrianglesIn); + + // make a welded index list of identical positions and attributes (pos, norm, texc) + //printf("gen welded index list begin\n"); + GenerateSharedVerticesIndexList(piTriListIn, pContext, iNrTrianglesIn); + //printf("gen welded index list end\n"); + + // Mark all degenerate triangles + iTotTris = iNrTrianglesIn; + iDegenTriangles = 0; + for (t=0; tm_pInterface->m_getNumVerticesOfFace(pContext, f); + if (verts!=3 && verts!=4) continue; + + + // I've decided to let degenerate triangles and group-with-anythings + // vary between left/right hand coordinate systems at the vertices. + // All healthy triangles on the other hand are built to always be either or. + + /*// force the coordinate system orientation to be uniform for every face. + // (this is already the case for good triangles but not for + // degenerate ones and those with bGroupWithAnything==true) + bool bOrient = psTspace[index].bOrient; + if (psTspace[index].iCounter == 0) // tspace was not derived from a group + { + // look for a space created in GenerateTSpaces() by iCounter>0 + bool bNotFound = true; + int i=1; + while (i 0) bNotFound=false; + else ++i; + } + if (!bNotFound) bOrient = psTspace[index+i].bOrient; + }*/ + + // set data + for (i=0; ivOs.x, pTSpace->vOs.y, pTSpace->vOs.z}; + float bitang[] = {pTSpace->vOt.x, pTSpace->vOt.y, pTSpace->vOt.z}; + if (pContext->m_pInterface->m_setTSpace!=NULL) + pContext->m_pInterface->m_setTSpace(pContext, tang, bitang, pTSpace->fMagS, pTSpace->fMagT, pTSpace->bOrient, f, i); + if (pContext->m_pInterface->m_setTSpaceBasic!=NULL) + pContext->m_pInterface->m_setTSpaceBasic(pContext, tang, pTSpace->bOrient==TTRUE ? 1.0f : (-1.0f), f, i); + + ++index; + } + } + + free(psTspace); + + + return TTRUE; +} + +/////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + +typedef struct { + float vert[3]; + int index; +} STmpVert; + +static const int g_iCells = 2048; + +#ifdef _MSC_VER +# define NOINLINE __declspec(noinline) +#else +# define NOINLINE __attribute__ ((noinline)) +#endif + +// it is IMPORTANT that this function is called to evaluate the hash since +// inlining could potentially reorder instructions and generate different +// results for the same effective input value fVal. +static NOINLINE int FindGridCell(const float fMin, const float fMax, const float fVal) +{ + const float fIndex = g_iCells * ((fVal-fMin)/(fMax-fMin)); + const int iIndex = (int)fIndex; + return iIndex < g_iCells ? (iIndex >= 0 ? iIndex : 0) : (g_iCells - 1); +} + +static void MergeVertsFast(int piTriList_in_and_out[], STmpVert pTmpVert[], const SMikkTSpaceContext * pContext, const int iL_in, const int iR_in); +static void MergeVertsSlow(int piTriList_in_and_out[], const SMikkTSpaceContext * pContext, const int pTable[], const int iEntries); +static void GenerateSharedVerticesIndexListSlow(int piTriList_in_and_out[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn); + +static void GenerateSharedVerticesIndexList(int piTriList_in_and_out[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn) +{ + + // Generate bounding box + int * piHashTable=NULL, * piHashCount=NULL, * piHashOffsets=NULL, * piHashCount2=NULL; + STmpVert * pTmpVert = NULL; + int i=0, iChannel=0, k=0, e=0; + int iMaxCount=0; + SVec3 vMin = GetPosition(pContext, 0), vMax = vMin, vDim; + float fMin, fMax; + for (i=1; i<(iNrTrianglesIn*3); i++) + { + const int index = piTriList_in_and_out[i]; + + const SVec3 vP = GetPosition(pContext, index); + if (vMin.x > vP.x) vMin.x = vP.x; + else if (vMax.x < vP.x) vMax.x = vP.x; + if (vMin.y > vP.y) vMin.y = vP.y; + else if (vMax.y < vP.y) vMax.y = vP.y; + if (vMin.z > vP.z) vMin.z = vP.z; + else if (vMax.z < vP.z) vMax.z = vP.z; + } + + vDim = vsub(vMax,vMin); + iChannel = 0; + fMin = vMin.x; fMax=vMax.x; + if (vDim.y>vDim.x && vDim.y>vDim.z) + { + iChannel=1; + fMin = vMin.y; + fMax = vMax.y; + } + else if (vDim.z>vDim.x) + { + iChannel=2; + fMin = vMin.z; + fMax = vMax.z; + } + + // make allocations + piHashTable = (int *) malloc(sizeof(int)*iNrTrianglesIn*3); + piHashCount = (int *) malloc(sizeof(int)*g_iCells); + piHashOffsets = (int *) malloc(sizeof(int)*g_iCells); + piHashCount2 = (int *) malloc(sizeof(int)*g_iCells); + + if (piHashTable==NULL || piHashCount==NULL || piHashOffsets==NULL || piHashCount2==NULL) + { + if (piHashTable!=NULL) free(piHashTable); + if (piHashCount!=NULL) free(piHashCount); + if (piHashOffsets!=NULL) free(piHashOffsets); + if (piHashCount2!=NULL) free(piHashCount2); + GenerateSharedVerticesIndexListSlow(piTriList_in_and_out, pContext, iNrTrianglesIn); + return; + } + memset(piHashCount, 0, sizeof(int)*g_iCells); + memset(piHashCount2, 0, sizeof(int)*g_iCells); + + // count amount of elements in each cell unit + for (i=0; i<(iNrTrianglesIn*3); i++) + { + const int index = piTriList_in_and_out[i]; + const SVec3 vP = GetPosition(pContext, index); + const float fVal = iChannel==0 ? vP.x : (iChannel==1 ? vP.y : vP.z); + const int iCell = FindGridCell(fMin, fMax, fVal); + ++piHashCount[iCell]; + } + + // evaluate start index of each cell. + piHashOffsets[0]=0; + for (k=1; kpTmpVert[l].vert[c]) fvMin[c]=pTmpVert[l].vert[c]; + if (fvMax[c]dx && dy>dz) channel=1; + else if (dz>dx) channel=2; + + fSep = 0.5f*(fvMax[channel]+fvMin[channel]); + + // stop if all vertices are NaNs + if (!isfinite(fSep)) + return; + + // terminate recursion when the separation/average value + // is no longer strictly between fMin and fMax values. + if (fSep>=fvMax[channel] || fSep<=fvMin[channel]) + { + // complete the weld + for (l=iL_in; l<=iR_in; l++) + { + int i = pTmpVert[l].index; + const int index = piTriList_in_and_out[i]; + const SVec3 vP = GetPosition(pContext, index); + const SVec3 vN = GetNormal(pContext, index); + const SVec3 vT = GetTexCoord(pContext, index); + + tbool bNotFound = TTRUE; + int l2=iL_in, i2rec=-1; + while (l20); // at least 2 entries + + // separate (by fSep) all points between iL_in and iR_in in pTmpVert[] + while (iL < iR) + { + tbool bReadyLeftSwap = TFALSE, bReadyRightSwap = TFALSE; + while ((!bReadyLeftSwap) && iL=iL_in && iL<=iR_in); + bReadyLeftSwap = !(pTmpVert[iL].vert[channel]=iL_in && iR<=iR_in); + bReadyRightSwap = pTmpVert[iR].vert[channel]m_pInterface->m_getNumFaces(pContext); f++) + { + const int verts = pContext->m_pInterface->m_getNumVerticesOfFace(pContext, f); + if (verts!=3 && verts!=4) continue; + + pTriInfos[iDstTriIndex].iOrgFaceNumber = f; + pTriInfos[iDstTriIndex].iTSpacesOffs = iTSpacesOffs; + + if (verts==3) + { + unsigned char * pVerts = pTriInfos[iDstTriIndex].vert_num; + pVerts[0]=0; pVerts[1]=1; pVerts[2]=2; + piTriList_out[iDstTriIndex*3+0] = MakeIndex(f, 0); + piTriList_out[iDstTriIndex*3+1] = MakeIndex(f, 1); + piTriList_out[iDstTriIndex*3+2] = MakeIndex(f, 2); + ++iDstTriIndex; // next + } + else + { + { + pTriInfos[iDstTriIndex+1].iOrgFaceNumber = f; + pTriInfos[iDstTriIndex+1].iTSpacesOffs = iTSpacesOffs; + } + + { + // need an order independent way to evaluate + // tspace on quads. This is done by splitting + // along the shortest diagonal. + const int i0 = MakeIndex(f, 0); + const int i1 = MakeIndex(f, 1); + const int i2 = MakeIndex(f, 2); + const int i3 = MakeIndex(f, 3); + const SVec3 T0 = GetTexCoord(pContext, i0); + const SVec3 T1 = GetTexCoord(pContext, i1); + const SVec3 T2 = GetTexCoord(pContext, i2); + const SVec3 T3 = GetTexCoord(pContext, i3); + const float distSQ_02 = LengthSquared(vsub(T2,T0)); + const float distSQ_13 = LengthSquared(vsub(T3,T1)); + tbool bQuadDiagIs_02; + if (distSQ_02m_pInterface->m_getPosition(pContext, pos, iF, iI); + res.x=pos[0]; res.y=pos[1]; res.z=pos[2]; + return res; +} + +static SVec3 GetNormal(const SMikkTSpaceContext * pContext, const int index) +{ + int iF, iI; + SVec3 res; float norm[3]; + IndexToData(&iF, &iI, index); + pContext->m_pInterface->m_getNormal(pContext, norm, iF, iI); + res.x=norm[0]; res.y=norm[1]; res.z=norm[2]; + return res; +} + +static SVec3 GetTexCoord(const SMikkTSpaceContext * pContext, const int index) +{ + int iF, iI; + SVec3 res; float texc[2]; + IndexToData(&iF, &iI, index); + pContext->m_pInterface->m_getTexCoord(pContext, texc, iF, iI); + res.x=texc[0]; res.y=texc[1]; res.z=1.0f; + return res; +} + +///////////////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////////////////////////////////////////////////////////////////////////////// + +typedef union { + struct + { + int i0, i1, f; + }; + int array[3]; +} SEdge; + +static void BuildNeighborsFast(STriInfo pTriInfos[], SEdge * pEdges, const int piTriListIn[], const int iNrTrianglesIn); +static void BuildNeighborsSlow(STriInfo pTriInfos[], const int piTriListIn[], const int iNrTrianglesIn); + +// returns the texture area times 2 +static float CalcTexArea(const SMikkTSpaceContext * pContext, const int indices[]) +{ + const SVec3 t1 = GetTexCoord(pContext, indices[0]); + const SVec3 t2 = GetTexCoord(pContext, indices[1]); + const SVec3 t3 = GetTexCoord(pContext, indices[2]); + + const float t21x = t2.x-t1.x; + const float t21y = t2.y-t1.y; + const float t31x = t3.x-t1.x; + const float t31y = t3.y-t1.y; + + const float fSignedAreaSTx2 = t21x*t31y - t21y*t31x; + + return fSignedAreaSTx2<0 ? (-fSignedAreaSTx2) : fSignedAreaSTx2; +} + +static void InitTriInfo(STriInfo pTriInfos[], const int piTriListIn[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn) +{ + int f=0, i=0, t=0; + // pTriInfos[f].iFlag is cleared in GenerateInitialVerticesIndexList() which is called before this function. + + // generate neighbor info list + for (f=0; f0 ? ORIENT_PRESERVING : 0); + + if ( NotZero(fSignedAreaSTx2) ) + { + const float fAbsArea = fabsf(fSignedAreaSTx2); + const float fLenOs = Length(vOs); + const float fLenOt = Length(vOt); + const float fS = (pTriInfos[f].iFlag&ORIENT_PRESERVING)==0 ? (-1.0f) : 1.0f; + if ( NotZero(fLenOs) ) pTriInfos[f].vOs = vscale(fS/fLenOs, vOs); + if ( NotZero(fLenOt) ) pTriInfos[f].vOt = vscale(fS/fLenOt, vOt); + + // evaluate magnitudes prior to normalization of vOs and vOt + pTriInfos[f].fMagS = fLenOs / fAbsArea; + pTriInfos[f].fMagT = fLenOt / fAbsArea; + + // if this is a good triangle + if ( NotZero(pTriInfos[f].fMagS) && NotZero(pTriInfos[f].fMagT)) + pTriInfos[f].iFlag &= (~GROUP_WITH_ANY); + } + } + + // force otherwise healthy quads to a fixed orientation + while (t<(iNrTrianglesIn-1)) + { + const int iFO_a = pTriInfos[t].iOrgFaceNumber; + const int iFO_b = pTriInfos[t+1].iOrgFaceNumber; + if (iFO_a==iFO_b) // this is a quad + { + const tbool bIsDeg_a = (pTriInfos[t].iFlag&MARK_DEGENERATE)!=0 ? TTRUE : TFALSE; + const tbool bIsDeg_b = (pTriInfos[t+1].iFlag&MARK_DEGENERATE)!=0 ? TTRUE : TFALSE; + + // bad triangles should already have been removed by + // DegenPrologue(), but just in case check bIsDeg_a and bIsDeg_a are false + if ((bIsDeg_a||bIsDeg_b)==TFALSE) + { + const tbool bOrientA = (pTriInfos[t].iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + const tbool bOrientB = (pTriInfos[t+1].iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + // if this happens the quad has extremely bad mapping!! + if (bOrientA!=bOrientB) + { + //printf("found quad with bad mapping\n"); + tbool bChooseOrientFirstTri = TFALSE; + if ((pTriInfos[t+1].iFlag&GROUP_WITH_ANY)!=0) bChooseOrientFirstTri = TTRUE; + else if ( CalcTexArea(pContext, &piTriListIn[t*3+0]) >= CalcTexArea(pContext, &piTriListIn[(t+1)*3+0]) ) + bChooseOrientFirstTri = TTRUE; + + // force match + { + const int t0 = bChooseOrientFirstTri ? t : (t+1); + const int t1 = bChooseOrientFirstTri ? (t+1) : t; + pTriInfos[t1].iFlag &= (~ORIENT_PRESERVING); // clear first + pTriInfos[t1].iFlag |= (pTriInfos[t0].iFlag&ORIENT_PRESERVING); // copy bit + } + } + } + t += 2; + } + else + ++t; + } + + // match up edge pairs + { + SEdge * pEdges = (SEdge *) malloc(sizeof(SEdge)*iNrTrianglesIn*3); + if (pEdges==NULL) + BuildNeighborsSlow(pTriInfos, piTriListIn, iNrTrianglesIn); + else + { + BuildNeighborsFast(pTriInfos, pEdges, piTriListIn, iNrTrianglesIn); + + free(pEdges); + } + } +} + +///////////////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////////////////////////////////////////////////////////////////////////////// + +static tbool AssignRecur(const int piTriListIn[], STriInfo psTriInfos[], const int iMyTriIndex, SGroup * pGroup); +static void AddTriToGroup(SGroup * pGroup, const int iTriIndex); + +static int Build4RuleGroups(STriInfo pTriInfos[], SGroup pGroups[], int piGroupTrianglesBuffer[], const int piTriListIn[], const int iNrTrianglesIn) +{ + const int iNrMaxGroups = iNrTrianglesIn*3; + int iNrActiveGroups = 0; + int iOffset = 0, f=0, i=0; + (void)iNrMaxGroups; /* quiet warnings in non debug mode */ + for (f=0; fiVertexRepresentitive = vert_index; + pTriInfos[f].AssignedGroup[i]->bOrientPreservering = (pTriInfos[f].iFlag&ORIENT_PRESERVING)!=0; + pTriInfos[f].AssignedGroup[i]->iNrFaces = 0; + pTriInfos[f].AssignedGroup[i]->pFaceIndices = &piGroupTrianglesBuffer[iOffset]; + ++iNrActiveGroups; + + AddTriToGroup(pTriInfos[f].AssignedGroup[i], f); + bOrPre = (pTriInfos[f].iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + neigh_indexL = pTriInfos[f].FaceNeighbors[i]; + neigh_indexR = pTriInfos[f].FaceNeighbors[i>0?(i-1):2]; + if (neigh_indexL>=0) // neighbor + { + const tbool bAnswer = + AssignRecur(piTriListIn, pTriInfos, neigh_indexL, + pTriInfos[f].AssignedGroup[i] ); + + const tbool bOrPre2 = (pTriInfos[neigh_indexL].iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + const tbool bDiff = bOrPre!=bOrPre2 ? TTRUE : TFALSE; + assert(bAnswer || bDiff); + (void)bAnswer, (void)bDiff; /* quiet warnings in non debug mode */ + } + if (neigh_indexR>=0) // neighbor + { + const tbool bAnswer = + AssignRecur(piTriListIn, pTriInfos, neigh_indexR, + pTriInfos[f].AssignedGroup[i] ); + + const tbool bOrPre2 = (pTriInfos[neigh_indexR].iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + const tbool bDiff = bOrPre!=bOrPre2 ? TTRUE : TFALSE; + assert(bAnswer || bDiff); + (void)bAnswer, (void)bDiff; /* quiet warnings in non debug mode */ + } + + // update offset + iOffset += pTriInfos[f].AssignedGroup[i]->iNrFaces; + // since the groups are disjoint a triangle can never + // belong to more than 3 groups. Subsequently something + // is completely screwed if this assertion ever hits. + assert(iOffset <= iNrMaxGroups); + } + } + } + + return iNrActiveGroups; +} + +static void AddTriToGroup(SGroup * pGroup, const int iTriIndex) +{ + pGroup->pFaceIndices[pGroup->iNrFaces] = iTriIndex; + ++pGroup->iNrFaces; +} + +static tbool AssignRecur(const int piTriListIn[], STriInfo psTriInfos[], + const int iMyTriIndex, SGroup * pGroup) +{ + STriInfo * pMyTriInfo = &psTriInfos[iMyTriIndex]; + + // track down vertex + const int iVertRep = pGroup->iVertexRepresentitive; + const int * pVerts = &piTriListIn[3*iMyTriIndex+0]; + int i=-1; + if (pVerts[0]==iVertRep) i=0; + else if (pVerts[1]==iVertRep) i=1; + else if (pVerts[2]==iVertRep) i=2; + assert(i>=0 && i<3); + + // early out + if (pMyTriInfo->AssignedGroup[i] == pGroup) return TTRUE; + else if (pMyTriInfo->AssignedGroup[i]!=NULL) return TFALSE; + if ((pMyTriInfo->iFlag&GROUP_WITH_ANY)!=0) + { + // first to group with a group-with-anything triangle + // determines it's orientation. + // This is the only existing order dependency in the code!! + if ( pMyTriInfo->AssignedGroup[0] == NULL && + pMyTriInfo->AssignedGroup[1] == NULL && + pMyTriInfo->AssignedGroup[2] == NULL ) + { + pMyTriInfo->iFlag &= (~ORIENT_PRESERVING); + pMyTriInfo->iFlag |= (pGroup->bOrientPreservering ? ORIENT_PRESERVING : 0); + } + } + { + const tbool bOrient = (pMyTriInfo->iFlag&ORIENT_PRESERVING)!=0 ? TTRUE : TFALSE; + if (bOrient != pGroup->bOrientPreservering) return TFALSE; + } + + AddTriToGroup(pGroup, iMyTriIndex); + pMyTriInfo->AssignedGroup[i] = pGroup; + + { + const int neigh_indexL = pMyTriInfo->FaceNeighbors[i]; + const int neigh_indexR = pMyTriInfo->FaceNeighbors[i>0?(i-1):2]; + if (neigh_indexL>=0) + AssignRecur(piTriListIn, psTriInfos, neigh_indexL, pGroup); + if (neigh_indexR>=0) + AssignRecur(piTriListIn, psTriInfos, neigh_indexR, pGroup); + } + + + + return TTRUE; +} + +///////////////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////////////////////////////////////////////////////////////////////////////// + +static tbool CompareSubGroups(const SSubGroup * pg1, const SSubGroup * pg2); +static void QuickSort(int* pSortBuffer, int iLeft, int iRight, unsigned int uSeed); +static STSpace EvalTspace(int face_indices[], const int iFaces, const int piTriListIn[], const STriInfo pTriInfos[], const SMikkTSpaceContext * pContext, const int iVertexRepresentitive); + +static tbool GenerateTSpaces(STSpace psTspace[], const STriInfo pTriInfos[], const SGroup pGroups[], + const int iNrActiveGroups, const int piTriListIn[], const float fThresCos, + const SMikkTSpaceContext * pContext) +{ + STSpace * pSubGroupTspace = NULL; + SSubGroup * pUniSubGroups = NULL; + int * pTmpMembers = NULL; + int iMaxNrFaces=0, iUniqueTspaces=0, g=0, i=0; + for (g=0; giNrFaces; i++) // triangles + { + const int f = pGroup->pFaceIndices[i]; // triangle number + int index=-1, iVertIndex=-1, iOF_1=-1, iMembers=0, j=0, l=0; + SSubGroup tmp_group; + tbool bFound; + SVec3 n, vOs, vOt; + if (pTriInfos[f].AssignedGroup[0]==pGroup) index=0; + else if (pTriInfos[f].AssignedGroup[1]==pGroup) index=1; + else if (pTriInfos[f].AssignedGroup[2]==pGroup) index=2; + assert(index>=0 && index<3); + + iVertIndex = piTriListIn[f*3+index]; + assert(iVertIndex==pGroup->iVertexRepresentitive); + + // is normalized already + n = GetNormal(pContext, iVertIndex); + + // project + vOs = vsub(pTriInfos[f].vOs, vscale(vdot(n,pTriInfos[f].vOs), n)); + vOt = vsub(pTriInfos[f].vOt, vscale(vdot(n,pTriInfos[f].vOt), n)); + if ( VNotZero(vOs) ) vOs = Normalize(vOs); + if ( VNotZero(vOt) ) vOt = Normalize(vOt); + + // original face number + iOF_1 = pTriInfos[f].iOrgFaceNumber; + + iMembers = 0; + for (j=0; jiNrFaces; j++) + { + const int t = pGroup->pFaceIndices[j]; // triangle number + const int iOF_2 = pTriInfos[t].iOrgFaceNumber; + + // project + SVec3 vOs2 = vsub(pTriInfos[t].vOs, vscale(vdot(n,pTriInfos[t].vOs), n)); + SVec3 vOt2 = vsub(pTriInfos[t].vOt, vscale(vdot(n,pTriInfos[t].vOt), n)); + if ( VNotZero(vOs2) ) vOs2 = Normalize(vOs2); + if ( VNotZero(vOt2) ) vOt2 = Normalize(vOt2); + + { + const tbool bAny = ( (pTriInfos[f].iFlag | pTriInfos[t].iFlag) & GROUP_WITH_ANY )!=0 ? TTRUE : TFALSE; + // make sure triangles which belong to the same quad are joined. + const tbool bSameOrgFace = iOF_1==iOF_2 ? TTRUE : TFALSE; + + const float fCosS = vdot(vOs,vOs2); + const float fCosT = vdot(vOt,vOt2); + + assert(f!=t || bSameOrgFace); // sanity check + if (bAny || bSameOrgFace || (fCosS>fThresCos && fCosT>fThresCos)) + pTmpMembers[iMembers++] = t; + } + } + + // sort pTmpMembers + tmp_group.iNrFaces = iMembers; + tmp_group.pTriMembers = pTmpMembers; + if (iMembers>1) + { + unsigned int uSeed = INTERNAL_RND_SORT_SEED; // could replace with a random seed? + QuickSort(pTmpMembers, 0, iMembers-1, uSeed); + } + + // look for an existing match + bFound = TFALSE; + l=0; + while (liVertexRepresentitive); + ++iUniqueSubGroups; + } + + // output tspace + { + const int iOffs = pTriInfos[f].iTSpacesOffs; + const int iVert = pTriInfos[f].vert_num[index]; + STSpace * pTS_out = &psTspace[iOffs+iVert]; + assert(pTS_out->iCounter<2); + assert(((pTriInfos[f].iFlag&ORIENT_PRESERVING)!=0) == pGroup->bOrientPreservering); + if (pTS_out->iCounter==1) + { + *pTS_out = AvgTSpace(pTS_out, &pSubGroupTspace[l]); + pTS_out->iCounter = 2; // update counter + pTS_out->bOrient = pGroup->bOrientPreservering; + } + else + { + assert(pTS_out->iCounter==0); + *pTS_out = pSubGroupTspace[l]; + pTS_out->iCounter = 1; // update counter + pTS_out->bOrient = pGroup->bOrientPreservering; + } + } + } + + // clean up and offset iUniqueTspaces + for (s=0; s=0 && i<3); + + // project + index = piTriListIn[3*f+i]; + n = GetNormal(pContext, index); + vOs = vsub(pTriInfos[f].vOs, vscale(vdot(n,pTriInfos[f].vOs), n)); + vOt = vsub(pTriInfos[f].vOt, vscale(vdot(n,pTriInfos[f].vOt), n)); + if ( VNotZero(vOs) ) vOs = Normalize(vOs); + if ( VNotZero(vOt) ) vOt = Normalize(vOt); + + i2 = piTriListIn[3*f + (i<2?(i+1):0)]; + i1 = piTriListIn[3*f + i]; + i0 = piTriListIn[3*f + (i>0?(i-1):2)]; + + p0 = GetPosition(pContext, i0); + p1 = GetPosition(pContext, i1); + p2 = GetPosition(pContext, i2); + v1 = vsub(p0,p1); + v2 = vsub(p2,p1); + + // project + v1 = vsub(v1, vscale(vdot(n,v1),n)); if ( VNotZero(v1) ) v1 = Normalize(v1); + v2 = vsub(v2, vscale(vdot(n,v2),n)); if ( VNotZero(v2) ) v2 = Normalize(v2); + + // weight contribution by the angle + // between the two edge vectors + fCos = vdot(v1,v2); fCos=fCos>1?1:(fCos<(-1) ? (-1) : fCos); + fAngle = (float) acos(fCos); + fMagS = pTriInfos[f].fMagS; + fMagT = pTriInfos[f].fMagT; + + res.vOs=vadd(res.vOs, vscale(fAngle,vOs)); + res.vOt=vadd(res.vOt,vscale(fAngle,vOt)); + res.fMagS+=(fAngle*fMagS); + res.fMagT+=(fAngle*fMagT); + fAngleSum += fAngle; + } + } + + // normalize + if ( VNotZero(res.vOs) ) res.vOs = Normalize(res.vOs); + if ( VNotZero(res.vOt) ) res.vOt = Normalize(res.vOt); + if (fAngleSum>0) + { + res.fMagS /= fAngleSum; + res.fMagT /= fAngleSum; + } + + return res; +} + +static tbool CompareSubGroups(const SSubGroup * pg1, const SSubGroup * pg2) +{ + tbool bStillSame=TTRUE; + int i=0; + if (pg1->iNrFaces!=pg2->iNrFaces) return TFALSE; + while (iiNrFaces && bStillSame) + { + bStillSame = pg1->pTriMembers[i]==pg2->pTriMembers[i] ? TTRUE : TFALSE; + if (bStillSame) ++i; + } + return bStillSame; +} + +static void QuickSort(int* pSortBuffer, int iLeft, int iRight, unsigned int uSeed) +{ + int iL, iR, n, index, iMid, iTmp; + + // Random + unsigned int t=uSeed&31; + t=(uSeed<>(32-t)); + uSeed=uSeed+t+3; + // Random end + + iL=iLeft; iR=iRight; + n = (iR-iL)+1; + assert(n>=0); + index = (int) (uSeed%n); + + iMid=pSortBuffer[index + iL]; + + + do + { + while (pSortBuffer[iL] < iMid) + ++iL; + while (pSortBuffer[iR] > iMid) + --iR; + + if (iL <= iR) + { + iTmp = pSortBuffer[iL]; + pSortBuffer[iL] = pSortBuffer[iR]; + pSortBuffer[iR] = iTmp; + ++iL; --iR; + } + } + while (iL <= iR); + + if (iLeft < iR) + QuickSort(pSortBuffer, iLeft, iR, uSeed); + if (iL < iRight) + QuickSort(pSortBuffer, iL, iRight, uSeed); +} + +///////////////////////////////////////////////////////////////////////////////////////////// +///////////////////////////////////////////////////////////////////////////////////////////// + +static void QuickSortEdges(SEdge * pSortBuffer, int iLeft, int iRight, const int channel, unsigned int uSeed); +static void GetEdge(int * i0_out, int * i1_out, int * edgenum_out, const int indices[], const int i0_in, const int i1_in); + +static void BuildNeighborsFast(STriInfo pTriInfos[], SEdge * pEdges, const int piTriListIn[], const int iNrTrianglesIn) +{ + // build array of edges + unsigned int uSeed = INTERNAL_RND_SORT_SEED; // could replace with a random seed? + int iEntries=0, iCurStartIndex=-1, f=0, i=0; + for (f=0; f pSortBuffer[iRight].array[channel]) + { + sTmp = pSortBuffer[iLeft]; + pSortBuffer[iLeft] = pSortBuffer[iRight]; + pSortBuffer[iRight] = sTmp; + } + return; + } + + // Random + t=uSeed&31; + t=(uSeed<>(32-t)); + uSeed=uSeed+t+3; + // Random end + + iL = iLeft; + iR = iRight; + n = (iR-iL)+1; + assert(n>=0); + index = (int) (uSeed%n); + + iMid=pSortBuffer[index + iL].array[channel]; + + do + { + while (pSortBuffer[iL].array[channel] < iMid) + ++iL; + while (pSortBuffer[iR].array[channel] > iMid) + --iR; + + if (iL <= iR) + { + sTmp = pSortBuffer[iL]; + pSortBuffer[iL] = pSortBuffer[iR]; + pSortBuffer[iR] = sTmp; + ++iL; --iR; + } + } + while (iL <= iR); + + if (iLeft < iR) + QuickSortEdges(pSortBuffer, iLeft, iR, channel, uSeed); + if (iL < iRight) + QuickSortEdges(pSortBuffer, iL, iRight, channel, uSeed); +} + +// resolve ordering and edge number +static void GetEdge(int * i0_out, int * i1_out, int * edgenum_out, const int indices[], const int i0_in, const int i1_in) +{ + *edgenum_out = -1; + + // test if first index is on the edge + if (indices[0]==i0_in || indices[0]==i1_in) + { + // test if second index is on the edge + if (indices[1]==i0_in || indices[1]==i1_in) + { + edgenum_out[0]=0; // first edge + i0_out[0]=indices[0]; + i1_out[0]=indices[1]; + } + else + { + edgenum_out[0]=2; // third edge + i0_out[0]=indices[2]; + i1_out[0]=indices[0]; + } + } + else + { + // only second and third index is on the edge + edgenum_out[0]=1; // second edge + i0_out[0]=indices[1]; + i1_out[0]=indices[2]; + } +} + + +///////////////////////////////////////////////////////////////////////////////////////////// +/////////////////////////////////// Degenerate triangles //////////////////////////////////// + +static void DegenPrologue(STriInfo pTriInfos[], int piTriList_out[], const int iNrTrianglesIn, const int iTotTris) +{ + int iNextGoodTriangleSearchIndex=-1; + tbool bStillFindingGoodOnes; + + // locate quads with only one good triangle + int t=0; + while (t<(iTotTris-1)) + { + const int iFO_a = pTriInfos[t].iOrgFaceNumber; + const int iFO_b = pTriInfos[t+1].iOrgFaceNumber; + if (iFO_a==iFO_b) // this is a quad + { + const tbool bIsDeg_a = (pTriInfos[t].iFlag&MARK_DEGENERATE)!=0 ? TTRUE : TFALSE; + const tbool bIsDeg_b = (pTriInfos[t+1].iFlag&MARK_DEGENERATE)!=0 ? TTRUE : TFALSE; + if ((bIsDeg_a^bIsDeg_b)!=0) + { + pTriInfos[t].iFlag |= QUAD_ONE_DEGEN_TRI; + pTriInfos[t+1].iFlag |= QUAD_ONE_DEGEN_TRI; + } + t += 2; + } + else + ++t; + } + + // reorder list so all degen triangles are moved to the back + // without reordering the good triangles + iNextGoodTriangleSearchIndex = 1; + t=0; + bStillFindingGoodOnes = TTRUE; + while (t (t+1)); + + // swap triangle t0 and t1 + if (!bJustADegenerate) + { + int i=0; + for (i=0; i<3; i++) + { + const int index = piTriList_out[t0*3+i]; + piTriList_out[t0*3+i] = piTriList_out[t1*3+i]; + piTriList_out[t1*3+i] = index; + } + { + const STriInfo tri_info = pTriInfos[t0]; + pTriInfos[t0] = pTriInfos[t1]; + pTriInfos[t1] = tri_info; + } + } + else + bStillFindingGoodOnes = TFALSE; // this is not supposed to happen + } + + if (bStillFindingGoodOnes) ++t; + } + + assert(bStillFindingGoodOnes); // code will still work. + assert(iNrTrianglesIn == t); +} + +static void DegenEpilogue(STSpace psTspace[], STriInfo pTriInfos[], int piTriListIn[], const SMikkTSpaceContext * pContext, const int iNrTrianglesIn, const int iTotTris) +{ + int t=0, i=0; + // deal with degenerate triangles + // punishment for degenerate triangles is O(N^2) + for (t=iNrTrianglesIn; t http://image.diku.dk/projects/media/morten.mikkelsen.08.pdf + * Note that though the tangent spaces at the vertices are generated in an order-independent way, + * by this implementation, the interpolated tangent space is still affected by which diagonal is + * chosen to split each quad. A sensible solution is to have your tools pipeline always + * split quads by the shortest diagonal. This choice is order-independent and works with mirroring. + * If these have the same length then compare the diagonals defined by the texture coordinates. + * XNormal which is a tool for baking normal maps allows you to write your own tangent space plugin + * and also quad triangulator plugin. + */ + + +typedef int tbool; +typedef struct SMikkTSpaceContext SMikkTSpaceContext; + +typedef struct { + // Returns the number of faces (triangles/quads) on the mesh to be processed. + int (*m_getNumFaces)(const SMikkTSpaceContext * pContext); + + // Returns the number of vertices on face number iFace + // iFace is a number in the range {0, 1, ..., getNumFaces()-1} + int (*m_getNumVerticesOfFace)(const SMikkTSpaceContext * pContext, const int iFace); + + // returns the position/normal/texcoord of the referenced face of vertex number iVert. + // iVert is in the range {0,1,2} for triangles and {0,1,2,3} for quads. + void (*m_getPosition)(const SMikkTSpaceContext * pContext, float fvPosOut[], const int iFace, const int iVert); + void (*m_getNormal)(const SMikkTSpaceContext * pContext, float fvNormOut[], const int iFace, const int iVert); + void (*m_getTexCoord)(const SMikkTSpaceContext * pContext, float fvTexcOut[], const int iFace, const int iVert); + + // either (or both) of the two setTSpace callbacks can be set. + // The call-back m_setTSpaceBasic() is sufficient for basic normal mapping. + + // This function is used to return the tangent and fSign to the application. + // fvTangent is a unit length vector. + // For normal maps it is sufficient to use the following simplified version of the bitangent which is generated at pixel/vertex level. + // bitangent = fSign * cross(vN, tangent); + // Note that the results are returned unindexed. It is possible to generate a new index list + // But averaging/overwriting tangent spaces by using an already existing index list WILL produce INCRORRECT results. + // DO NOT! use an already existing index list. + void (*m_setTSpaceBasic)(const SMikkTSpaceContext * pContext, const float fvTangent[], const float fSign, const int iFace, const int iVert); + + // This function is used to return tangent space results to the application. + // fvTangent and fvBiTangent are unit length vectors and fMagS and fMagT are their + // true magnitudes which can be used for relief mapping effects. + // fvBiTangent is the "real" bitangent and thus may not be perpendicular to fvTangent. + // However, both are perpendicular to the vertex normal. + // For normal maps it is sufficient to use the following simplified version of the bitangent which is generated at pixel/vertex level. + // fSign = bIsOrientationPreserving ? 1.0f : (-1.0f); + // bitangent = fSign * cross(vN, tangent); + // Note that the results are returned unindexed. It is possible to generate a new index list + // But averaging/overwriting tangent spaces by using an already existing index list WILL produce INCRORRECT results. + // DO NOT! use an already existing index list. + void (*m_setTSpace)(const SMikkTSpaceContext * pContext, const float fvTangent[], const float fvBiTangent[], const float fMagS, const float fMagT, + const tbool bIsOrientationPreserving, const int iFace, const int iVert); +} SMikkTSpaceInterface; + +struct SMikkTSpaceContext +{ + SMikkTSpaceInterface * m_pInterface; // initialized with callback functions + void * m_pUserData; // pointer to client side mesh data etc. (passed as the first parameter with every interface call) +}; + +// these are both thread safe! +tbool genTangSpaceDefault(const SMikkTSpaceContext * pContext); // Default (recommended) fAngularThreshold is 180 degrees (which means threshold disabled) +tbool genTangSpace(const SMikkTSpaceContext * pContext, const float fAngularThreshold); + + +// To avoid visual errors (distortions/unwanted hard edges in lighting), when using sampled normal maps, the +// normal map sampler must use the exact inverse of the pixel shader transformation. +// The most efficient transformation we can possibly do in the pixel shader is +// achieved by using, directly, the "unnormalized" interpolated tangent, bitangent and vertex normal: vT, vB and vN. +// pixel shader (fast transform out) +// vNout = normalize( vNt.x * vT + vNt.y * vB + vNt.z * vN ); +// where vNt is the tangent space normal. The normal map sampler must likewise use the +// interpolated and "unnormalized" tangent, bitangent and vertex normal to be compliant with the pixel shader. +// sampler does (exact inverse of pixel shader): +// float3 row0 = cross(vB, vN); +// float3 row1 = cross(vN, vT); +// float3 row2 = cross(vT, vB); +// float fSign = dot(vT, row0)<0 ? -1 : 1; +// vNt = normalize( fSign * float3(dot(vNout,row0), dot(vNout,row1), dot(vNout,row2)) ); +// where vNout is the sampled normal in some chosen 3D space. +// +// Should you choose to reconstruct the bitangent in the pixel shader instead +// of the vertex shader, as explained earlier, then be sure to do this in the normal map sampler also. +// Finally, beware of quad triangulations. If the normal map sampler doesn't use the same triangulation of +// quads as your renderer then problems will occur since the interpolated tangent spaces will differ +// eventhough the vertex level tangent spaces match. This can be solved either by triangulating before +// sampling/exporting or by using the order-independent choice of diagonal for splitting quads suggested earlier. +// However, this must be used both by the sampler and your tools/rendering pipeline. + +#ifdef __cplusplus +} +#endif + +#endif diff --git a/src/external/tinygltf/LICENSE b/src/external/tinygltf/LICENSE new file mode 100644 index 00000000..34398adf --- /dev/null +++ b/src/external/tinygltf/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2017 Syoyo Fujita, Aurélien Chatelain and many contributors + +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. diff --git a/src/external/tinygltf/README.md b/src/external/tinygltf/README.md new file mode 100644 index 00000000..557734a1 --- /dev/null +++ b/src/external/tinygltf/README.md @@ -0,0 +1,267 @@ +# Header only C++ tiny glTF library(loader/saver). + +`TinyGLTF` is a header only C++11 glTF 2.0 https://github.com/KhronosGroup/glTF library. + +`TinyGLTF` uses Niels Lohmann's json library (https://github.com/nlohmann/json), so now it requires C++11 compiler. +(Also, you can use RadpidJSON as an JSON backend) +If you are looking for old, C++03 version, please use `devel-picojson` branch (but not maintained anymore). + +## Status + +Currently TinyGLTF is stable and maintenance mode. No drastic changes and feature additions planned. + + - v2.8.0 Add URICallbacks for custom URI handling in Buffer and Image. PR#397 + - v2.7.0 Change WriteImageDataFunction user callback function signature. PR#393 + - v2.6.0 Support serializing sparse accessor(Thanks to @fynv). + - v2.5.0 Add SetPreserveImageChannels() option to load image data as is. + - v2.4.0 Experimental RapidJSON support. Experimental C++14 support(C++14 may give better performance) + - v2.3.0 Modified Material representation according to glTF 2.0 schema(and introduced TextureInfo class) + - v2.2.0 release(Support loading 16bit PNG. Sparse accessor support) + - v2.1.0 release(Draco decoding support) + - v2.0.0 release(22 Aug, 2018)! + +### Branches + +* `sajson` : Use sajson to parse JSON. Parsing only but faster compile time(2x reduction compared to json.hpp and RapidJson), but not well maintained. + +## Builds + +[![Build Status](https://travis-ci.org/syoyo/tinygltf.svg?branch=devel)](https://travis-ci.org/syoyo/tinygltf) + +[![Build status](https://ci.appveyor.com/api/projects/status/warngenu9wjjhlm8?svg=true)](https://ci.appveyor.com/project/syoyo/tinygltf) + +![C/C++ CI](https://github.com/syoyo/tinygltf/workflows/C/C++%20CI/badge.svg) + +## Features + +Probably mostly feature-complete. Last missing feature is Draco encoding: https://github.com/syoyo/tinygltf/issues/207 + +* Written in portable C++. C++-11 with STL dependency only. + * [x] macOS + clang(LLVM) + * [x] iOS + clang + * [x] Linux + gcc/clang + * [x] Windows + MinGW + * [x] Windows + Visual Studio 2015 Update 3 or later. + * Visual Studio 2013 is not supported since they have limited C++11 support and failed to compile `json.hpp`. + * [x] Android NDK + * [x] Android + CrystaX(NDK drop-in replacement) GCC + * [x] Web using Emscripten(LLVM) +* Moderate parsing time and memory consumption. +* glTF specification v2.0.0 + * [x] ASCII glTF + * [x] Load + * [x] Save + * [x] Binary glTF(GLB) + * [x] Load + * [x] Save(.bin embedded .glb) +* Buffers + * [x] Parse BASE64 encoded embedded buffer data(DataURI). + * [x] Load `.bin` file. +* Image(Using stb_image) + * [x] Parse BASE64 encoded embedded image data(DataURI). + * [x] Load external image file. + * [x] Load PNG(8bit and 16bit) + * [x] Load JPEG(8bit only) + * [x] Load BMP + * [x] Load GIF + * [x] Custom Image decoder callback(e.g. for decoding OpenEXR image) +* Morph traget + * [x] Sparse accessor +* Load glTF from memory +* Custom callback handler + * [x] Image load + * [x] Image save +* Extensions + * [x] Draco mesh decoding + * [ ] Draco mesh encoding + +## Note on extension property + +In extension(`ExtensionMap`), JSON number value is parsed as int or float(number) and stored as `tinygltf::Value` object. If you want a floating point value from `tinygltf::Value`, use `GetNumberAsDouble()` method. + +`IsNumber()` returns true if the underlying value is an int value or a floating point value. + +## Examples + +* [glview](examples/glview) : Simple glTF geometry viewer. +* [validator](examples/validator) : Simple glTF validator with JSON schema. +* [basic](examples/basic) : Basic glTF viewer with texturing support. +* [build-gltf](examples/build-gltf) : Build simple glTF scene from a scratch. + +### WASI/WASM build + +Users who want to run TinyGLTF securely and safely(e.g. need to handle malcious glTF file to serve online glTF conver), +I recommend to build TinyGLTF for WASM target. +WASI build example is located in [wasm](wasm) . + +## Projects using TinyGLTF + +* px_render Single header C++ Libraries for Thread Scheduling, Rendering, and so on... https://github.com/pplux/px +* Physical based rendering with Vulkan using glTF 2.0 models https://github.com/SaschaWillems/Vulkan-glTF-PBR +* GLTF loader plugin for OGRE 2.1. Support for PBR materials via HLMS/PBS https://github.com/Ybalrid/Ogre_glTF +* [TinyGltfImporter](http://doc.magnum.graphics/magnum/classMagnum_1_1Trade_1_1TinyGltfImporter.html) plugin for [Magnum](https://github.com/mosra/magnum), a lightweight and modular C++11/C++14 graphics middleware for games and data visualization. +* [Diligent Engine](https://github.com/DiligentGraphics/DiligentEngine) - A modern cross-platform low-level graphics library and rendering framework +* Lighthouse 2: a rendering framework for real-time ray tracing / path tracing experiments. https://github.com/jbikker/lighthouse2 +* [QuickLook GLTF](https://github.com/toshiks/glTF-quicklook) - quicklook plugin for macos. Also SceneKit wrapper for tinygltf. +* [GlslViewer](https://github.com/patriciogonzalezvivo/glslViewer) - live GLSL coding for MacOS and Linux +* [Vulkan-Samples](https://github.com/KhronosGroup/Vulkan-Samples) - The Vulkan Samples is collection of resources to help you develop optimized Vulkan applications. +* [TDME2](https://github.com/andreasdr/tdme2) - TDME2 - ThreeDeeMiniEngine2 is a lightweight 3D engine including tools suited for 3D game development using C++11 +* [SanityEngine](https://github.com/DethRaid/SanityEngine) - A C++/D3D12 renderer focused on the personal and professional development of its developer +* [Open3D](http://www.open3d.org/) - A Modern Library for 3D Data Processing +* [Supernova Engine](https://github.com/supernovaengine/supernova) - Game engine for 2D and 3D projects with Lua or C++ in data oriented design. +* Your projects here! (Please send PR) + +## TODOs + +* [ ] Robust URI decoding/encoding. https://github.com/syoyo/tinygltf/issues/369 +* [ ] Mesh Compression/decompression(Open3DGC, etc) + * [x] Load Draco compressed mesh + * [ ] Save Draco compressed mesh + * [ ] Open3DGC? +* [x] Support `extensions` and `extras` property +* [ ] HDR image? + * [ ] OpenEXR extension through TinyEXR. +* [ ] 16bit PNG support in Serialization +* [ ] Write example and tests for `animation` and `skin` + +### Optional + +* [ ] Write C++ code generator which emits C++ code from JSON schema for robust parsing? + +## Licenses + +TinyGLTF is licensed under MIT license. + +TinyGLTF uses the following third party libraries. + +* json.hpp : Copyright (c) 2013-2017 Niels Lohmann. MIT license. +* base64 : Copyright (C) 2004-2008 René Nyffenegger +* stb_image.h : v2.08 - public domain image loader - [Github link](https://github.com/nothings/stb/blob/master/stb_image.h) +* stb_image_write.h : v1.09 - public domain image writer - [Github link](https://github.com/nothings/stb/blob/master/stb_image_write.h) + + +## Build and example + +Copy `stb_image.h`, `stb_image_write.h`, `json.hpp` and `tiny_gltf.h` to your project. + +### Loading glTF 2.0 model + +```c++ +// Define these only in *one* .cc file. +#define TINYGLTF_IMPLEMENTATION +#define STB_IMAGE_IMPLEMENTATION +#define STB_IMAGE_WRITE_IMPLEMENTATION +// #define TINYGLTF_NOEXCEPTION // optional. disable exception handling. +#include "tiny_gltf.h" + +using namespace tinygltf; + +Model model; +TinyGLTF loader; +std::string err; +std::string warn; + +bool ret = loader.LoadASCIIFromFile(&model, &err, &warn, argv[1]); +//bool ret = loader.LoadBinaryFromFile(&model, &err, &warn, argv[1]); // for binary glTF(.glb) + +if (!warn.empty()) { + printf("Warn: %s\n", warn.c_str()); +} + +if (!err.empty()) { + printf("Err: %s\n", err.c_str()); +} + +if (!ret) { + printf("Failed to parse glTF\n"); + return -1; +} +``` + +#### Loader options + +* `TinyGLTF::SetPreserveimageChannels(bool onoff)`. `true` to preserve image channels as stored in image file for loaded image. `false` by default for backward compatibility(image channels are widen to `RGBA` 4 channels). Effective only when using builtin image loader(STB image loader). + +## Compile options + +* `TINYGLTF_NOEXCEPTION` : Disable C++ exception in JSON parsing. You can use `-fno-exceptions` or by defining the symbol `JSON_NOEXCEPTION` and `TINYGLTF_NOEXCEPTION` to fully remove C++ exception codes when compiling TinyGLTF. +* `TINYGLTF_NO_STB_IMAGE` : Do not load images with stb_image. Instead use `TinyGLTF::SetImageLoader(LoadimageDataFunction LoadImageData, void *user_data)` to set a callback for loading images. +* `TINYGLTF_NO_STB_IMAGE_WRITE` : Do not write images with stb_image_write. Instead use `TinyGLTF::SetImageWriter(WriteimageDataFunction WriteImageData, void *user_data)` to set a callback for writing images. +* `TINYGLTF_NO_EXTERNAL_IMAGE` : Do not try to load external image file. This option would be helpful if you do not want to load image files during glTF parsing. +* `TINYGLTF_ANDROID_LOAD_FROM_ASSETS`: Load all files from packaged app assets instead of the regular file system. **Note:** You must pass a valid asset manager from your android app to `tinygltf::asset_manager` beforehand. +* `TINYGLTF_ENABLE_DRACO`: Enable Draco compression. User must provide include path and link correspnding libraries in your project file. +* `TINYGLTF_NO_INCLUDE_JSON `: Disable including `json.hpp` from within `tiny_gltf.h` because it has been already included before or you want to include it using custom path before including `tiny_gltf.h`. +* `TINYGLTF_NO_INCLUDE_RAPIDJSON `: Disable including RapidJson's header files from within `tiny_gltf.h` because it has been already included before or you want to include it using custom path before including `tiny_gltf.h`. +* `TINYGLTF_NO_INCLUDE_STB_IMAGE `: Disable including `stb_image.h` from within `tiny_gltf.h` because it has been already included before or you want to include it using custom path before including `tiny_gltf.h`. +* `TINYGLTF_NO_INCLUDE_STB_IMAGE_WRITE `: Disable including `stb_image_write.h` from within `tiny_gltf.h` because it has been already included before or you want to include it using custom path before including `tiny_gltf.h`. +* `TINYGLTF_USE_RAPIDJSON` : Use RapidJSON as a JSON parser/serializer. RapidJSON files are not included in TinyGLTF repo. Please set an include path to RapidJSON if you enable this feature. +* `TINYGLTF_USE_CPP14` : Use C++14 feature(requires C++14 compiler). This may give better performance than C++11. + +## CMake options + +You can add tinygltf using `add_subdirectory` feature. +If you add tinygltf to your project using `add_subdirectory`, it would be better to set `TINYGLTF_HEADER_ONLY` on(just add an include path to tinygltf) and `TINYGLTF_INSTALL` off(Which does not install tinygltf files). + +``` +// Your project's CMakeLists.txt +... + +set(TINYGLTF_HEADER_ONLY ON CACHE INTERNAL "" FORCE) +set(TINYGLTF_INSTALL OFF CACHE INTERNAL "" FORCE) +add_subdirectory(/path/to/tinygltf) +``` + + +### Saving gltTF 2.0 model + +* Buffers. + * [x] To file + * [x] Embedded + * [ ] Draco compressed? +* [x] Images + * [x] To file + * [x] Embedded +* Binary(.glb) + * [x] .bin embedded single .glb + * [ ] External .bin + +## Running tests. + +### glTF parsing test + +#### Setup + +Python required. +Git clone https://github.com/KhronosGroup/glTF-Sample-Models to your local dir. + +#### Run parsing test + +After building `loader_example`, edit `test_runner.py`, then, + +```bash +$ python test_runner.py +``` + +### Unit tests + +```bash +$ cd tests +$ make +$ ./tester +$ ./tester_noexcept +``` + +### Fuzzing tests + +See `tests/fuzzer` for details. + +After running fuzzer on Ryzen9 3950X a week, at least `LoadASCIIFromString` looks safe except for out-of-memory error in Fuzzer. +We may be better to introduce bounded memory size checking when parsing glTF data. + +## Third party licenses + +* json.hpp : Licensed under the MIT License . Copyright (c) 2013-2017 Niels Lohmann . +* stb_image : Public domain. +* catch : Copyright (c) 2012 Two Blue Cubes Ltd. All rights reserved. Distributed under the Boost Software License, Version 1.0. +* RapidJSON : Copyright (C) 2015 THL A29 Limited, a Tencent company, and Milo Yip. All rights reserved. http://rapidjson.org/ +* dlib(uridecode, uriencode) : Copyright (C) 2003 Davis E. King Boost Software License 1.0. http://dlib.net/dlib/server/server_http.cpp.html diff --git a/src/external/tinygltf/json.hpp b/src/external/tinygltf/json.hpp new file mode 100644 index 00000000..87475ab3 --- /dev/null +++ b/src/external/tinygltf/json.hpp @@ -0,0 +1,26753 @@ +/* + __ _____ _____ _____ + __| | __| | | | JSON for Modern C++ +| | |__ | | | | | | version 3.10.4 +|_____|_____|_____|_|___| https://github.com/nlohmann/json + +Licensed under the MIT License . +SPDX-License-Identifier: MIT +Copyright (c) 2013-2019 Niels Lohmann . + +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. +*/ + +#ifndef INCLUDE_NLOHMANN_JSON_HPP_ +#define INCLUDE_NLOHMANN_JSON_HPP_ + +#define NLOHMANN_JSON_VERSION_MAJOR 3 +#define NLOHMANN_JSON_VERSION_MINOR 10 +#define NLOHMANN_JSON_VERSION_PATCH 4 + +#include // all_of, find, for_each +#include // nullptr_t, ptrdiff_t, size_t +#include // hash, less +#include // initializer_list +#ifndef JSON_NO_IO + #include // istream, ostream +#endif // JSON_NO_IO +#include // random_access_iterator_tag +#include // unique_ptr +#include // accumulate +#include // string, stoi, to_string +#include // declval, forward, move, pair, swap +#include // vector + +// #include + + +#include +#include + +// #include + + +#include // transform +#include // array +#include // forward_list +#include // inserter, front_inserter, end +#include // map +#include // string +#include // tuple, make_tuple +#include // is_arithmetic, is_same, is_enum, underlying_type, is_convertible +#include // unordered_map +#include // pair, declval +#include // valarray + +// #include + + +#include // exception +#include // runtime_error +#include // to_string +#include // vector + +// #include + + +#include // array +#include // size_t +#include // uint8_t +#include // string + +namespace nlohmann +{ +namespace detail +{ +/////////////////////////// +// JSON type enumeration // +/////////////////////////// + +/*! +@brief the JSON type enumeration + +This enumeration collects the different JSON types. It is internally used to +distinguish the stored values, and the functions @ref basic_json::is_null(), +@ref basic_json::is_object(), @ref basic_json::is_array(), +@ref basic_json::is_string(), @ref basic_json::is_boolean(), +@ref basic_json::is_number() (with @ref basic_json::is_number_integer(), +@ref basic_json::is_number_unsigned(), and @ref basic_json::is_number_float()), +@ref basic_json::is_discarded(), @ref basic_json::is_primitive(), and +@ref basic_json::is_structured() rely on it. + +@note There are three enumeration entries (number_integer, number_unsigned, and +number_float), because the library distinguishes these three types for numbers: +@ref basic_json::number_unsigned_t is used for unsigned integers, +@ref basic_json::number_integer_t is used for signed integers, and +@ref basic_json::number_float_t is used for floating-point numbers or to +approximate integers which do not fit in the limits of their respective type. + +@sa see @ref basic_json::basic_json(const value_t value_type) -- create a JSON +value with the default value for a given type + +@since version 1.0.0 +*/ +enum class value_t : std::uint8_t +{ + null, ///< null value + object, ///< object (unordered set of name/value pairs) + array, ///< array (ordered collection of values) + string, ///< string value + boolean, ///< boolean value + number_integer, ///< number value (signed integer) + number_unsigned, ///< number value (unsigned integer) + number_float, ///< number value (floating-point) + binary, ///< binary array (ordered collection of bytes) + discarded ///< discarded by the parser callback function +}; + +/*! +@brief comparison operator for JSON types + +Returns an ordering that is similar to Python: +- order: null < boolean < number < object < array < string < binary +- furthermore, each type is not smaller than itself +- discarded values are not comparable +- binary is represented as a b"" string in python and directly comparable to a + string; however, making a binary array directly comparable with a string would + be surprising behavior in a JSON file. + +@since version 1.0.0 +*/ +inline bool operator<(const value_t lhs, const value_t rhs) noexcept +{ + static constexpr std::array order = {{ + 0 /* null */, 3 /* object */, 4 /* array */, 5 /* string */, + 1 /* boolean */, 2 /* integer */, 2 /* unsigned */, 2 /* float */, + 6 /* binary */ + } + }; + + const auto l_index = static_cast(lhs); + const auto r_index = static_cast(rhs); + return l_index < order.size() && r_index < order.size() && order[l_index] < order[r_index]; +} +} // namespace detail +} // namespace nlohmann + +// #include + + +#include +// #include + + +#include // declval, pair +// #include + + +/* Hedley - https://nemequ.github.io/hedley + * Created by Evan Nemerson + * + * To the extent possible under law, the author(s) have dedicated all + * copyright and related and neighboring rights to this software to + * the public domain worldwide. This software is distributed without + * any warranty. + * + * For details, see . + * SPDX-License-Identifier: CC0-1.0 + */ + +#if !defined(JSON_HEDLEY_VERSION) || (JSON_HEDLEY_VERSION < 15) +#if defined(JSON_HEDLEY_VERSION) + #undef JSON_HEDLEY_VERSION +#endif +#define JSON_HEDLEY_VERSION 15 + +#if defined(JSON_HEDLEY_STRINGIFY_EX) + #undef JSON_HEDLEY_STRINGIFY_EX +#endif +#define JSON_HEDLEY_STRINGIFY_EX(x) #x + +#if defined(JSON_HEDLEY_STRINGIFY) + #undef JSON_HEDLEY_STRINGIFY +#endif +#define JSON_HEDLEY_STRINGIFY(x) JSON_HEDLEY_STRINGIFY_EX(x) + +#if defined(JSON_HEDLEY_CONCAT_EX) + #undef JSON_HEDLEY_CONCAT_EX +#endif +#define JSON_HEDLEY_CONCAT_EX(a,b) a##b + +#if defined(JSON_HEDLEY_CONCAT) + #undef JSON_HEDLEY_CONCAT +#endif +#define JSON_HEDLEY_CONCAT(a,b) JSON_HEDLEY_CONCAT_EX(a,b) + +#if defined(JSON_HEDLEY_CONCAT3_EX) + #undef JSON_HEDLEY_CONCAT3_EX +#endif +#define JSON_HEDLEY_CONCAT3_EX(a,b,c) a##b##c + +#if defined(JSON_HEDLEY_CONCAT3) + #undef JSON_HEDLEY_CONCAT3 +#endif +#define JSON_HEDLEY_CONCAT3(a,b,c) JSON_HEDLEY_CONCAT3_EX(a,b,c) + +#if defined(JSON_HEDLEY_VERSION_ENCODE) + #undef JSON_HEDLEY_VERSION_ENCODE +#endif +#define JSON_HEDLEY_VERSION_ENCODE(major,minor,revision) (((major) * 1000000) + ((minor) * 1000) + (revision)) + +#if defined(JSON_HEDLEY_VERSION_DECODE_MAJOR) + #undef JSON_HEDLEY_VERSION_DECODE_MAJOR +#endif +#define JSON_HEDLEY_VERSION_DECODE_MAJOR(version) ((version) / 1000000) + +#if defined(JSON_HEDLEY_VERSION_DECODE_MINOR) + #undef JSON_HEDLEY_VERSION_DECODE_MINOR +#endif +#define JSON_HEDLEY_VERSION_DECODE_MINOR(version) (((version) % 1000000) / 1000) + +#if defined(JSON_HEDLEY_VERSION_DECODE_REVISION) + #undef JSON_HEDLEY_VERSION_DECODE_REVISION +#endif +#define JSON_HEDLEY_VERSION_DECODE_REVISION(version) ((version) % 1000) + +#if defined(JSON_HEDLEY_GNUC_VERSION) + #undef JSON_HEDLEY_GNUC_VERSION +#endif +#if defined(__GNUC__) && defined(__GNUC_PATCHLEVEL__) + #define JSON_HEDLEY_GNUC_VERSION JSON_HEDLEY_VERSION_ENCODE(__GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__) +#elif defined(__GNUC__) + #define JSON_HEDLEY_GNUC_VERSION JSON_HEDLEY_VERSION_ENCODE(__GNUC__, __GNUC_MINOR__, 0) +#endif + +#if defined(JSON_HEDLEY_GNUC_VERSION_CHECK) + #undef JSON_HEDLEY_GNUC_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_GNUC_VERSION) + #define JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_GNUC_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_MSVC_VERSION) + #undef JSON_HEDLEY_MSVC_VERSION +#endif +#if defined(_MSC_FULL_VER) && (_MSC_FULL_VER >= 140000000) && !defined(__ICL) + #define JSON_HEDLEY_MSVC_VERSION JSON_HEDLEY_VERSION_ENCODE(_MSC_FULL_VER / 10000000, (_MSC_FULL_VER % 10000000) / 100000, (_MSC_FULL_VER % 100000) / 100) +#elif defined(_MSC_FULL_VER) && !defined(__ICL) + #define JSON_HEDLEY_MSVC_VERSION JSON_HEDLEY_VERSION_ENCODE(_MSC_FULL_VER / 1000000, (_MSC_FULL_VER % 1000000) / 10000, (_MSC_FULL_VER % 10000) / 10) +#elif defined(_MSC_VER) && !defined(__ICL) + #define JSON_HEDLEY_MSVC_VERSION JSON_HEDLEY_VERSION_ENCODE(_MSC_VER / 100, _MSC_VER % 100, 0) +#endif + +#if defined(JSON_HEDLEY_MSVC_VERSION_CHECK) + #undef JSON_HEDLEY_MSVC_VERSION_CHECK +#endif +#if !defined(JSON_HEDLEY_MSVC_VERSION) + #define JSON_HEDLEY_MSVC_VERSION_CHECK(major,minor,patch) (0) +#elif defined(_MSC_VER) && (_MSC_VER >= 1400) + #define JSON_HEDLEY_MSVC_VERSION_CHECK(major,minor,patch) (_MSC_FULL_VER >= ((major * 10000000) + (minor * 100000) + (patch))) +#elif defined(_MSC_VER) && (_MSC_VER >= 1200) + #define JSON_HEDLEY_MSVC_VERSION_CHECK(major,minor,patch) (_MSC_FULL_VER >= ((major * 1000000) + (minor * 10000) + (patch))) +#else + #define JSON_HEDLEY_MSVC_VERSION_CHECK(major,minor,patch) (_MSC_VER >= ((major * 100) + (minor))) +#endif + +#if defined(JSON_HEDLEY_INTEL_VERSION) + #undef JSON_HEDLEY_INTEL_VERSION +#endif +#if defined(__INTEL_COMPILER) && defined(__INTEL_COMPILER_UPDATE) && !defined(__ICL) + #define JSON_HEDLEY_INTEL_VERSION JSON_HEDLEY_VERSION_ENCODE(__INTEL_COMPILER / 100, __INTEL_COMPILER % 100, __INTEL_COMPILER_UPDATE) +#elif defined(__INTEL_COMPILER) && !defined(__ICL) + #define JSON_HEDLEY_INTEL_VERSION JSON_HEDLEY_VERSION_ENCODE(__INTEL_COMPILER / 100, __INTEL_COMPILER % 100, 0) +#endif + +#if defined(JSON_HEDLEY_INTEL_VERSION_CHECK) + #undef JSON_HEDLEY_INTEL_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_INTEL_VERSION) + #define JSON_HEDLEY_INTEL_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_INTEL_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_INTEL_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_INTEL_CL_VERSION) + #undef JSON_HEDLEY_INTEL_CL_VERSION +#endif +#if defined(__INTEL_COMPILER) && defined(__INTEL_COMPILER_UPDATE) && defined(__ICL) + #define JSON_HEDLEY_INTEL_CL_VERSION JSON_HEDLEY_VERSION_ENCODE(__INTEL_COMPILER, __INTEL_COMPILER_UPDATE, 0) +#endif + +#if defined(JSON_HEDLEY_INTEL_CL_VERSION_CHECK) + #undef JSON_HEDLEY_INTEL_CL_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_INTEL_CL_VERSION) + #define JSON_HEDLEY_INTEL_CL_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_INTEL_CL_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_INTEL_CL_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_PGI_VERSION) + #undef JSON_HEDLEY_PGI_VERSION +#endif +#if defined(__PGI) && defined(__PGIC__) && defined(__PGIC_MINOR__) && defined(__PGIC_PATCHLEVEL__) + #define JSON_HEDLEY_PGI_VERSION JSON_HEDLEY_VERSION_ENCODE(__PGIC__, __PGIC_MINOR__, __PGIC_PATCHLEVEL__) +#endif + +#if defined(JSON_HEDLEY_PGI_VERSION_CHECK) + #undef JSON_HEDLEY_PGI_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_PGI_VERSION) + #define JSON_HEDLEY_PGI_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_PGI_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_PGI_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_SUNPRO_VERSION) + #undef JSON_HEDLEY_SUNPRO_VERSION +#endif +#if defined(__SUNPRO_C) && (__SUNPRO_C > 0x1000) + #define JSON_HEDLEY_SUNPRO_VERSION JSON_HEDLEY_VERSION_ENCODE((((__SUNPRO_C >> 16) & 0xf) * 10) + ((__SUNPRO_C >> 12) & 0xf), (((__SUNPRO_C >> 8) & 0xf) * 10) + ((__SUNPRO_C >> 4) & 0xf), (__SUNPRO_C & 0xf) * 10) +#elif defined(__SUNPRO_C) + #define JSON_HEDLEY_SUNPRO_VERSION JSON_HEDLEY_VERSION_ENCODE((__SUNPRO_C >> 8) & 0xf, (__SUNPRO_C >> 4) & 0xf, (__SUNPRO_C) & 0xf) +#elif defined(__SUNPRO_CC) && (__SUNPRO_CC > 0x1000) + #define JSON_HEDLEY_SUNPRO_VERSION JSON_HEDLEY_VERSION_ENCODE((((__SUNPRO_CC >> 16) & 0xf) * 10) + ((__SUNPRO_CC >> 12) & 0xf), (((__SUNPRO_CC >> 8) & 0xf) * 10) + ((__SUNPRO_CC >> 4) & 0xf), (__SUNPRO_CC & 0xf) * 10) +#elif defined(__SUNPRO_CC) + #define JSON_HEDLEY_SUNPRO_VERSION JSON_HEDLEY_VERSION_ENCODE((__SUNPRO_CC >> 8) & 0xf, (__SUNPRO_CC >> 4) & 0xf, (__SUNPRO_CC) & 0xf) +#endif + +#if defined(JSON_HEDLEY_SUNPRO_VERSION_CHECK) + #undef JSON_HEDLEY_SUNPRO_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_SUNPRO_VERSION) + #define JSON_HEDLEY_SUNPRO_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_SUNPRO_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_SUNPRO_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_EMSCRIPTEN_VERSION) + #undef JSON_HEDLEY_EMSCRIPTEN_VERSION +#endif +#if defined(__EMSCRIPTEN__) + #define JSON_HEDLEY_EMSCRIPTEN_VERSION JSON_HEDLEY_VERSION_ENCODE(__EMSCRIPTEN_major__, __EMSCRIPTEN_minor__, __EMSCRIPTEN_tiny__) +#endif + +#if defined(JSON_HEDLEY_EMSCRIPTEN_VERSION_CHECK) + #undef JSON_HEDLEY_EMSCRIPTEN_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_EMSCRIPTEN_VERSION) + #define JSON_HEDLEY_EMSCRIPTEN_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_EMSCRIPTEN_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_EMSCRIPTEN_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_ARM_VERSION) + #undef JSON_HEDLEY_ARM_VERSION +#endif +#if defined(__CC_ARM) && defined(__ARMCOMPILER_VERSION) + #define JSON_HEDLEY_ARM_VERSION JSON_HEDLEY_VERSION_ENCODE(__ARMCOMPILER_VERSION / 1000000, (__ARMCOMPILER_VERSION % 1000000) / 10000, (__ARMCOMPILER_VERSION % 10000) / 100) +#elif defined(__CC_ARM) && defined(__ARMCC_VERSION) + #define JSON_HEDLEY_ARM_VERSION JSON_HEDLEY_VERSION_ENCODE(__ARMCC_VERSION / 1000000, (__ARMCC_VERSION % 1000000) / 10000, (__ARMCC_VERSION % 10000) / 100) +#endif + +#if defined(JSON_HEDLEY_ARM_VERSION_CHECK) + #undef JSON_HEDLEY_ARM_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_ARM_VERSION) + #define JSON_HEDLEY_ARM_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_ARM_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_ARM_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_IBM_VERSION) + #undef JSON_HEDLEY_IBM_VERSION +#endif +#if defined(__ibmxl__) + #define JSON_HEDLEY_IBM_VERSION JSON_HEDLEY_VERSION_ENCODE(__ibmxl_version__, __ibmxl_release__, __ibmxl_modification__) +#elif defined(__xlC__) && defined(__xlC_ver__) + #define JSON_HEDLEY_IBM_VERSION JSON_HEDLEY_VERSION_ENCODE(__xlC__ >> 8, __xlC__ & 0xff, (__xlC_ver__ >> 8) & 0xff) +#elif defined(__xlC__) + #define JSON_HEDLEY_IBM_VERSION JSON_HEDLEY_VERSION_ENCODE(__xlC__ >> 8, __xlC__ & 0xff, 0) +#endif + +#if defined(JSON_HEDLEY_IBM_VERSION_CHECK) + #undef JSON_HEDLEY_IBM_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_IBM_VERSION) + #define JSON_HEDLEY_IBM_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_IBM_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_IBM_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_VERSION) + #undef JSON_HEDLEY_TI_VERSION +#endif +#if \ + defined(__TI_COMPILER_VERSION__) && \ + ( \ + defined(__TMS470__) || defined(__TI_ARM__) || \ + defined(__MSP430__) || \ + defined(__TMS320C2000__) \ + ) +#if (__TI_COMPILER_VERSION__ >= 16000000) + #define JSON_HEDLEY_TI_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif +#endif + +#if defined(JSON_HEDLEY_TI_VERSION_CHECK) + #undef JSON_HEDLEY_TI_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_VERSION) + #define JSON_HEDLEY_TI_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_CL2000_VERSION) + #undef JSON_HEDLEY_TI_CL2000_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && defined(__TMS320C2000__) + #define JSON_HEDLEY_TI_CL2000_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_CL2000_VERSION_CHECK) + #undef JSON_HEDLEY_TI_CL2000_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_CL2000_VERSION) + #define JSON_HEDLEY_TI_CL2000_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_CL2000_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_CL2000_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_CL430_VERSION) + #undef JSON_HEDLEY_TI_CL430_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && defined(__MSP430__) + #define JSON_HEDLEY_TI_CL430_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_CL430_VERSION_CHECK) + #undef JSON_HEDLEY_TI_CL430_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_CL430_VERSION) + #define JSON_HEDLEY_TI_CL430_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_CL430_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_CL430_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_ARMCL_VERSION) + #undef JSON_HEDLEY_TI_ARMCL_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && (defined(__TMS470__) || defined(__TI_ARM__)) + #define JSON_HEDLEY_TI_ARMCL_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_ARMCL_VERSION_CHECK) + #undef JSON_HEDLEY_TI_ARMCL_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_ARMCL_VERSION) + #define JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_ARMCL_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_CL6X_VERSION) + #undef JSON_HEDLEY_TI_CL6X_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && defined(__TMS320C6X__) + #define JSON_HEDLEY_TI_CL6X_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_CL6X_VERSION_CHECK) + #undef JSON_HEDLEY_TI_CL6X_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_CL6X_VERSION) + #define JSON_HEDLEY_TI_CL6X_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_CL6X_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_CL6X_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_CL7X_VERSION) + #undef JSON_HEDLEY_TI_CL7X_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && defined(__C7000__) + #define JSON_HEDLEY_TI_CL7X_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_CL7X_VERSION_CHECK) + #undef JSON_HEDLEY_TI_CL7X_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_CL7X_VERSION) + #define JSON_HEDLEY_TI_CL7X_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_CL7X_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_CL7X_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TI_CLPRU_VERSION) + #undef JSON_HEDLEY_TI_CLPRU_VERSION +#endif +#if defined(__TI_COMPILER_VERSION__) && defined(__PRU__) + #define JSON_HEDLEY_TI_CLPRU_VERSION JSON_HEDLEY_VERSION_ENCODE(__TI_COMPILER_VERSION__ / 1000000, (__TI_COMPILER_VERSION__ % 1000000) / 1000, (__TI_COMPILER_VERSION__ % 1000)) +#endif + +#if defined(JSON_HEDLEY_TI_CLPRU_VERSION_CHECK) + #undef JSON_HEDLEY_TI_CLPRU_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TI_CLPRU_VERSION) + #define JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TI_CLPRU_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_CRAY_VERSION) + #undef JSON_HEDLEY_CRAY_VERSION +#endif +#if defined(_CRAYC) + #if defined(_RELEASE_PATCHLEVEL) + #define JSON_HEDLEY_CRAY_VERSION JSON_HEDLEY_VERSION_ENCODE(_RELEASE_MAJOR, _RELEASE_MINOR, _RELEASE_PATCHLEVEL) + #else + #define JSON_HEDLEY_CRAY_VERSION JSON_HEDLEY_VERSION_ENCODE(_RELEASE_MAJOR, _RELEASE_MINOR, 0) + #endif +#endif + +#if defined(JSON_HEDLEY_CRAY_VERSION_CHECK) + #undef JSON_HEDLEY_CRAY_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_CRAY_VERSION) + #define JSON_HEDLEY_CRAY_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_CRAY_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_CRAY_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_IAR_VERSION) + #undef JSON_HEDLEY_IAR_VERSION +#endif +#if defined(__IAR_SYSTEMS_ICC__) + #if __VER__ > 1000 + #define JSON_HEDLEY_IAR_VERSION JSON_HEDLEY_VERSION_ENCODE((__VER__ / 1000000), ((__VER__ / 1000) % 1000), (__VER__ % 1000)) + #else + #define JSON_HEDLEY_IAR_VERSION JSON_HEDLEY_VERSION_ENCODE(__VER__ / 100, __VER__ % 100, 0) + #endif +#endif + +#if defined(JSON_HEDLEY_IAR_VERSION_CHECK) + #undef JSON_HEDLEY_IAR_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_IAR_VERSION) + #define JSON_HEDLEY_IAR_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_IAR_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_IAR_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_TINYC_VERSION) + #undef JSON_HEDLEY_TINYC_VERSION +#endif +#if defined(__TINYC__) + #define JSON_HEDLEY_TINYC_VERSION JSON_HEDLEY_VERSION_ENCODE(__TINYC__ / 1000, (__TINYC__ / 100) % 10, __TINYC__ % 100) +#endif + +#if defined(JSON_HEDLEY_TINYC_VERSION_CHECK) + #undef JSON_HEDLEY_TINYC_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_TINYC_VERSION) + #define JSON_HEDLEY_TINYC_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_TINYC_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_TINYC_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_DMC_VERSION) + #undef JSON_HEDLEY_DMC_VERSION +#endif +#if defined(__DMC__) + #define JSON_HEDLEY_DMC_VERSION JSON_HEDLEY_VERSION_ENCODE(__DMC__ >> 8, (__DMC__ >> 4) & 0xf, __DMC__ & 0xf) +#endif + +#if defined(JSON_HEDLEY_DMC_VERSION_CHECK) + #undef JSON_HEDLEY_DMC_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_DMC_VERSION) + #define JSON_HEDLEY_DMC_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_DMC_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_DMC_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_COMPCERT_VERSION) + #undef JSON_HEDLEY_COMPCERT_VERSION +#endif +#if defined(__COMPCERT_VERSION__) + #define JSON_HEDLEY_COMPCERT_VERSION JSON_HEDLEY_VERSION_ENCODE(__COMPCERT_VERSION__ / 10000, (__COMPCERT_VERSION__ / 100) % 100, __COMPCERT_VERSION__ % 100) +#endif + +#if defined(JSON_HEDLEY_COMPCERT_VERSION_CHECK) + #undef JSON_HEDLEY_COMPCERT_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_COMPCERT_VERSION) + #define JSON_HEDLEY_COMPCERT_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_COMPCERT_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_COMPCERT_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_PELLES_VERSION) + #undef JSON_HEDLEY_PELLES_VERSION +#endif +#if defined(__POCC__) + #define JSON_HEDLEY_PELLES_VERSION JSON_HEDLEY_VERSION_ENCODE(__POCC__ / 100, __POCC__ % 100, 0) +#endif + +#if defined(JSON_HEDLEY_PELLES_VERSION_CHECK) + #undef JSON_HEDLEY_PELLES_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_PELLES_VERSION) + #define JSON_HEDLEY_PELLES_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_PELLES_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_PELLES_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_MCST_LCC_VERSION) + #undef JSON_HEDLEY_MCST_LCC_VERSION +#endif +#if defined(__LCC__) && defined(__LCC_MINOR__) + #define JSON_HEDLEY_MCST_LCC_VERSION JSON_HEDLEY_VERSION_ENCODE(__LCC__ / 100, __LCC__ % 100, __LCC_MINOR__) +#endif + +#if defined(JSON_HEDLEY_MCST_LCC_VERSION_CHECK) + #undef JSON_HEDLEY_MCST_LCC_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_MCST_LCC_VERSION) + #define JSON_HEDLEY_MCST_LCC_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_MCST_LCC_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_MCST_LCC_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_GCC_VERSION) + #undef JSON_HEDLEY_GCC_VERSION +#endif +#if \ + defined(JSON_HEDLEY_GNUC_VERSION) && \ + !defined(__clang__) && \ + !defined(JSON_HEDLEY_INTEL_VERSION) && \ + !defined(JSON_HEDLEY_PGI_VERSION) && \ + !defined(JSON_HEDLEY_ARM_VERSION) && \ + !defined(JSON_HEDLEY_CRAY_VERSION) && \ + !defined(JSON_HEDLEY_TI_VERSION) && \ + !defined(JSON_HEDLEY_TI_ARMCL_VERSION) && \ + !defined(JSON_HEDLEY_TI_CL430_VERSION) && \ + !defined(JSON_HEDLEY_TI_CL2000_VERSION) && \ + !defined(JSON_HEDLEY_TI_CL6X_VERSION) && \ + !defined(JSON_HEDLEY_TI_CL7X_VERSION) && \ + !defined(JSON_HEDLEY_TI_CLPRU_VERSION) && \ + !defined(__COMPCERT__) && \ + !defined(JSON_HEDLEY_MCST_LCC_VERSION) + #define JSON_HEDLEY_GCC_VERSION JSON_HEDLEY_GNUC_VERSION +#endif + +#if defined(JSON_HEDLEY_GCC_VERSION_CHECK) + #undef JSON_HEDLEY_GCC_VERSION_CHECK +#endif +#if defined(JSON_HEDLEY_GCC_VERSION) + #define JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) (JSON_HEDLEY_GCC_VERSION >= JSON_HEDLEY_VERSION_ENCODE(major, minor, patch)) +#else + #define JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) (0) +#endif + +#if defined(JSON_HEDLEY_HAS_ATTRIBUTE) + #undef JSON_HEDLEY_HAS_ATTRIBUTE +#endif +#if \ + defined(__has_attribute) && \ + ( \ + (!defined(JSON_HEDLEY_IAR_VERSION) || JSON_HEDLEY_IAR_VERSION_CHECK(8,5,9)) \ + ) +# define JSON_HEDLEY_HAS_ATTRIBUTE(attribute) __has_attribute(attribute) +#else +# define JSON_HEDLEY_HAS_ATTRIBUTE(attribute) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_ATTRIBUTE) + #undef JSON_HEDLEY_GNUC_HAS_ATTRIBUTE +#endif +#if defined(__has_attribute) + #define JSON_HEDLEY_GNUC_HAS_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_HAS_ATTRIBUTE(attribute) +#else + #define JSON_HEDLEY_GNUC_HAS_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_ATTRIBUTE) + #undef JSON_HEDLEY_GCC_HAS_ATTRIBUTE +#endif +#if defined(__has_attribute) + #define JSON_HEDLEY_GCC_HAS_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_HAS_ATTRIBUTE(attribute) +#else + #define JSON_HEDLEY_GCC_HAS_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_CPP_ATTRIBUTE) + #undef JSON_HEDLEY_HAS_CPP_ATTRIBUTE +#endif +#if \ + defined(__has_cpp_attribute) && \ + defined(__cplusplus) && \ + (!defined(JSON_HEDLEY_SUNPRO_VERSION) || JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,15,0)) + #define JSON_HEDLEY_HAS_CPP_ATTRIBUTE(attribute) __has_cpp_attribute(attribute) +#else + #define JSON_HEDLEY_HAS_CPP_ATTRIBUTE(attribute) (0) +#endif + +#if defined(JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS) + #undef JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS +#endif +#if !defined(__cplusplus) || !defined(__has_cpp_attribute) + #define JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS(ns,attribute) (0) +#elif \ + !defined(JSON_HEDLEY_PGI_VERSION) && \ + !defined(JSON_HEDLEY_IAR_VERSION) && \ + (!defined(JSON_HEDLEY_SUNPRO_VERSION) || JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,15,0)) && \ + (!defined(JSON_HEDLEY_MSVC_VERSION) || JSON_HEDLEY_MSVC_VERSION_CHECK(19,20,0)) + #define JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS(ns,attribute) JSON_HEDLEY_HAS_CPP_ATTRIBUTE(ns::attribute) +#else + #define JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS(ns,attribute) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_CPP_ATTRIBUTE) + #undef JSON_HEDLEY_GNUC_HAS_CPP_ATTRIBUTE +#endif +#if defined(__has_cpp_attribute) && defined(__cplusplus) + #define JSON_HEDLEY_GNUC_HAS_CPP_ATTRIBUTE(attribute,major,minor,patch) __has_cpp_attribute(attribute) +#else + #define JSON_HEDLEY_GNUC_HAS_CPP_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_CPP_ATTRIBUTE) + #undef JSON_HEDLEY_GCC_HAS_CPP_ATTRIBUTE +#endif +#if defined(__has_cpp_attribute) && defined(__cplusplus) + #define JSON_HEDLEY_GCC_HAS_CPP_ATTRIBUTE(attribute,major,minor,patch) __has_cpp_attribute(attribute) +#else + #define JSON_HEDLEY_GCC_HAS_CPP_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_BUILTIN) + #undef JSON_HEDLEY_HAS_BUILTIN +#endif +#if defined(__has_builtin) + #define JSON_HEDLEY_HAS_BUILTIN(builtin) __has_builtin(builtin) +#else + #define JSON_HEDLEY_HAS_BUILTIN(builtin) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_BUILTIN) + #undef JSON_HEDLEY_GNUC_HAS_BUILTIN +#endif +#if defined(__has_builtin) + #define JSON_HEDLEY_GNUC_HAS_BUILTIN(builtin,major,minor,patch) __has_builtin(builtin) +#else + #define JSON_HEDLEY_GNUC_HAS_BUILTIN(builtin,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_BUILTIN) + #undef JSON_HEDLEY_GCC_HAS_BUILTIN +#endif +#if defined(__has_builtin) + #define JSON_HEDLEY_GCC_HAS_BUILTIN(builtin,major,minor,patch) __has_builtin(builtin) +#else + #define JSON_HEDLEY_GCC_HAS_BUILTIN(builtin,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_FEATURE) + #undef JSON_HEDLEY_HAS_FEATURE +#endif +#if defined(__has_feature) + #define JSON_HEDLEY_HAS_FEATURE(feature) __has_feature(feature) +#else + #define JSON_HEDLEY_HAS_FEATURE(feature) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_FEATURE) + #undef JSON_HEDLEY_GNUC_HAS_FEATURE +#endif +#if defined(__has_feature) + #define JSON_HEDLEY_GNUC_HAS_FEATURE(feature,major,minor,patch) __has_feature(feature) +#else + #define JSON_HEDLEY_GNUC_HAS_FEATURE(feature,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_FEATURE) + #undef JSON_HEDLEY_GCC_HAS_FEATURE +#endif +#if defined(__has_feature) + #define JSON_HEDLEY_GCC_HAS_FEATURE(feature,major,minor,patch) __has_feature(feature) +#else + #define JSON_HEDLEY_GCC_HAS_FEATURE(feature,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_EXTENSION) + #undef JSON_HEDLEY_HAS_EXTENSION +#endif +#if defined(__has_extension) + #define JSON_HEDLEY_HAS_EXTENSION(extension) __has_extension(extension) +#else + #define JSON_HEDLEY_HAS_EXTENSION(extension) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_EXTENSION) + #undef JSON_HEDLEY_GNUC_HAS_EXTENSION +#endif +#if defined(__has_extension) + #define JSON_HEDLEY_GNUC_HAS_EXTENSION(extension,major,minor,patch) __has_extension(extension) +#else + #define JSON_HEDLEY_GNUC_HAS_EXTENSION(extension,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_EXTENSION) + #undef JSON_HEDLEY_GCC_HAS_EXTENSION +#endif +#if defined(__has_extension) + #define JSON_HEDLEY_GCC_HAS_EXTENSION(extension,major,minor,patch) __has_extension(extension) +#else + #define JSON_HEDLEY_GCC_HAS_EXTENSION(extension,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE) + #undef JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE +#endif +#if defined(__has_declspec_attribute) + #define JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE(attribute) __has_declspec_attribute(attribute) +#else + #define JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE(attribute) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_DECLSPEC_ATTRIBUTE) + #undef JSON_HEDLEY_GNUC_HAS_DECLSPEC_ATTRIBUTE +#endif +#if defined(__has_declspec_attribute) + #define JSON_HEDLEY_GNUC_HAS_DECLSPEC_ATTRIBUTE(attribute,major,minor,patch) __has_declspec_attribute(attribute) +#else + #define JSON_HEDLEY_GNUC_HAS_DECLSPEC_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_DECLSPEC_ATTRIBUTE) + #undef JSON_HEDLEY_GCC_HAS_DECLSPEC_ATTRIBUTE +#endif +#if defined(__has_declspec_attribute) + #define JSON_HEDLEY_GCC_HAS_DECLSPEC_ATTRIBUTE(attribute,major,minor,patch) __has_declspec_attribute(attribute) +#else + #define JSON_HEDLEY_GCC_HAS_DECLSPEC_ATTRIBUTE(attribute,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_HAS_WARNING) + #undef JSON_HEDLEY_HAS_WARNING +#endif +#if defined(__has_warning) + #define JSON_HEDLEY_HAS_WARNING(warning) __has_warning(warning) +#else + #define JSON_HEDLEY_HAS_WARNING(warning) (0) +#endif + +#if defined(JSON_HEDLEY_GNUC_HAS_WARNING) + #undef JSON_HEDLEY_GNUC_HAS_WARNING +#endif +#if defined(__has_warning) + #define JSON_HEDLEY_GNUC_HAS_WARNING(warning,major,minor,patch) __has_warning(warning) +#else + #define JSON_HEDLEY_GNUC_HAS_WARNING(warning,major,minor,patch) JSON_HEDLEY_GNUC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_GCC_HAS_WARNING) + #undef JSON_HEDLEY_GCC_HAS_WARNING +#endif +#if defined(__has_warning) + #define JSON_HEDLEY_GCC_HAS_WARNING(warning,major,minor,patch) __has_warning(warning) +#else + #define JSON_HEDLEY_GCC_HAS_WARNING(warning,major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if \ + (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) || \ + defined(__clang__) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(18,4,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,7,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(2,0,1) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,1,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,0,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_CRAY_VERSION_CHECK(5,0,0) || \ + JSON_HEDLEY_TINYC_VERSION_CHECK(0,9,17) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(8,0,0) || \ + (JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) && defined(__C99_PRAGMA_OPERATOR)) + #define JSON_HEDLEY_PRAGMA(value) _Pragma(#value) +#elif JSON_HEDLEY_MSVC_VERSION_CHECK(15,0,0) + #define JSON_HEDLEY_PRAGMA(value) __pragma(value) +#else + #define JSON_HEDLEY_PRAGMA(value) +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_PUSH) + #undef JSON_HEDLEY_DIAGNOSTIC_PUSH +#endif +#if defined(JSON_HEDLEY_DIAGNOSTIC_POP) + #undef JSON_HEDLEY_DIAGNOSTIC_POP +#endif +#if defined(__clang__) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("clang diagnostic push") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("clang diagnostic pop") +#elif JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("warning(push)") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("warning(pop)") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(4,6,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("GCC diagnostic push") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("GCC diagnostic pop") +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(15,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH __pragma(warning(push)) + #define JSON_HEDLEY_DIAGNOSTIC_POP __pragma(warning(pop)) +#elif JSON_HEDLEY_ARM_VERSION_CHECK(5,6,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("push") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("pop") +#elif \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,4,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,1,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("diag_push") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("diag_pop") +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(2,90,0) + #define JSON_HEDLEY_DIAGNOSTIC_PUSH _Pragma("warning(push)") + #define JSON_HEDLEY_DIAGNOSTIC_POP _Pragma("warning(pop)") +#else + #define JSON_HEDLEY_DIAGNOSTIC_PUSH + #define JSON_HEDLEY_DIAGNOSTIC_POP +#endif + +/* JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_ is for + HEDLEY INTERNAL USE ONLY. API subject to change without notice. */ +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_ +#endif +#if defined(__cplusplus) +# if JSON_HEDLEY_HAS_WARNING("-Wc++98-compat") +# if JSON_HEDLEY_HAS_WARNING("-Wc++17-extensions") +# if JSON_HEDLEY_HAS_WARNING("-Wc++1z-extensions") +# define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(xpr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wc++98-compat\"") \ + _Pragma("clang diagnostic ignored \"-Wc++17-extensions\"") \ + _Pragma("clang diagnostic ignored \"-Wc++1z-extensions\"") \ + xpr \ + JSON_HEDLEY_DIAGNOSTIC_POP +# else +# define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(xpr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wc++98-compat\"") \ + _Pragma("clang diagnostic ignored \"-Wc++17-extensions\"") \ + xpr \ + JSON_HEDLEY_DIAGNOSTIC_POP +# endif +# else +# define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(xpr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wc++98-compat\"") \ + xpr \ + JSON_HEDLEY_DIAGNOSTIC_POP +# endif +# endif +#endif +#if !defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(x) x +#endif + +#if defined(JSON_HEDLEY_CONST_CAST) + #undef JSON_HEDLEY_CONST_CAST +#endif +#if defined(__cplusplus) +# define JSON_HEDLEY_CONST_CAST(T, expr) (const_cast(expr)) +#elif \ + JSON_HEDLEY_HAS_WARNING("-Wcast-qual") || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,6,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) +# define JSON_HEDLEY_CONST_CAST(T, expr) (__extension__ ({ \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL \ + ((T) (expr)); \ + JSON_HEDLEY_DIAGNOSTIC_POP \ + })) +#else +# define JSON_HEDLEY_CONST_CAST(T, expr) ((T) (expr)) +#endif + +#if defined(JSON_HEDLEY_REINTERPRET_CAST) + #undef JSON_HEDLEY_REINTERPRET_CAST +#endif +#if defined(__cplusplus) + #define JSON_HEDLEY_REINTERPRET_CAST(T, expr) (reinterpret_cast(expr)) +#else + #define JSON_HEDLEY_REINTERPRET_CAST(T, expr) ((T) (expr)) +#endif + +#if defined(JSON_HEDLEY_STATIC_CAST) + #undef JSON_HEDLEY_STATIC_CAST +#endif +#if defined(__cplusplus) + #define JSON_HEDLEY_STATIC_CAST(T, expr) (static_cast(expr)) +#else + #define JSON_HEDLEY_STATIC_CAST(T, expr) ((T) (expr)) +#endif + +#if defined(JSON_HEDLEY_CPP_CAST) + #undef JSON_HEDLEY_CPP_CAST +#endif +#if defined(__cplusplus) +# if JSON_HEDLEY_HAS_WARNING("-Wold-style-cast") +# define JSON_HEDLEY_CPP_CAST(T, expr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wold-style-cast\"") \ + ((T) (expr)) \ + JSON_HEDLEY_DIAGNOSTIC_POP +# elif JSON_HEDLEY_IAR_VERSION_CHECK(8,3,0) +# define JSON_HEDLEY_CPP_CAST(T, expr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("diag_suppress=Pe137") \ + JSON_HEDLEY_DIAGNOSTIC_POP +# else +# define JSON_HEDLEY_CPP_CAST(T, expr) ((T) (expr)) +# endif +#else +# define JSON_HEDLEY_CPP_CAST(T, expr) (expr) +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wdeprecated-declarations") + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("clang diagnostic ignored \"-Wdeprecated-declarations\"") +#elif JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("warning(disable:1478 1786)") +#elif JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED __pragma(warning(disable:1478 1786)) +#elif JSON_HEDLEY_PGI_VERSION_CHECK(20,7,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("diag_suppress 1215,1216,1444,1445") +#elif JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("diag_suppress 1215,1444") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(4,3,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("GCC diagnostic ignored \"-Wdeprecated-declarations\"") +#elif JSON_HEDLEY_MSVC_VERSION_CHECK(15,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED __pragma(warning(disable:4996)) +#elif JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("diag_suppress 1215,1444") +#elif \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("diag_suppress 1291,1718") +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,13,0) && !defined(__cplusplus) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("error_messages(off,E_DEPRECATED_ATT,E_DEPRECATED_ATT_MESS)") +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,13,0) && defined(__cplusplus) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("error_messages(off,symdeprecated,symdeprecated2)") +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("diag_suppress=Pe1444,Pe1215") +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(2,90,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED _Pragma("warn(disable:2241)") +#else + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wunknown-pragmas") + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("clang diagnostic ignored \"-Wunknown-pragmas\"") +#elif JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("warning(disable:161)") +#elif JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS __pragma(warning(disable:161)) +#elif JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("diag_suppress 1675") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(4,3,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("GCC diagnostic ignored \"-Wunknown-pragmas\"") +#elif JSON_HEDLEY_MSVC_VERSION_CHECK(15,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS __pragma(warning(disable:4068)) +#elif \ + JSON_HEDLEY_TI_VERSION_CHECK(16,9,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,0,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,3,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("diag_suppress 163") +#elif JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("diag_suppress 163") +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("diag_suppress=Pe161") +#elif JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS _Pragma("diag_suppress 161") +#else + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wunknown-attributes") + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("clang diagnostic ignored \"-Wunknown-attributes\"") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(4,6,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("GCC diagnostic ignored \"-Wdeprecated-declarations\"") +#elif JSON_HEDLEY_INTEL_VERSION_CHECK(17,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("warning(disable:1292)") +#elif JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES __pragma(warning(disable:1292)) +#elif JSON_HEDLEY_MSVC_VERSION_CHECK(19,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES __pragma(warning(disable:5030)) +#elif JSON_HEDLEY_PGI_VERSION_CHECK(20,7,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("diag_suppress 1097,1098") +#elif JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("diag_suppress 1097") +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,14,0) && defined(__cplusplus) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("error_messages(off,attrskipunsup)") +#elif \ + JSON_HEDLEY_TI_VERSION_CHECK(18,1,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,3,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("diag_suppress 1173") +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("diag_suppress=Pe1097") +#elif JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES _Pragma("diag_suppress 1097") +#else + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wcast-qual") + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL _Pragma("clang diagnostic ignored \"-Wcast-qual\"") +#elif JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL _Pragma("warning(disable:2203 2331)") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(3,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL _Pragma("GCC diagnostic ignored \"-Wcast-qual\"") +#else + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL +#endif + +#if defined(JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION) + #undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wunused-function") + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION _Pragma("clang diagnostic ignored \"-Wunused-function\"") +#elif JSON_HEDLEY_GCC_VERSION_CHECK(3,4,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION _Pragma("GCC diagnostic ignored \"-Wunused-function\"") +#elif JSON_HEDLEY_MSVC_VERSION_CHECK(1,0,0) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION __pragma(warning(disable:4505)) +#elif JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION _Pragma("diag_suppress 3142") +#else + #define JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION +#endif + +#if defined(JSON_HEDLEY_DEPRECATED) + #undef JSON_HEDLEY_DEPRECATED +#endif +#if defined(JSON_HEDLEY_DEPRECATED_FOR) + #undef JSON_HEDLEY_DEPRECATED_FOR +#endif +#if \ + JSON_HEDLEY_MSVC_VERSION_CHECK(14,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DEPRECATED(since) __declspec(deprecated("Since " # since)) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) __declspec(deprecated("Since " #since "; use " #replacement)) +#elif \ + (JSON_HEDLEY_HAS_EXTENSION(attribute_deprecated_with_message) && !defined(JSON_HEDLEY_IAR_VERSION)) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,5,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(5,6,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,13,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(18,1,0) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(18,1,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,3,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,3,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_DEPRECATED(since) __attribute__((__deprecated__("Since " #since))) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) __attribute__((__deprecated__("Since " #since "; use " #replacement))) +#elif defined(__cplusplus) && (__cplusplus >= 201402L) + #define JSON_HEDLEY_DEPRECATED(since) JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[deprecated("Since " #since)]]) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[deprecated("Since " #since "; use " #replacement)]]) +#elif \ + JSON_HEDLEY_HAS_ATTRIBUTE(deprecated) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,10,0) + #define JSON_HEDLEY_DEPRECATED(since) __attribute__((__deprecated__)) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) __attribute__((__deprecated__)) +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(13,10,0) || \ + JSON_HEDLEY_PELLES_VERSION_CHECK(6,50,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_DEPRECATED(since) __declspec(deprecated) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) __declspec(deprecated) +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_DEPRECATED(since) _Pragma("deprecated") + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) _Pragma("deprecated") +#else + #define JSON_HEDLEY_DEPRECATED(since) + #define JSON_HEDLEY_DEPRECATED_FOR(since, replacement) +#endif + +#if defined(JSON_HEDLEY_UNAVAILABLE) + #undef JSON_HEDLEY_UNAVAILABLE +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(warning) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,3,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_UNAVAILABLE(available_since) __attribute__((__warning__("Not available until " #available_since))) +#else + #define JSON_HEDLEY_UNAVAILABLE(available_since) +#endif + +#if defined(JSON_HEDLEY_WARN_UNUSED_RESULT) + #undef JSON_HEDLEY_WARN_UNUSED_RESULT +#endif +#if defined(JSON_HEDLEY_WARN_UNUSED_RESULT_MSG) + #undef JSON_HEDLEY_WARN_UNUSED_RESULT_MSG +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(warn_unused_result) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,4,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + (JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,15,0) && defined(__cplusplus)) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_WARN_UNUSED_RESULT __attribute__((__warn_unused_result__)) + #define JSON_HEDLEY_WARN_UNUSED_RESULT_MSG(msg) __attribute__((__warn_unused_result__)) +#elif (JSON_HEDLEY_HAS_CPP_ATTRIBUTE(nodiscard) >= 201907L) + #define JSON_HEDLEY_WARN_UNUSED_RESULT JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[nodiscard]]) + #define JSON_HEDLEY_WARN_UNUSED_RESULT_MSG(msg) JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[nodiscard(msg)]]) +#elif JSON_HEDLEY_HAS_CPP_ATTRIBUTE(nodiscard) + #define JSON_HEDLEY_WARN_UNUSED_RESULT JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[nodiscard]]) + #define JSON_HEDLEY_WARN_UNUSED_RESULT_MSG(msg) JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[nodiscard]]) +#elif defined(_Check_return_) /* SAL */ + #define JSON_HEDLEY_WARN_UNUSED_RESULT _Check_return_ + #define JSON_HEDLEY_WARN_UNUSED_RESULT_MSG(msg) _Check_return_ +#else + #define JSON_HEDLEY_WARN_UNUSED_RESULT + #define JSON_HEDLEY_WARN_UNUSED_RESULT_MSG(msg) +#endif + +#if defined(JSON_HEDLEY_SENTINEL) + #undef JSON_HEDLEY_SENTINEL +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(sentinel) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(5,4,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_SENTINEL(position) __attribute__((__sentinel__(position))) +#else + #define JSON_HEDLEY_SENTINEL(position) +#endif + +#if defined(JSON_HEDLEY_NO_RETURN) + #undef JSON_HEDLEY_NO_RETURN +#endif +#if JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_NO_RETURN __noreturn +#elif \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_NO_RETURN __attribute__((__noreturn__)) +#elif defined(__STDC_VERSION__) && __STDC_VERSION__ >= 201112L + #define JSON_HEDLEY_NO_RETURN _Noreturn +#elif defined(__cplusplus) && (__cplusplus >= 201103L) + #define JSON_HEDLEY_NO_RETURN JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[noreturn]]) +#elif \ + JSON_HEDLEY_HAS_ATTRIBUTE(noreturn) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,2,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,10,0) + #define JSON_HEDLEY_NO_RETURN __attribute__((__noreturn__)) +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,10,0) + #define JSON_HEDLEY_NO_RETURN _Pragma("does_not_return") +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(13,10,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_NO_RETURN __declspec(noreturn) +#elif JSON_HEDLEY_TI_CL6X_VERSION_CHECK(6,0,0) && defined(__cplusplus) + #define JSON_HEDLEY_NO_RETURN _Pragma("FUNC_NEVER_RETURNS;") +#elif JSON_HEDLEY_COMPCERT_VERSION_CHECK(3,2,0) + #define JSON_HEDLEY_NO_RETURN __attribute((noreturn)) +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(9,0,0) + #define JSON_HEDLEY_NO_RETURN __declspec(noreturn) +#else + #define JSON_HEDLEY_NO_RETURN +#endif + +#if defined(JSON_HEDLEY_NO_ESCAPE) + #undef JSON_HEDLEY_NO_ESCAPE +#endif +#if JSON_HEDLEY_HAS_ATTRIBUTE(noescape) + #define JSON_HEDLEY_NO_ESCAPE __attribute__((__noescape__)) +#else + #define JSON_HEDLEY_NO_ESCAPE +#endif + +#if defined(JSON_HEDLEY_UNREACHABLE) + #undef JSON_HEDLEY_UNREACHABLE +#endif +#if defined(JSON_HEDLEY_UNREACHABLE_RETURN) + #undef JSON_HEDLEY_UNREACHABLE_RETURN +#endif +#if defined(JSON_HEDLEY_ASSUME) + #undef JSON_HEDLEY_ASSUME +#endif +#if \ + JSON_HEDLEY_MSVC_VERSION_CHECK(13,10,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_ASSUME(expr) __assume(expr) +#elif JSON_HEDLEY_HAS_BUILTIN(__builtin_assume) + #define JSON_HEDLEY_ASSUME(expr) __builtin_assume(expr) +#elif \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,2,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(4,0,0) + #if defined(__cplusplus) + #define JSON_HEDLEY_ASSUME(expr) std::_nassert(expr) + #else + #define JSON_HEDLEY_ASSUME(expr) _nassert(expr) + #endif +#endif +#if \ + (JSON_HEDLEY_HAS_BUILTIN(__builtin_unreachable) && (!defined(JSON_HEDLEY_ARM_VERSION))) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,5,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(18,10,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(13,1,5) || \ + JSON_HEDLEY_CRAY_VERSION_CHECK(10,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_UNREACHABLE() __builtin_unreachable() +#elif defined(JSON_HEDLEY_ASSUME) + #define JSON_HEDLEY_UNREACHABLE() JSON_HEDLEY_ASSUME(0) +#endif +#if !defined(JSON_HEDLEY_ASSUME) + #if defined(JSON_HEDLEY_UNREACHABLE) + #define JSON_HEDLEY_ASSUME(expr) JSON_HEDLEY_STATIC_CAST(void, ((expr) ? 1 : (JSON_HEDLEY_UNREACHABLE(), 1))) + #else + #define JSON_HEDLEY_ASSUME(expr) JSON_HEDLEY_STATIC_CAST(void, expr) + #endif +#endif +#if defined(JSON_HEDLEY_UNREACHABLE) + #if \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,2,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(4,0,0) + #define JSON_HEDLEY_UNREACHABLE_RETURN(value) return (JSON_HEDLEY_STATIC_CAST(void, JSON_HEDLEY_ASSUME(0)), (value)) + #else + #define JSON_HEDLEY_UNREACHABLE_RETURN(value) JSON_HEDLEY_UNREACHABLE() + #endif +#else + #define JSON_HEDLEY_UNREACHABLE_RETURN(value) return (value) +#endif +#if !defined(JSON_HEDLEY_UNREACHABLE) + #define JSON_HEDLEY_UNREACHABLE() JSON_HEDLEY_ASSUME(0) +#endif + +JSON_HEDLEY_DIAGNOSTIC_PUSH +#if JSON_HEDLEY_HAS_WARNING("-Wpedantic") + #pragma clang diagnostic ignored "-Wpedantic" +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wc++98-compat-pedantic") && defined(__cplusplus) + #pragma clang diagnostic ignored "-Wc++98-compat-pedantic" +#endif +#if JSON_HEDLEY_GCC_HAS_WARNING("-Wvariadic-macros",4,0,0) + #if defined(__clang__) + #pragma clang diagnostic ignored "-Wvariadic-macros" + #elif defined(JSON_HEDLEY_GCC_VERSION) + #pragma GCC diagnostic ignored "-Wvariadic-macros" + #endif +#endif +#if defined(JSON_HEDLEY_NON_NULL) + #undef JSON_HEDLEY_NON_NULL +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(nonnull) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,3,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) + #define JSON_HEDLEY_NON_NULL(...) __attribute__((__nonnull__(__VA_ARGS__))) +#else + #define JSON_HEDLEY_NON_NULL(...) +#endif +JSON_HEDLEY_DIAGNOSTIC_POP + +#if defined(JSON_HEDLEY_PRINTF_FORMAT) + #undef JSON_HEDLEY_PRINTF_FORMAT +#endif +#if defined(__MINGW32__) && JSON_HEDLEY_GCC_HAS_ATTRIBUTE(format,4,4,0) && !defined(__USE_MINGW_ANSI_STDIO) + #define JSON_HEDLEY_PRINTF_FORMAT(string_idx,first_to_check) __attribute__((__format__(ms_printf, string_idx, first_to_check))) +#elif defined(__MINGW32__) && JSON_HEDLEY_GCC_HAS_ATTRIBUTE(format,4,4,0) && defined(__USE_MINGW_ANSI_STDIO) + #define JSON_HEDLEY_PRINTF_FORMAT(string_idx,first_to_check) __attribute__((__format__(gnu_printf, string_idx, first_to_check))) +#elif \ + JSON_HEDLEY_HAS_ATTRIBUTE(format) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(5,6,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_PRINTF_FORMAT(string_idx,first_to_check) __attribute__((__format__(__printf__, string_idx, first_to_check))) +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(6,0,0) + #define JSON_HEDLEY_PRINTF_FORMAT(string_idx,first_to_check) __declspec(vaformat(printf,string_idx,first_to_check)) +#else + #define JSON_HEDLEY_PRINTF_FORMAT(string_idx,first_to_check) +#endif + +#if defined(JSON_HEDLEY_CONSTEXPR) + #undef JSON_HEDLEY_CONSTEXPR +#endif +#if defined(__cplusplus) + #if __cplusplus >= 201103L + #define JSON_HEDLEY_CONSTEXPR JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(constexpr) + #endif +#endif +#if !defined(JSON_HEDLEY_CONSTEXPR) + #define JSON_HEDLEY_CONSTEXPR +#endif + +#if defined(JSON_HEDLEY_PREDICT) + #undef JSON_HEDLEY_PREDICT +#endif +#if defined(JSON_HEDLEY_LIKELY) + #undef JSON_HEDLEY_LIKELY +#endif +#if defined(JSON_HEDLEY_UNLIKELY) + #undef JSON_HEDLEY_UNLIKELY +#endif +#if defined(JSON_HEDLEY_UNPREDICTABLE) + #undef JSON_HEDLEY_UNPREDICTABLE +#endif +#if JSON_HEDLEY_HAS_BUILTIN(__builtin_unpredictable) + #define JSON_HEDLEY_UNPREDICTABLE(expr) __builtin_unpredictable((expr)) +#endif +#if \ + (JSON_HEDLEY_HAS_BUILTIN(__builtin_expect_with_probability) && !defined(JSON_HEDLEY_PGI_VERSION)) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(9,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) +# define JSON_HEDLEY_PREDICT(expr, value, probability) __builtin_expect_with_probability( (expr), (value), (probability)) +# define JSON_HEDLEY_PREDICT_TRUE(expr, probability) __builtin_expect_with_probability(!!(expr), 1 , (probability)) +# define JSON_HEDLEY_PREDICT_FALSE(expr, probability) __builtin_expect_with_probability(!!(expr), 0 , (probability)) +# define JSON_HEDLEY_LIKELY(expr) __builtin_expect (!!(expr), 1 ) +# define JSON_HEDLEY_UNLIKELY(expr) __builtin_expect (!!(expr), 0 ) +#elif \ + (JSON_HEDLEY_HAS_BUILTIN(__builtin_expect) && !defined(JSON_HEDLEY_INTEL_CL_VERSION)) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + (JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,15,0) && defined(__cplusplus)) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,7,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,1,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(6,1,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_TINYC_VERSION_CHECK(0,9,27) || \ + JSON_HEDLEY_CRAY_VERSION_CHECK(8,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) +# define JSON_HEDLEY_PREDICT(expr, expected, probability) \ + (((probability) >= 0.9) ? __builtin_expect((expr), (expected)) : (JSON_HEDLEY_STATIC_CAST(void, expected), (expr))) +# define JSON_HEDLEY_PREDICT_TRUE(expr, probability) \ + (__extension__ ({ \ + double hedley_probability_ = (probability); \ + ((hedley_probability_ >= 0.9) ? __builtin_expect(!!(expr), 1) : ((hedley_probability_ <= 0.1) ? __builtin_expect(!!(expr), 0) : !!(expr))); \ + })) +# define JSON_HEDLEY_PREDICT_FALSE(expr, probability) \ + (__extension__ ({ \ + double hedley_probability_ = (probability); \ + ((hedley_probability_ >= 0.9) ? __builtin_expect(!!(expr), 0) : ((hedley_probability_ <= 0.1) ? __builtin_expect(!!(expr), 1) : !!(expr))); \ + })) +# define JSON_HEDLEY_LIKELY(expr) __builtin_expect(!!(expr), 1) +# define JSON_HEDLEY_UNLIKELY(expr) __builtin_expect(!!(expr), 0) +#else +# define JSON_HEDLEY_PREDICT(expr, expected, probability) (JSON_HEDLEY_STATIC_CAST(void, expected), (expr)) +# define JSON_HEDLEY_PREDICT_TRUE(expr, probability) (!!(expr)) +# define JSON_HEDLEY_PREDICT_FALSE(expr, probability) (!!(expr)) +# define JSON_HEDLEY_LIKELY(expr) (!!(expr)) +# define JSON_HEDLEY_UNLIKELY(expr) (!!(expr)) +#endif +#if !defined(JSON_HEDLEY_UNPREDICTABLE) + #define JSON_HEDLEY_UNPREDICTABLE(expr) JSON_HEDLEY_PREDICT(expr, 1, 0.5) +#endif + +#if defined(JSON_HEDLEY_MALLOC) + #undef JSON_HEDLEY_MALLOC +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(malloc) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(12,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_MALLOC __attribute__((__malloc__)) +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,10,0) + #define JSON_HEDLEY_MALLOC _Pragma("returns_new_memory") +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(14,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_MALLOC __declspec(restrict) +#else + #define JSON_HEDLEY_MALLOC +#endif + +#if defined(JSON_HEDLEY_PURE) + #undef JSON_HEDLEY_PURE +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(pure) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(2,96,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) +# define JSON_HEDLEY_PURE __attribute__((__pure__)) +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,10,0) +# define JSON_HEDLEY_PURE _Pragma("does_not_write_global_data") +#elif defined(__cplusplus) && \ + ( \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(2,0,1) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(4,0,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) \ + ) +# define JSON_HEDLEY_PURE _Pragma("FUNC_IS_PURE;") +#else +# define JSON_HEDLEY_PURE +#endif + +#if defined(JSON_HEDLEY_CONST) + #undef JSON_HEDLEY_CONST +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(const) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(2,5,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_CONST __attribute__((__const__)) +#elif \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,10,0) + #define JSON_HEDLEY_CONST _Pragma("no_side_effect") +#else + #define JSON_HEDLEY_CONST JSON_HEDLEY_PURE +#endif + +#if defined(JSON_HEDLEY_RESTRICT) + #undef JSON_HEDLEY_RESTRICT +#endif +#if defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) && !defined(__cplusplus) + #define JSON_HEDLEY_RESTRICT restrict +#elif \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_MSVC_VERSION_CHECK(14,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(17,10,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,2,4) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,1,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + (JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,14,0) && defined(__cplusplus)) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) || \ + defined(__clang__) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_RESTRICT __restrict +#elif JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,3,0) && !defined(__cplusplus) + #define JSON_HEDLEY_RESTRICT _Restrict +#else + #define JSON_HEDLEY_RESTRICT +#endif + +#if defined(JSON_HEDLEY_INLINE) + #undef JSON_HEDLEY_INLINE +#endif +#if \ + (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L)) || \ + (defined(__cplusplus) && (__cplusplus >= 199711L)) + #define JSON_HEDLEY_INLINE inline +#elif \ + defined(JSON_HEDLEY_GCC_VERSION) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(6,2,0) + #define JSON_HEDLEY_INLINE __inline__ +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(12,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,1,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(3,1,0) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,2,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(8,0,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_INLINE __inline +#else + #define JSON_HEDLEY_INLINE +#endif + +#if defined(JSON_HEDLEY_ALWAYS_INLINE) + #undef JSON_HEDLEY_ALWAYS_INLINE +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(always_inline) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,10,0) +# define JSON_HEDLEY_ALWAYS_INLINE __attribute__((__always_inline__)) JSON_HEDLEY_INLINE +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(12,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) +# define JSON_HEDLEY_ALWAYS_INLINE __forceinline +#elif defined(__cplusplus) && \ + ( \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(6,1,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) \ + ) +# define JSON_HEDLEY_ALWAYS_INLINE _Pragma("FUNC_ALWAYS_INLINE;") +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) +# define JSON_HEDLEY_ALWAYS_INLINE _Pragma("inline=forced") +#else +# define JSON_HEDLEY_ALWAYS_INLINE JSON_HEDLEY_INLINE +#endif + +#if defined(JSON_HEDLEY_NEVER_INLINE) + #undef JSON_HEDLEY_NEVER_INLINE +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(noinline) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(10,1,0) || \ + JSON_HEDLEY_TI_VERSION_CHECK(15,12,0) || \ + (JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(4,8,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_ARMCL_VERSION_CHECK(5,2,0) || \ + (JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL2000_VERSION_CHECK(6,4,0) || \ + (JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,0,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(4,3,0) || \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) || \ + JSON_HEDLEY_TI_CL7X_VERSION_CHECK(1,2,0) || \ + JSON_HEDLEY_TI_CLPRU_VERSION_CHECK(2,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) || \ + JSON_HEDLEY_IAR_VERSION_CHECK(8,10,0) + #define JSON_HEDLEY_NEVER_INLINE __attribute__((__noinline__)) +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(13,10,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_NEVER_INLINE __declspec(noinline) +#elif JSON_HEDLEY_PGI_VERSION_CHECK(10,2,0) + #define JSON_HEDLEY_NEVER_INLINE _Pragma("noinline") +#elif JSON_HEDLEY_TI_CL6X_VERSION_CHECK(6,0,0) && defined(__cplusplus) + #define JSON_HEDLEY_NEVER_INLINE _Pragma("FUNC_CANNOT_INLINE;") +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) + #define JSON_HEDLEY_NEVER_INLINE _Pragma("inline=never") +#elif JSON_HEDLEY_COMPCERT_VERSION_CHECK(3,2,0) + #define JSON_HEDLEY_NEVER_INLINE __attribute((noinline)) +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(9,0,0) + #define JSON_HEDLEY_NEVER_INLINE __declspec(noinline) +#else + #define JSON_HEDLEY_NEVER_INLINE +#endif + +#if defined(JSON_HEDLEY_PRIVATE) + #undef JSON_HEDLEY_PRIVATE +#endif +#if defined(JSON_HEDLEY_PUBLIC) + #undef JSON_HEDLEY_PUBLIC +#endif +#if defined(JSON_HEDLEY_IMPORT) + #undef JSON_HEDLEY_IMPORT +#endif +#if defined(_WIN32) || defined(__CYGWIN__) +# define JSON_HEDLEY_PRIVATE +# define JSON_HEDLEY_PUBLIC __declspec(dllexport) +# define JSON_HEDLEY_IMPORT __declspec(dllimport) +#else +# if \ + JSON_HEDLEY_HAS_ATTRIBUTE(visibility) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,3,0) || \ + JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,11,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(13,1,0) || \ + ( \ + defined(__TI_EABI__) && \ + ( \ + (JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,2,0) && defined(__TI_GNU_ATTRIBUTE_SUPPORT__)) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(7,5,0) \ + ) \ + ) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) +# define JSON_HEDLEY_PRIVATE __attribute__((__visibility__("hidden"))) +# define JSON_HEDLEY_PUBLIC __attribute__((__visibility__("default"))) +# else +# define JSON_HEDLEY_PRIVATE +# define JSON_HEDLEY_PUBLIC +# endif +# define JSON_HEDLEY_IMPORT extern +#endif + +#if defined(JSON_HEDLEY_NO_THROW) + #undef JSON_HEDLEY_NO_THROW +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(nothrow) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,3,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_NO_THROW __attribute__((__nothrow__)) +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(13,1,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) + #define JSON_HEDLEY_NO_THROW __declspec(nothrow) +#else + #define JSON_HEDLEY_NO_THROW +#endif + +#if defined(JSON_HEDLEY_FALL_THROUGH) + #undef JSON_HEDLEY_FALL_THROUGH +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(fallthrough) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(7,0,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_FALL_THROUGH __attribute__((__fallthrough__)) +#elif JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS(clang,fallthrough) + #define JSON_HEDLEY_FALL_THROUGH JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[clang::fallthrough]]) +#elif JSON_HEDLEY_HAS_CPP_ATTRIBUTE(fallthrough) + #define JSON_HEDLEY_FALL_THROUGH JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_([[fallthrough]]) +#elif defined(__fallthrough) /* SAL */ + #define JSON_HEDLEY_FALL_THROUGH __fallthrough +#else + #define JSON_HEDLEY_FALL_THROUGH +#endif + +#if defined(JSON_HEDLEY_RETURNS_NON_NULL) + #undef JSON_HEDLEY_RETURNS_NON_NULL +#endif +#if \ + JSON_HEDLEY_HAS_ATTRIBUTE(returns_nonnull) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,9,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_RETURNS_NON_NULL __attribute__((__returns_nonnull__)) +#elif defined(_Ret_notnull_) /* SAL */ + #define JSON_HEDLEY_RETURNS_NON_NULL _Ret_notnull_ +#else + #define JSON_HEDLEY_RETURNS_NON_NULL +#endif + +#if defined(JSON_HEDLEY_ARRAY_PARAM) + #undef JSON_HEDLEY_ARRAY_PARAM +#endif +#if \ + defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) && \ + !defined(__STDC_NO_VLA__) && \ + !defined(__cplusplus) && \ + !defined(JSON_HEDLEY_PGI_VERSION) && \ + !defined(JSON_HEDLEY_TINYC_VERSION) + #define JSON_HEDLEY_ARRAY_PARAM(name) (name) +#else + #define JSON_HEDLEY_ARRAY_PARAM(name) +#endif + +#if defined(JSON_HEDLEY_IS_CONSTANT) + #undef JSON_HEDLEY_IS_CONSTANT +#endif +#if defined(JSON_HEDLEY_REQUIRE_CONSTEXPR) + #undef JSON_HEDLEY_REQUIRE_CONSTEXPR +#endif +/* JSON_HEDLEY_IS_CONSTEXPR_ is for + HEDLEY INTERNAL USE ONLY. API subject to change without notice. */ +#if defined(JSON_HEDLEY_IS_CONSTEXPR_) + #undef JSON_HEDLEY_IS_CONSTEXPR_ +#endif +#if \ + JSON_HEDLEY_HAS_BUILTIN(__builtin_constant_p) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,4,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_TINYC_VERSION_CHECK(0,9,19) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(4,1,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(13,1,0) || \ + JSON_HEDLEY_TI_CL6X_VERSION_CHECK(6,1,0) || \ + (JSON_HEDLEY_SUNPRO_VERSION_CHECK(5,10,0) && !defined(__cplusplus)) || \ + JSON_HEDLEY_CRAY_VERSION_CHECK(8,1,0) || \ + JSON_HEDLEY_MCST_LCC_VERSION_CHECK(1,25,10) + #define JSON_HEDLEY_IS_CONSTANT(expr) __builtin_constant_p(expr) +#endif +#if !defined(__cplusplus) +# if \ + JSON_HEDLEY_HAS_BUILTIN(__builtin_types_compatible_p) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(3,4,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(13,1,0) || \ + JSON_HEDLEY_CRAY_VERSION_CHECK(8,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(5,4,0) || \ + JSON_HEDLEY_TINYC_VERSION_CHECK(0,9,24) +#if defined(__INTPTR_TYPE__) + #define JSON_HEDLEY_IS_CONSTEXPR_(expr) __builtin_types_compatible_p(__typeof__((1 ? (void*) ((__INTPTR_TYPE__) ((expr) * 0)) : (int*) 0)), int*) +#else + #include + #define JSON_HEDLEY_IS_CONSTEXPR_(expr) __builtin_types_compatible_p(__typeof__((1 ? (void*) ((intptr_t) ((expr) * 0)) : (int*) 0)), int*) +#endif +# elif \ + ( \ + defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L) && \ + !defined(JSON_HEDLEY_SUNPRO_VERSION) && \ + !defined(JSON_HEDLEY_PGI_VERSION) && \ + !defined(JSON_HEDLEY_IAR_VERSION)) || \ + (JSON_HEDLEY_HAS_EXTENSION(c_generic_selections) && !defined(JSON_HEDLEY_IAR_VERSION)) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,9,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(17,0,0) || \ + JSON_HEDLEY_IBM_VERSION_CHECK(12,1,0) || \ + JSON_HEDLEY_ARM_VERSION_CHECK(5,3,0) +#if defined(__INTPTR_TYPE__) + #define JSON_HEDLEY_IS_CONSTEXPR_(expr) _Generic((1 ? (void*) ((__INTPTR_TYPE__) ((expr) * 0)) : (int*) 0), int*: 1, void*: 0) +#else + #include + #define JSON_HEDLEY_IS_CONSTEXPR_(expr) _Generic((1 ? (void*) ((intptr_t) * 0) : (int*) 0), int*: 1, void*: 0) +#endif +# elif \ + defined(JSON_HEDLEY_GCC_VERSION) || \ + defined(JSON_HEDLEY_INTEL_VERSION) || \ + defined(JSON_HEDLEY_TINYC_VERSION) || \ + defined(JSON_HEDLEY_TI_ARMCL_VERSION) || \ + JSON_HEDLEY_TI_CL430_VERSION_CHECK(18,12,0) || \ + defined(JSON_HEDLEY_TI_CL2000_VERSION) || \ + defined(JSON_HEDLEY_TI_CL6X_VERSION) || \ + defined(JSON_HEDLEY_TI_CL7X_VERSION) || \ + defined(JSON_HEDLEY_TI_CLPRU_VERSION) || \ + defined(__clang__) +# define JSON_HEDLEY_IS_CONSTEXPR_(expr) ( \ + sizeof(void) != \ + sizeof(*( \ + 1 ? \ + ((void*) ((expr) * 0L) ) : \ +((struct { char v[sizeof(void) * 2]; } *) 1) \ + ) \ + ) \ + ) +# endif +#endif +#if defined(JSON_HEDLEY_IS_CONSTEXPR_) + #if !defined(JSON_HEDLEY_IS_CONSTANT) + #define JSON_HEDLEY_IS_CONSTANT(expr) JSON_HEDLEY_IS_CONSTEXPR_(expr) + #endif + #define JSON_HEDLEY_REQUIRE_CONSTEXPR(expr) (JSON_HEDLEY_IS_CONSTEXPR_(expr) ? (expr) : (-1)) +#else + #if !defined(JSON_HEDLEY_IS_CONSTANT) + #define JSON_HEDLEY_IS_CONSTANT(expr) (0) + #endif + #define JSON_HEDLEY_REQUIRE_CONSTEXPR(expr) (expr) +#endif + +#if defined(JSON_HEDLEY_BEGIN_C_DECLS) + #undef JSON_HEDLEY_BEGIN_C_DECLS +#endif +#if defined(JSON_HEDLEY_END_C_DECLS) + #undef JSON_HEDLEY_END_C_DECLS +#endif +#if defined(JSON_HEDLEY_C_DECL) + #undef JSON_HEDLEY_C_DECL +#endif +#if defined(__cplusplus) + #define JSON_HEDLEY_BEGIN_C_DECLS extern "C" { + #define JSON_HEDLEY_END_C_DECLS } + #define JSON_HEDLEY_C_DECL extern "C" +#else + #define JSON_HEDLEY_BEGIN_C_DECLS + #define JSON_HEDLEY_END_C_DECLS + #define JSON_HEDLEY_C_DECL +#endif + +#if defined(JSON_HEDLEY_STATIC_ASSERT) + #undef JSON_HEDLEY_STATIC_ASSERT +#endif +#if \ + !defined(__cplusplus) && ( \ + (defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L)) || \ + (JSON_HEDLEY_HAS_FEATURE(c_static_assert) && !defined(JSON_HEDLEY_INTEL_CL_VERSION)) || \ + JSON_HEDLEY_GCC_VERSION_CHECK(6,0,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) || \ + defined(_Static_assert) \ + ) +# define JSON_HEDLEY_STATIC_ASSERT(expr, message) _Static_assert(expr, message) +#elif \ + (defined(__cplusplus) && (__cplusplus >= 201103L)) || \ + JSON_HEDLEY_MSVC_VERSION_CHECK(16,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) +# define JSON_HEDLEY_STATIC_ASSERT(expr, message) JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(static_assert(expr, message)) +#else +# define JSON_HEDLEY_STATIC_ASSERT(expr, message) +#endif + +#if defined(JSON_HEDLEY_NULL) + #undef JSON_HEDLEY_NULL +#endif +#if defined(__cplusplus) + #if __cplusplus >= 201103L + #define JSON_HEDLEY_NULL JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_(nullptr) + #elif defined(NULL) + #define JSON_HEDLEY_NULL NULL + #else + #define JSON_HEDLEY_NULL JSON_HEDLEY_STATIC_CAST(void*, 0) + #endif +#elif defined(NULL) + #define JSON_HEDLEY_NULL NULL +#else + #define JSON_HEDLEY_NULL ((void*) 0) +#endif + +#if defined(JSON_HEDLEY_MESSAGE) + #undef JSON_HEDLEY_MESSAGE +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wunknown-pragmas") +# define JSON_HEDLEY_MESSAGE(msg) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS \ + JSON_HEDLEY_PRAGMA(message msg) \ + JSON_HEDLEY_DIAGNOSTIC_POP +#elif \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,4,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) +# define JSON_HEDLEY_MESSAGE(msg) JSON_HEDLEY_PRAGMA(message msg) +#elif JSON_HEDLEY_CRAY_VERSION_CHECK(5,0,0) +# define JSON_HEDLEY_MESSAGE(msg) JSON_HEDLEY_PRAGMA(_CRI message msg) +#elif JSON_HEDLEY_IAR_VERSION_CHECK(8,0,0) +# define JSON_HEDLEY_MESSAGE(msg) JSON_HEDLEY_PRAGMA(message(msg)) +#elif JSON_HEDLEY_PELLES_VERSION_CHECK(2,0,0) +# define JSON_HEDLEY_MESSAGE(msg) JSON_HEDLEY_PRAGMA(message(msg)) +#else +# define JSON_HEDLEY_MESSAGE(msg) +#endif + +#if defined(JSON_HEDLEY_WARNING) + #undef JSON_HEDLEY_WARNING +#endif +#if JSON_HEDLEY_HAS_WARNING("-Wunknown-pragmas") +# define JSON_HEDLEY_WARNING(msg) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS \ + JSON_HEDLEY_PRAGMA(clang warning msg) \ + JSON_HEDLEY_DIAGNOSTIC_POP +#elif \ + JSON_HEDLEY_GCC_VERSION_CHECK(4,8,0) || \ + JSON_HEDLEY_PGI_VERSION_CHECK(18,4,0) || \ + JSON_HEDLEY_INTEL_VERSION_CHECK(13,0,0) +# define JSON_HEDLEY_WARNING(msg) JSON_HEDLEY_PRAGMA(GCC warning msg) +#elif \ + JSON_HEDLEY_MSVC_VERSION_CHECK(15,0,0) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) +# define JSON_HEDLEY_WARNING(msg) JSON_HEDLEY_PRAGMA(message(msg)) +#else +# define JSON_HEDLEY_WARNING(msg) JSON_HEDLEY_MESSAGE(msg) +#endif + +#if defined(JSON_HEDLEY_REQUIRE) + #undef JSON_HEDLEY_REQUIRE +#endif +#if defined(JSON_HEDLEY_REQUIRE_MSG) + #undef JSON_HEDLEY_REQUIRE_MSG +#endif +#if JSON_HEDLEY_HAS_ATTRIBUTE(diagnose_if) +# if JSON_HEDLEY_HAS_WARNING("-Wgcc-compat") +# define JSON_HEDLEY_REQUIRE(expr) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wgcc-compat\"") \ + __attribute__((diagnose_if(!(expr), #expr, "error"))) \ + JSON_HEDLEY_DIAGNOSTIC_POP +# define JSON_HEDLEY_REQUIRE_MSG(expr,msg) \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("clang diagnostic ignored \"-Wgcc-compat\"") \ + __attribute__((diagnose_if(!(expr), msg, "error"))) \ + JSON_HEDLEY_DIAGNOSTIC_POP +# else +# define JSON_HEDLEY_REQUIRE(expr) __attribute__((diagnose_if(!(expr), #expr, "error"))) +# define JSON_HEDLEY_REQUIRE_MSG(expr,msg) __attribute__((diagnose_if(!(expr), msg, "error"))) +# endif +#else +# define JSON_HEDLEY_REQUIRE(expr) +# define JSON_HEDLEY_REQUIRE_MSG(expr,msg) +#endif + +#if defined(JSON_HEDLEY_FLAGS) + #undef JSON_HEDLEY_FLAGS +#endif +#if JSON_HEDLEY_HAS_ATTRIBUTE(flag_enum) && (!defined(__cplusplus) || JSON_HEDLEY_HAS_WARNING("-Wbitfield-enum-conversion")) + #define JSON_HEDLEY_FLAGS __attribute__((__flag_enum__)) +#else + #define JSON_HEDLEY_FLAGS +#endif + +#if defined(JSON_HEDLEY_FLAGS_CAST) + #undef JSON_HEDLEY_FLAGS_CAST +#endif +#if JSON_HEDLEY_INTEL_VERSION_CHECK(19,0,0) +# define JSON_HEDLEY_FLAGS_CAST(T, expr) (__extension__ ({ \ + JSON_HEDLEY_DIAGNOSTIC_PUSH \ + _Pragma("warning(disable:188)") \ + ((T) (expr)); \ + JSON_HEDLEY_DIAGNOSTIC_POP \ + })) +#else +# define JSON_HEDLEY_FLAGS_CAST(T, expr) JSON_HEDLEY_STATIC_CAST(T, expr) +#endif + +#if defined(JSON_HEDLEY_EMPTY_BASES) + #undef JSON_HEDLEY_EMPTY_BASES +#endif +#if \ + (JSON_HEDLEY_MSVC_VERSION_CHECK(19,0,23918) && !JSON_HEDLEY_MSVC_VERSION_CHECK(20,0,0)) || \ + JSON_HEDLEY_INTEL_CL_VERSION_CHECK(2021,1,0) + #define JSON_HEDLEY_EMPTY_BASES __declspec(empty_bases) +#else + #define JSON_HEDLEY_EMPTY_BASES +#endif + +/* Remaining macros are deprecated. */ + +#if defined(JSON_HEDLEY_GCC_NOT_CLANG_VERSION_CHECK) + #undef JSON_HEDLEY_GCC_NOT_CLANG_VERSION_CHECK +#endif +#if defined(__clang__) + #define JSON_HEDLEY_GCC_NOT_CLANG_VERSION_CHECK(major,minor,patch) (0) +#else + #define JSON_HEDLEY_GCC_NOT_CLANG_VERSION_CHECK(major,minor,patch) JSON_HEDLEY_GCC_VERSION_CHECK(major,minor,patch) +#endif + +#if defined(JSON_HEDLEY_CLANG_HAS_ATTRIBUTE) + #undef JSON_HEDLEY_CLANG_HAS_ATTRIBUTE +#endif +#define JSON_HEDLEY_CLANG_HAS_ATTRIBUTE(attribute) JSON_HEDLEY_HAS_ATTRIBUTE(attribute) + +#if defined(JSON_HEDLEY_CLANG_HAS_CPP_ATTRIBUTE) + #undef JSON_HEDLEY_CLANG_HAS_CPP_ATTRIBUTE +#endif +#define JSON_HEDLEY_CLANG_HAS_CPP_ATTRIBUTE(attribute) JSON_HEDLEY_HAS_CPP_ATTRIBUTE(attribute) + +#if defined(JSON_HEDLEY_CLANG_HAS_BUILTIN) + #undef JSON_HEDLEY_CLANG_HAS_BUILTIN +#endif +#define JSON_HEDLEY_CLANG_HAS_BUILTIN(builtin) JSON_HEDLEY_HAS_BUILTIN(builtin) + +#if defined(JSON_HEDLEY_CLANG_HAS_FEATURE) + #undef JSON_HEDLEY_CLANG_HAS_FEATURE +#endif +#define JSON_HEDLEY_CLANG_HAS_FEATURE(feature) JSON_HEDLEY_HAS_FEATURE(feature) + +#if defined(JSON_HEDLEY_CLANG_HAS_EXTENSION) + #undef JSON_HEDLEY_CLANG_HAS_EXTENSION +#endif +#define JSON_HEDLEY_CLANG_HAS_EXTENSION(extension) JSON_HEDLEY_HAS_EXTENSION(extension) + +#if defined(JSON_HEDLEY_CLANG_HAS_DECLSPEC_DECLSPEC_ATTRIBUTE) + #undef JSON_HEDLEY_CLANG_HAS_DECLSPEC_DECLSPEC_ATTRIBUTE +#endif +#define JSON_HEDLEY_CLANG_HAS_DECLSPEC_ATTRIBUTE(attribute) JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE(attribute) + +#if defined(JSON_HEDLEY_CLANG_HAS_WARNING) + #undef JSON_HEDLEY_CLANG_HAS_WARNING +#endif +#define JSON_HEDLEY_CLANG_HAS_WARNING(warning) JSON_HEDLEY_HAS_WARNING(warning) + +#endif /* !defined(JSON_HEDLEY_VERSION) || (JSON_HEDLEY_VERSION < X) */ + +// #include + + +#include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template struct make_void +{ + using type = void; +}; +template using void_t = typename make_void::type; +} // namespace detail +} // namespace nlohmann + + +// https://en.cppreference.com/w/cpp/experimental/is_detected +namespace nlohmann +{ +namespace detail +{ +struct nonesuch +{ + nonesuch() = delete; + ~nonesuch() = delete; + nonesuch(nonesuch const&) = delete; + nonesuch(nonesuch const&&) = delete; + void operator=(nonesuch const&) = delete; + void operator=(nonesuch&&) = delete; +}; + +template class Op, + class... Args> +struct detector +{ + using value_t = std::false_type; + using type = Default; +}; + +template class Op, class... Args> +struct detector>, Op, Args...> +{ + using value_t = std::true_type; + using type = Op; +}; + +template class Op, class... Args> +using is_detected = typename detector::value_t; + +template class Op, class... Args> +struct is_detected_lazy : is_detected { }; + +template class Op, class... Args> +using detected_t = typename detector::type; + +template class Op, class... Args> +using detected_or = detector; + +template class Op, class... Args> +using detected_or_t = typename detected_or::type; + +template class Op, class... Args> +using is_detected_exact = std::is_same>; + +template class Op, class... Args> +using is_detected_convertible = + std::is_convertible, To>; +} // namespace detail +} // namespace nlohmann + + +// This file contains all internal macro definitions +// You MUST include macro_unscope.hpp at the end of json.hpp to undef all of them + +// exclude unsupported compilers +#if !defined(JSON_SKIP_UNSUPPORTED_COMPILER_CHECK) + #if defined(__clang__) + #if (__clang_major__ * 10000 + __clang_minor__ * 100 + __clang_patchlevel__) < 30400 + #error "unsupported Clang version - see https://github.com/nlohmann/json#supported-compilers" + #endif + #elif defined(__GNUC__) && !(defined(__ICC) || defined(__INTEL_COMPILER)) + #if (__GNUC__ * 10000 + __GNUC_MINOR__ * 100 + __GNUC_PATCHLEVEL__) < 40800 + #error "unsupported GCC version - see https://github.com/nlohmann/json#supported-compilers" + #endif + #endif +#endif + +// C++ language standard detection +// if the user manually specified the used c++ version this is skipped +#if !defined(JSON_HAS_CPP_20) && !defined(JSON_HAS_CPP_17) && !defined(JSON_HAS_CPP_14) && !defined(JSON_HAS_CPP_11) + #if (defined(__cplusplus) && __cplusplus >= 202002L) || (defined(_MSVC_LANG) && _MSVC_LANG >= 202002L) + #define JSON_HAS_CPP_20 + #define JSON_HAS_CPP_17 + #define JSON_HAS_CPP_14 + #elif (defined(__cplusplus) && __cplusplus >= 201703L) || (defined(_HAS_CXX17) && _HAS_CXX17 == 1) // fix for issue #464 + #define JSON_HAS_CPP_17 + #define JSON_HAS_CPP_14 + #elif (defined(__cplusplus) && __cplusplus >= 201402L) || (defined(_HAS_CXX14) && _HAS_CXX14 == 1) + #define JSON_HAS_CPP_14 + #endif + // the cpp 11 flag is always specified because it is the minimal required version + #define JSON_HAS_CPP_11 +#endif + +// disable documentation warnings on clang +#if defined(__clang__) + #pragma clang diagnostic push + #pragma clang diagnostic ignored "-Wdocumentation" + #pragma clang diagnostic ignored "-Wdocumentation-unknown-command" +#endif + +// allow to disable exceptions +#if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || defined(_CPPUNWIND)) && !defined(JSON_NOEXCEPTION) + #define JSON_THROW(exception) throw exception + #define JSON_TRY try + #define JSON_CATCH(exception) catch(exception) + #define JSON_INTERNAL_CATCH(exception) catch(exception) +#else + #include + #define JSON_THROW(exception) std::abort() + #define JSON_TRY if(true) + #define JSON_CATCH(exception) if(false) + #define JSON_INTERNAL_CATCH(exception) if(false) +#endif + +// override exception macros +#if defined(JSON_THROW_USER) + #undef JSON_THROW + #define JSON_THROW JSON_THROW_USER +#endif +#if defined(JSON_TRY_USER) + #undef JSON_TRY + #define JSON_TRY JSON_TRY_USER +#endif +#if defined(JSON_CATCH_USER) + #undef JSON_CATCH + #define JSON_CATCH JSON_CATCH_USER + #undef JSON_INTERNAL_CATCH + #define JSON_INTERNAL_CATCH JSON_CATCH_USER +#endif +#if defined(JSON_INTERNAL_CATCH_USER) + #undef JSON_INTERNAL_CATCH + #define JSON_INTERNAL_CATCH JSON_INTERNAL_CATCH_USER +#endif + +// allow to override assert +#if !defined(JSON_ASSERT) + #include // assert + #define JSON_ASSERT(x) assert(x) +#endif + +// allow to access some private functions (needed by the test suite) +#if defined(JSON_TESTS_PRIVATE) + #define JSON_PRIVATE_UNLESS_TESTED public +#else + #define JSON_PRIVATE_UNLESS_TESTED private +#endif + +/*! +@brief macro to briefly define a mapping between an enum and JSON +@def NLOHMANN_JSON_SERIALIZE_ENUM +@since version 3.4.0 +*/ +#define NLOHMANN_JSON_SERIALIZE_ENUM(ENUM_TYPE, ...) \ + template \ + inline void to_json(BasicJsonType& j, const ENUM_TYPE& e) \ + { \ + static_assert(std::is_enum::value, #ENUM_TYPE " must be an enum!"); \ + static const std::pair m[] = __VA_ARGS__; \ + auto it = std::find_if(std::begin(m), std::end(m), \ + [e](const std::pair& ej_pair) -> bool \ + { \ + return ej_pair.first == e; \ + }); \ + j = ((it != std::end(m)) ? it : std::begin(m))->second; \ + } \ + template \ + inline void from_json(const BasicJsonType& j, ENUM_TYPE& e) \ + { \ + static_assert(std::is_enum::value, #ENUM_TYPE " must be an enum!"); \ + static const std::pair m[] = __VA_ARGS__; \ + auto it = std::find_if(std::begin(m), std::end(m), \ + [&j](const std::pair& ej_pair) -> bool \ + { \ + return ej_pair.second == j; \ + }); \ + e = ((it != std::end(m)) ? it : std::begin(m))->first; \ + } + +// Ugly macros to avoid uglier copy-paste when specializing basic_json. They +// may be removed in the future once the class is split. + +#define NLOHMANN_BASIC_JSON_TPL_DECLARATION \ + template class ObjectType, \ + template class ArrayType, \ + class StringType, class BooleanType, class NumberIntegerType, \ + class NumberUnsignedType, class NumberFloatType, \ + template class AllocatorType, \ + template class JSONSerializer, \ + class BinaryType> + +#define NLOHMANN_BASIC_JSON_TPL \ + basic_json + +// Macros to simplify conversion from/to types + +#define NLOHMANN_JSON_EXPAND( x ) x +#define NLOHMANN_JSON_GET_MACRO(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, _17, _18, _19, _20, _21, _22, _23, _24, _25, _26, _27, _28, _29, _30, _31, _32, _33, _34, _35, _36, _37, _38, _39, _40, _41, _42, _43, _44, _45, _46, _47, _48, _49, _50, _51, _52, _53, _54, _55, _56, _57, _58, _59, _60, _61, _62, _63, _64, NAME,...) NAME +#define NLOHMANN_JSON_PASTE(...) NLOHMANN_JSON_EXPAND(NLOHMANN_JSON_GET_MACRO(__VA_ARGS__, \ + NLOHMANN_JSON_PASTE64, \ + NLOHMANN_JSON_PASTE63, \ + NLOHMANN_JSON_PASTE62, \ + NLOHMANN_JSON_PASTE61, \ + NLOHMANN_JSON_PASTE60, \ + NLOHMANN_JSON_PASTE59, \ + NLOHMANN_JSON_PASTE58, \ + NLOHMANN_JSON_PASTE57, \ + NLOHMANN_JSON_PASTE56, \ + NLOHMANN_JSON_PASTE55, \ + NLOHMANN_JSON_PASTE54, \ + NLOHMANN_JSON_PASTE53, \ + NLOHMANN_JSON_PASTE52, \ + NLOHMANN_JSON_PASTE51, \ + NLOHMANN_JSON_PASTE50, \ + NLOHMANN_JSON_PASTE49, \ + NLOHMANN_JSON_PASTE48, \ + NLOHMANN_JSON_PASTE47, \ + NLOHMANN_JSON_PASTE46, \ + NLOHMANN_JSON_PASTE45, \ + NLOHMANN_JSON_PASTE44, \ + NLOHMANN_JSON_PASTE43, \ + NLOHMANN_JSON_PASTE42, \ + NLOHMANN_JSON_PASTE41, \ + NLOHMANN_JSON_PASTE40, \ + NLOHMANN_JSON_PASTE39, \ + NLOHMANN_JSON_PASTE38, \ + NLOHMANN_JSON_PASTE37, \ + NLOHMANN_JSON_PASTE36, \ + NLOHMANN_JSON_PASTE35, \ + NLOHMANN_JSON_PASTE34, \ + NLOHMANN_JSON_PASTE33, \ + NLOHMANN_JSON_PASTE32, \ + NLOHMANN_JSON_PASTE31, \ + NLOHMANN_JSON_PASTE30, \ + NLOHMANN_JSON_PASTE29, \ + NLOHMANN_JSON_PASTE28, \ + NLOHMANN_JSON_PASTE27, \ + NLOHMANN_JSON_PASTE26, \ + NLOHMANN_JSON_PASTE25, \ + NLOHMANN_JSON_PASTE24, \ + NLOHMANN_JSON_PASTE23, \ + NLOHMANN_JSON_PASTE22, \ + NLOHMANN_JSON_PASTE21, \ + NLOHMANN_JSON_PASTE20, \ + NLOHMANN_JSON_PASTE19, \ + NLOHMANN_JSON_PASTE18, \ + NLOHMANN_JSON_PASTE17, \ + NLOHMANN_JSON_PASTE16, \ + NLOHMANN_JSON_PASTE15, \ + NLOHMANN_JSON_PASTE14, \ + NLOHMANN_JSON_PASTE13, \ + NLOHMANN_JSON_PASTE12, \ + NLOHMANN_JSON_PASTE11, \ + NLOHMANN_JSON_PASTE10, \ + NLOHMANN_JSON_PASTE9, \ + NLOHMANN_JSON_PASTE8, \ + NLOHMANN_JSON_PASTE7, \ + NLOHMANN_JSON_PASTE6, \ + NLOHMANN_JSON_PASTE5, \ + NLOHMANN_JSON_PASTE4, \ + NLOHMANN_JSON_PASTE3, \ + NLOHMANN_JSON_PASTE2, \ + NLOHMANN_JSON_PASTE1)(__VA_ARGS__)) +#define NLOHMANN_JSON_PASTE2(func, v1) func(v1) +#define NLOHMANN_JSON_PASTE3(func, v1, v2) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE2(func, v2) +#define NLOHMANN_JSON_PASTE4(func, v1, v2, v3) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE3(func, v2, v3) +#define NLOHMANN_JSON_PASTE5(func, v1, v2, v3, v4) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE4(func, v2, v3, v4) +#define NLOHMANN_JSON_PASTE6(func, v1, v2, v3, v4, v5) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE5(func, v2, v3, v4, v5) +#define NLOHMANN_JSON_PASTE7(func, v1, v2, v3, v4, v5, v6) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE6(func, v2, v3, v4, v5, v6) +#define NLOHMANN_JSON_PASTE8(func, v1, v2, v3, v4, v5, v6, v7) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE7(func, v2, v3, v4, v5, v6, v7) +#define NLOHMANN_JSON_PASTE9(func, v1, v2, v3, v4, v5, v6, v7, v8) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE8(func, v2, v3, v4, v5, v6, v7, v8) +#define NLOHMANN_JSON_PASTE10(func, v1, v2, v3, v4, v5, v6, v7, v8, v9) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE9(func, v2, v3, v4, v5, v6, v7, v8, v9) +#define NLOHMANN_JSON_PASTE11(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE10(func, v2, v3, v4, v5, v6, v7, v8, v9, v10) +#define NLOHMANN_JSON_PASTE12(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE11(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11) +#define NLOHMANN_JSON_PASTE13(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE12(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12) +#define NLOHMANN_JSON_PASTE14(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE13(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13) +#define NLOHMANN_JSON_PASTE15(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE14(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14) +#define NLOHMANN_JSON_PASTE16(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE15(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15) +#define NLOHMANN_JSON_PASTE17(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE16(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16) +#define NLOHMANN_JSON_PASTE18(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE17(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17) +#define NLOHMANN_JSON_PASTE19(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE18(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18) +#define NLOHMANN_JSON_PASTE20(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE19(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19) +#define NLOHMANN_JSON_PASTE21(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE20(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20) +#define NLOHMANN_JSON_PASTE22(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE21(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21) +#define NLOHMANN_JSON_PASTE23(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE22(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22) +#define NLOHMANN_JSON_PASTE24(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE23(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23) +#define NLOHMANN_JSON_PASTE25(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE24(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24) +#define NLOHMANN_JSON_PASTE26(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE25(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25) +#define NLOHMANN_JSON_PASTE27(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE26(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26) +#define NLOHMANN_JSON_PASTE28(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE27(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27) +#define NLOHMANN_JSON_PASTE29(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE28(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28) +#define NLOHMANN_JSON_PASTE30(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE29(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29) +#define NLOHMANN_JSON_PASTE31(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE30(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30) +#define NLOHMANN_JSON_PASTE32(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE31(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31) +#define NLOHMANN_JSON_PASTE33(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE32(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32) +#define NLOHMANN_JSON_PASTE34(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE33(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33) +#define NLOHMANN_JSON_PASTE35(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE34(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34) +#define NLOHMANN_JSON_PASTE36(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE35(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35) +#define NLOHMANN_JSON_PASTE37(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE36(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36) +#define NLOHMANN_JSON_PASTE38(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE37(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37) +#define NLOHMANN_JSON_PASTE39(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE38(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38) +#define NLOHMANN_JSON_PASTE40(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE39(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39) +#define NLOHMANN_JSON_PASTE41(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE40(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40) +#define NLOHMANN_JSON_PASTE42(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE41(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41) +#define NLOHMANN_JSON_PASTE43(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE42(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42) +#define NLOHMANN_JSON_PASTE44(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE43(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43) +#define NLOHMANN_JSON_PASTE45(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE44(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44) +#define NLOHMANN_JSON_PASTE46(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE45(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45) +#define NLOHMANN_JSON_PASTE47(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE46(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46) +#define NLOHMANN_JSON_PASTE48(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE47(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47) +#define NLOHMANN_JSON_PASTE49(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE48(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48) +#define NLOHMANN_JSON_PASTE50(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE49(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49) +#define NLOHMANN_JSON_PASTE51(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE50(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50) +#define NLOHMANN_JSON_PASTE52(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE51(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51) +#define NLOHMANN_JSON_PASTE53(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE52(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52) +#define NLOHMANN_JSON_PASTE54(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE53(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53) +#define NLOHMANN_JSON_PASTE55(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE54(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54) +#define NLOHMANN_JSON_PASTE56(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE55(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55) +#define NLOHMANN_JSON_PASTE57(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE56(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56) +#define NLOHMANN_JSON_PASTE58(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE57(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57) +#define NLOHMANN_JSON_PASTE59(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE58(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58) +#define NLOHMANN_JSON_PASTE60(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE59(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59) +#define NLOHMANN_JSON_PASTE61(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE60(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60) +#define NLOHMANN_JSON_PASTE62(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE61(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61) +#define NLOHMANN_JSON_PASTE63(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61, v62) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE62(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61, v62) +#define NLOHMANN_JSON_PASTE64(func, v1, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61, v62, v63) NLOHMANN_JSON_PASTE2(func, v1) NLOHMANN_JSON_PASTE63(func, v2, v3, v4, v5, v6, v7, v8, v9, v10, v11, v12, v13, v14, v15, v16, v17, v18, v19, v20, v21, v22, v23, v24, v25, v26, v27, v28, v29, v30, v31, v32, v33, v34, v35, v36, v37, v38, v39, v40, v41, v42, v43, v44, v45, v46, v47, v48, v49, v50, v51, v52, v53, v54, v55, v56, v57, v58, v59, v60, v61, v62, v63) + +#define NLOHMANN_JSON_TO(v1) nlohmann_json_j[#v1] = nlohmann_json_t.v1; +#define NLOHMANN_JSON_FROM(v1) nlohmann_json_j.at(#v1).get_to(nlohmann_json_t.v1); + +/*! +@brief macro +@def NLOHMANN_DEFINE_TYPE_INTRUSIVE +@since version 3.9.0 +*/ +#define NLOHMANN_DEFINE_TYPE_INTRUSIVE(Type, ...) \ + friend void to_json(nlohmann::json& nlohmann_json_j, const Type& nlohmann_json_t) { NLOHMANN_JSON_EXPAND(NLOHMANN_JSON_PASTE(NLOHMANN_JSON_TO, __VA_ARGS__)) } \ + friend void from_json(const nlohmann::json& nlohmann_json_j, Type& nlohmann_json_t) { NLOHMANN_JSON_EXPAND(NLOHMANN_JSON_PASTE(NLOHMANN_JSON_FROM, __VA_ARGS__)) } + +/*! +@brief macro +@def NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE +@since version 3.9.0 +*/ +#define NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE(Type, ...) \ + inline void to_json(nlohmann::json& nlohmann_json_j, const Type& nlohmann_json_t) { NLOHMANN_JSON_EXPAND(NLOHMANN_JSON_PASTE(NLOHMANN_JSON_TO, __VA_ARGS__)) } \ + inline void from_json(const nlohmann::json& nlohmann_json_j, Type& nlohmann_json_t) { NLOHMANN_JSON_EXPAND(NLOHMANN_JSON_PASTE(NLOHMANN_JSON_FROM, __VA_ARGS__)) } + + +// inspired from https://stackoverflow.com/a/26745591 +// allows to call any std function as if (e.g. with begin): +// using std::begin; begin(x); +// +// it allows using the detected idiom to retrieve the return type +// of such an expression +#define NLOHMANN_CAN_CALL_STD_FUNC_IMPL(std_name) \ + namespace detail { \ + using std::std_name; \ + \ + template \ + using result_of_##std_name = decltype(std_name(std::declval()...)); \ + } \ + \ + namespace detail2 { \ + struct std_name##_tag \ + { \ + }; \ + \ + template \ + std_name##_tag std_name(T&&...); \ + \ + template \ + using result_of_##std_name = decltype(std_name(std::declval()...)); \ + \ + template \ + struct would_call_std_##std_name \ + { \ + static constexpr auto const value = ::nlohmann::detail:: \ + is_detected_exact::value; \ + }; \ + } /* namespace detail2 */ \ + \ + template \ + struct would_call_std_##std_name : detail2::would_call_std_##std_name \ + { \ + } + +#ifndef JSON_USE_IMPLICIT_CONVERSIONS + #define JSON_USE_IMPLICIT_CONVERSIONS 1 +#endif + +#if JSON_USE_IMPLICIT_CONVERSIONS + #define JSON_EXPLICIT +#else + #define JSON_EXPLICIT explicit +#endif + +#ifndef JSON_DIAGNOSTICS + #define JSON_DIAGNOSTICS 0 +#endif + + +namespace nlohmann +{ +namespace detail +{ + +/*! +@brief replace all occurrences of a substring by another string + +@param[in,out] s the string to manipulate; changed so that all + occurrences of @a f are replaced with @a t +@param[in] f the substring to replace with @a t +@param[in] t the string to replace @a f + +@pre The search string @a f must not be empty. **This precondition is +enforced with an assertion.** + +@since version 2.0.0 +*/ +inline void replace_substring(std::string& s, const std::string& f, + const std::string& t) +{ + JSON_ASSERT(!f.empty()); + for (auto pos = s.find(f); // find first occurrence of f + pos != std::string::npos; // make sure f was found + s.replace(pos, f.size(), t), // replace with t, and + pos = s.find(f, pos + t.size())) // find next occurrence of f + {} +} + +/*! + * @brief string escaping as described in RFC 6901 (Sect. 4) + * @param[in] s string to escape + * @return escaped string + * + * Note the order of escaping "~" to "~0" and "/" to "~1" is important. + */ +inline std::string escape(std::string s) +{ + replace_substring(s, "~", "~0"); + replace_substring(s, "/", "~1"); + return s; +} + +/*! + * @brief string unescaping as described in RFC 6901 (Sect. 4) + * @param[in] s string to unescape + * @return unescaped string + * + * Note the order of escaping "~1" to "/" and "~0" to "~" is important. + */ +static void unescape(std::string& s) +{ + replace_substring(s, "~1", "/"); + replace_substring(s, "~0", "~"); +} + +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // size_t + +namespace nlohmann +{ +namespace detail +{ +/// struct to capture the start position of the current token +struct position_t +{ + /// the total number of characters read + std::size_t chars_read_total = 0; + /// the number of characters read in the current line + std::size_t chars_read_current_line = 0; + /// the number of lines read + std::size_t lines_read = 0; + + /// conversion to size_t to preserve SAX interface + constexpr operator size_t() const + { + return chars_read_total; + } +}; + +} // namespace detail +} // namespace nlohmann + +// #include + + +namespace nlohmann +{ +namespace detail +{ +//////////////// +// exceptions // +//////////////// + +/*! +@brief general exception of the @ref basic_json class + +This class is an extension of `std::exception` objects with a member @a id for +exception ids. It is used as the base class for all exceptions thrown by the +@ref basic_json class. This class can hence be used as "wildcard" to catch +exceptions. + +Subclasses: +- @ref parse_error for exceptions indicating a parse error +- @ref invalid_iterator for exceptions indicating errors with iterators +- @ref type_error for exceptions indicating executing a member function with + a wrong type +- @ref out_of_range for exceptions indicating access out of the defined range +- @ref other_error for exceptions indicating other library errors + +@internal +@note To have nothrow-copy-constructible exceptions, we internally use + `std::runtime_error` which can cope with arbitrary-length error messages. + Intermediate strings are built with static functions and then passed to + the actual constructor. +@endinternal + +@liveexample{The following code shows how arbitrary library exceptions can be +caught.,exception} + +@since version 3.0.0 +*/ +class exception : public std::exception +{ + public: + /// returns the explanatory string + const char* what() const noexcept override + { + return m.what(); + } + + /// the id of the exception + const int id; // NOLINT(cppcoreguidelines-non-private-member-variables-in-classes) + + protected: + JSON_HEDLEY_NON_NULL(3) + exception(int id_, const char* what_arg) : id(id_), m(what_arg) {} + + static std::string name(const std::string& ename, int id_) + { + return "[json.exception." + ename + "." + std::to_string(id_) + "] "; + } + + template + static std::string diagnostics(const BasicJsonType& leaf_element) + { +#if JSON_DIAGNOSTICS + std::vector tokens; + for (const auto* current = &leaf_element; current->m_parent != nullptr; current = current->m_parent) + { + switch (current->m_parent->type()) + { + case value_t::array: + { + for (std::size_t i = 0; i < current->m_parent->m_value.array->size(); ++i) + { + if (¤t->m_parent->m_value.array->operator[](i) == current) + { + tokens.emplace_back(std::to_string(i)); + break; + } + } + break; + } + + case value_t::object: + { + for (const auto& element : *current->m_parent->m_value.object) + { + if (&element.second == current) + { + tokens.emplace_back(element.first.c_str()); + break; + } + } + break; + } + + case value_t::null: // LCOV_EXCL_LINE + case value_t::string: // LCOV_EXCL_LINE + case value_t::boolean: // LCOV_EXCL_LINE + case value_t::number_integer: // LCOV_EXCL_LINE + case value_t::number_unsigned: // LCOV_EXCL_LINE + case value_t::number_float: // LCOV_EXCL_LINE + case value_t::binary: // LCOV_EXCL_LINE + case value_t::discarded: // LCOV_EXCL_LINE + default: // LCOV_EXCL_LINE + break; // LCOV_EXCL_LINE + } + } + + if (tokens.empty()) + { + return ""; + } + + return "(" + std::accumulate(tokens.rbegin(), tokens.rend(), std::string{}, + [](const std::string & a, const std::string & b) + { + return a + "/" + detail::escape(b); + }) + ") "; +#else + static_cast(leaf_element); + return ""; +#endif + } + + private: + /// an exception object as storage for error messages + std::runtime_error m; +}; + +/*! +@brief exception indicating a parse error + +This exception is thrown by the library when a parse error occurs. Parse errors +can occur during the deserialization of JSON text, CBOR, MessagePack, as well +as when using JSON Patch. + +Member @a byte holds the byte index of the last read character in the input +file. + +Exceptions have ids 1xx. + +name / id | example message | description +------------------------------ | --------------- | ------------------------- +json.exception.parse_error.101 | parse error at 2: unexpected end of input; expected string literal | This error indicates a syntax error while deserializing a JSON text. The error message describes that an unexpected token (character) was encountered, and the member @a byte indicates the error position. +json.exception.parse_error.102 | parse error at 14: missing or wrong low surrogate | JSON uses the `\uxxxx` format to describe Unicode characters. Code points above above 0xFFFF are split into two `\uxxxx` entries ("surrogate pairs"). This error indicates that the surrogate pair is incomplete or contains an invalid code point. +json.exception.parse_error.103 | parse error: code points above 0x10FFFF are invalid | Unicode supports code points up to 0x10FFFF. Code points above 0x10FFFF are invalid. +json.exception.parse_error.104 | parse error: JSON patch must be an array of objects | [RFC 6902](https://tools.ietf.org/html/rfc6902) requires a JSON Patch document to be a JSON document that represents an array of objects. +json.exception.parse_error.105 | parse error: operation must have string member 'op' | An operation of a JSON Patch document must contain exactly one "op" member, whose value indicates the operation to perform. Its value must be one of "add", "remove", "replace", "move", "copy", or "test"; other values are errors. +json.exception.parse_error.106 | parse error: array index '01' must not begin with '0' | An array index in a JSON Pointer ([RFC 6901](https://tools.ietf.org/html/rfc6901)) may be `0` or any number without a leading `0`. +json.exception.parse_error.107 | parse error: JSON pointer must be empty or begin with '/' - was: 'foo' | A JSON Pointer must be a Unicode string containing a sequence of zero or more reference tokens, each prefixed by a `/` character. +json.exception.parse_error.108 | parse error: escape character '~' must be followed with '0' or '1' | In a JSON Pointer, only `~0` and `~1` are valid escape sequences. +json.exception.parse_error.109 | parse error: array index 'one' is not a number | A JSON Pointer array index must be a number. +json.exception.parse_error.110 | parse error at 1: cannot read 2 bytes from vector | When parsing CBOR or MessagePack, the byte vector ends before the complete value has been read. +json.exception.parse_error.112 | parse error at 1: error reading CBOR; last byte: 0xF8 | Not all types of CBOR or MessagePack are supported. This exception occurs if an unsupported byte was read. +json.exception.parse_error.113 | parse error at 2: expected a CBOR string; last byte: 0x98 | While parsing a map key, a value that is not a string has been read. +json.exception.parse_error.114 | parse error: Unsupported BSON record type 0x0F | The parsing of the corresponding BSON record type is not implemented (yet). +json.exception.parse_error.115 | parse error at byte 5: syntax error while parsing UBJSON high-precision number: invalid number text: 1A | A UBJSON high-precision number could not be parsed. + +@note For an input with n bytes, 1 is the index of the first character and n+1 + is the index of the terminating null byte or the end of file. This also + holds true when reading a byte vector (CBOR or MessagePack). + +@liveexample{The following code shows how a `parse_error` exception can be +caught.,parse_error} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class parse_error : public exception +{ + public: + /*! + @brief create a parse error exception + @param[in] id_ the id of the exception + @param[in] pos the position where the error occurred (or with + chars_read_total=0 if the position cannot be + determined) + @param[in] what_arg the explanatory string + @return parse_error object + */ + template + static parse_error create(int id_, const position_t& pos, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + position_string(pos) + ": " + exception::diagnostics(context) + what_arg; + return parse_error(id_, pos.chars_read_total, w.c_str()); + } + + template + static parse_error create(int id_, std::size_t byte_, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("parse_error", id_) + "parse error" + + (byte_ != 0 ? (" at byte " + std::to_string(byte_)) : "") + + ": " + exception::diagnostics(context) + what_arg; + return parse_error(id_, byte_, w.c_str()); + } + + /*! + @brief byte index of the parse error + + The byte index of the last read character in the input file. + + @note For an input with n bytes, 1 is the index of the first character and + n+1 is the index of the terminating null byte or the end of file. + This also holds true when reading a byte vector (CBOR or MessagePack). + */ + const std::size_t byte; + + private: + parse_error(int id_, std::size_t byte_, const char* what_arg) + : exception(id_, what_arg), byte(byte_) {} + + static std::string position_string(const position_t& pos) + { + return " at line " + std::to_string(pos.lines_read + 1) + + ", column " + std::to_string(pos.chars_read_current_line); + } +}; + +/*! +@brief exception indicating errors with iterators + +This exception is thrown if iterators passed to a library function do not match +the expected semantics. + +Exceptions have ids 2xx. + +name / id | example message | description +----------------------------------- | --------------- | ------------------------- +json.exception.invalid_iterator.201 | iterators are not compatible | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.202 | iterator does not fit current value | In an erase or insert function, the passed iterator @a pos does not belong to the JSON value for which the function was called. It hence does not define a valid position for the deletion/insertion. +json.exception.invalid_iterator.203 | iterators do not fit current value | Either iterator passed to function @ref erase(IteratorType first, IteratorType last) does not belong to the JSON value from which values shall be erased. It hence does not define a valid range to delete values from. +json.exception.invalid_iterator.204 | iterators out of range | When an iterator range for a primitive type (number, boolean, or string) is passed to a constructor or an erase function, this range has to be exactly (@ref begin(), @ref end()), because this is the only way the single stored value is expressed. All other ranges are invalid. +json.exception.invalid_iterator.205 | iterator out of range | When an iterator for a primitive type (number, boolean, or string) is passed to an erase function, the iterator has to be the @ref begin() iterator, because it is the only way to address the stored value. All other iterators are invalid. +json.exception.invalid_iterator.206 | cannot construct with iterators from null | The iterators passed to constructor @ref basic_json(InputIT first, InputIT last) belong to a JSON null value and hence to not define a valid range. +json.exception.invalid_iterator.207 | cannot use key() for non-object iterators | The key() member function can only be used on iterators belonging to a JSON object, because other types do not have a concept of a key. +json.exception.invalid_iterator.208 | cannot use operator[] for object iterators | The operator[] to specify a concrete offset cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.209 | cannot use offsets with object iterators | The offset operators (+, -, +=, -=) cannot be used on iterators belonging to a JSON object, because JSON objects are unordered. +json.exception.invalid_iterator.210 | iterators do not fit | The iterator range passed to the insert function are not compatible, meaning they do not belong to the same container. Therefore, the range (@a first, @a last) is invalid. +json.exception.invalid_iterator.211 | passed iterators may not belong to container | The iterator range passed to the insert function must not be a subrange of the container to insert to. +json.exception.invalid_iterator.212 | cannot compare iterators of different containers | When two iterators are compared, they must belong to the same container. +json.exception.invalid_iterator.213 | cannot compare order of object iterators | The order of object iterators cannot be compared, because JSON objects are unordered. +json.exception.invalid_iterator.214 | cannot get value | Cannot get value for iterator: Either the iterator belongs to a null value or it is an iterator to a primitive type (number, boolean, or string), but the iterator is different to @ref begin(). + +@liveexample{The following code shows how an `invalid_iterator` exception can be +caught.,invalid_iterator} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class invalid_iterator : public exception +{ + public: + template + static invalid_iterator create(int id_, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("invalid_iterator", id_) + exception::diagnostics(context) + what_arg; + return invalid_iterator(id_, w.c_str()); + } + + private: + JSON_HEDLEY_NON_NULL(3) + invalid_iterator(int id_, const char* what_arg) + : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating executing a member function with a wrong type + +This exception is thrown in case of a type error; that is, a library function is +executed on a JSON value whose type does not match the expected semantics. + +Exceptions have ids 3xx. + +name / id | example message | description +----------------------------- | --------------- | ------------------------- +json.exception.type_error.301 | cannot create object from initializer list | To create an object from an initializer list, the initializer list must consist only of a list of pairs whose first element is a string. When this constraint is violated, an array is created instead. +json.exception.type_error.302 | type must be object, but is array | During implicit or explicit value conversion, the JSON type must be compatible to the target type. For instance, a JSON string can only be converted into string types, but not into numbers or boolean types. +json.exception.type_error.303 | incompatible ReferenceType for get_ref, actual type is object | To retrieve a reference to a value stored in a @ref basic_json object with @ref get_ref, the type of the reference must match the value type. For instance, for a JSON array, the @a ReferenceType must be @ref array_t &. +json.exception.type_error.304 | cannot use at() with string | The @ref at() member functions can only be executed for certain JSON types. +json.exception.type_error.305 | cannot use operator[] with string | The @ref operator[] member functions can only be executed for certain JSON types. +json.exception.type_error.306 | cannot use value() with string | The @ref value() member functions can only be executed for certain JSON types. +json.exception.type_error.307 | cannot use erase() with string | The @ref erase() member functions can only be executed for certain JSON types. +json.exception.type_error.308 | cannot use push_back() with string | The @ref push_back() and @ref operator+= member functions can only be executed for certain JSON types. +json.exception.type_error.309 | cannot use insert() with | The @ref insert() member functions can only be executed for certain JSON types. +json.exception.type_error.310 | cannot use swap() with number | The @ref swap() member functions can only be executed for certain JSON types. +json.exception.type_error.311 | cannot use emplace_back() with string | The @ref emplace_back() member function can only be executed for certain JSON types. +json.exception.type_error.312 | cannot use update() with string | The @ref update() member functions can only be executed for certain JSON types. +json.exception.type_error.313 | invalid value to unflatten | The @ref unflatten function converts an object whose keys are JSON Pointers back into an arbitrary nested JSON value. The JSON Pointers must not overlap, because then the resulting value would not be well defined. +json.exception.type_error.314 | only objects can be unflattened | The @ref unflatten function only works for an object whose keys are JSON Pointers. +json.exception.type_error.315 | values in object must be primitive | The @ref unflatten function only works for an object whose keys are JSON Pointers and whose values are primitive. +json.exception.type_error.316 | invalid UTF-8 byte at index 10: 0x7E | The @ref dump function only works with UTF-8 encoded strings; that is, if you assign a `std::string` to a JSON value, make sure it is UTF-8 encoded. | +json.exception.type_error.317 | JSON value cannot be serialized to requested format | The dynamic type of the object cannot be represented in the requested serialization format (e.g. a raw `true` or `null` JSON object cannot be serialized to BSON) | + +@liveexample{The following code shows how a `type_error` exception can be +caught.,type_error} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref out_of_range for exceptions indicating access out of the defined range +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class type_error : public exception +{ + public: + template + static type_error create(int id_, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("type_error", id_) + exception::diagnostics(context) + what_arg; + return type_error(id_, w.c_str()); + } + + private: + JSON_HEDLEY_NON_NULL(3) + type_error(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating access out of the defined range + +This exception is thrown in case a library function is called on an input +parameter that exceeds the expected range, for instance in case of array +indices or nonexisting object keys. + +Exceptions have ids 4xx. + +name / id | example message | description +------------------------------- | --------------- | ------------------------- +json.exception.out_of_range.401 | array index 3 is out of range | The provided array index @a i is larger than @a size-1. +json.exception.out_of_range.402 | array index '-' (3) is out of range | The special array index `-` in a JSON Pointer never describes a valid element of the array, but the index past the end. That is, it can only be used to add elements at this position, but not to read it. +json.exception.out_of_range.403 | key 'foo' not found | The provided key was not found in the JSON object. +json.exception.out_of_range.404 | unresolved reference token 'foo' | A reference token in a JSON Pointer could not be resolved. +json.exception.out_of_range.405 | JSON pointer has no parent | The JSON Patch operations 'remove' and 'add' can not be applied to the root element of the JSON value. +json.exception.out_of_range.406 | number overflow parsing '10E1000' | A parsed number could not be stored as without changing it to NaN or INF. +json.exception.out_of_range.407 | number overflow serializing '9223372036854775808' | UBJSON and BSON only support integer numbers up to 9223372036854775807. (until version 3.8.0) | +json.exception.out_of_range.408 | excessive array size: 8658170730974374167 | The size (following `#`) of an UBJSON array or object exceeds the maximal capacity. | +json.exception.out_of_range.409 | BSON key cannot contain code point U+0000 (at byte 2) | Key identifiers to be serialized to BSON cannot contain code point U+0000, since the key is stored as zero-terminated c-string | + +@liveexample{The following code shows how an `out_of_range` exception can be +caught.,out_of_range} + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref other_error for exceptions indicating other library errors + +@since version 3.0.0 +*/ +class out_of_range : public exception +{ + public: + template + static out_of_range create(int id_, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("out_of_range", id_) + exception::diagnostics(context) + what_arg; + return out_of_range(id_, w.c_str()); + } + + private: + JSON_HEDLEY_NON_NULL(3) + out_of_range(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; + +/*! +@brief exception indicating other library errors + +This exception is thrown in case of errors that cannot be classified with the +other exception types. + +Exceptions have ids 5xx. + +name / id | example message | description +------------------------------ | --------------- | ------------------------- +json.exception.other_error.501 | unsuccessful: {"op":"test","path":"/baz", "value":"bar"} | A JSON Patch operation 'test' failed. The unsuccessful operation is also printed. + +@sa - @ref exception for the base class of the library exceptions +@sa - @ref parse_error for exceptions indicating a parse error +@sa - @ref invalid_iterator for exceptions indicating errors with iterators +@sa - @ref type_error for exceptions indicating executing a member function with + a wrong type +@sa - @ref out_of_range for exceptions indicating access out of the defined range + +@liveexample{The following code shows how an `other_error` exception can be +caught.,other_error} + +@since version 3.0.0 +*/ +class other_error : public exception +{ + public: + template + static other_error create(int id_, const std::string& what_arg, const BasicJsonType& context) + { + std::string w = exception::name("other_error", id_) + exception::diagnostics(context) + what_arg; + return other_error(id_, w.c_str()); + } + + private: + JSON_HEDLEY_NON_NULL(3) + other_error(int id_, const char* what_arg) : exception(id_, what_arg) {} +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // size_t +#include // conditional, enable_if, false_type, integral_constant, is_constructible, is_integral, is_same, remove_cv, remove_reference, true_type +#include // index_sequence, make_index_sequence, index_sequence_for + +// #include + + +namespace nlohmann +{ +namespace detail +{ + +template +using uncvref_t = typename std::remove_cv::type>::type; + +#ifdef JSON_HAS_CPP_14 + +// the following utilities are natively available in C++14 +using std::enable_if_t; +using std::index_sequence; +using std::make_index_sequence; +using std::index_sequence_for; + +#else + +// alias templates to reduce boilerplate +template +using enable_if_t = typename std::enable_if::type; + +// The following code is taken from https://github.com/abseil/abseil-cpp/blob/10cb35e459f5ecca5b2ff107635da0bfa41011b4/absl/utility/utility.h +// which is part of Google Abseil (https://github.com/abseil/abseil-cpp), licensed under the Apache License 2.0. + +//// START OF CODE FROM GOOGLE ABSEIL + +// integer_sequence +// +// Class template representing a compile-time integer sequence. An instantiation +// of `integer_sequence` has a sequence of integers encoded in its +// type through its template arguments (which is a common need when +// working with C++11 variadic templates). `absl::integer_sequence` is designed +// to be a drop-in replacement for C++14's `std::integer_sequence`. +// +// Example: +// +// template< class T, T... Ints > +// void user_function(integer_sequence); +// +// int main() +// { +// // user_function's `T` will be deduced to `int` and `Ints...` +// // will be deduced to `0, 1, 2, 3, 4`. +// user_function(make_integer_sequence()); +// } +template +struct integer_sequence +{ + using value_type = T; + static constexpr std::size_t size() noexcept + { + return sizeof...(Ints); + } +}; + +// index_sequence +// +// A helper template for an `integer_sequence` of `size_t`, +// `absl::index_sequence` is designed to be a drop-in replacement for C++14's +// `std::index_sequence`. +template +using index_sequence = integer_sequence; + +namespace utility_internal +{ + +template +struct Extend; + +// Note that SeqSize == sizeof...(Ints). It's passed explicitly for efficiency. +template +struct Extend, SeqSize, 0> +{ + using type = integer_sequence < T, Ints..., (Ints + SeqSize)... >; +}; + +template +struct Extend, SeqSize, 1> +{ + using type = integer_sequence < T, Ints..., (Ints + SeqSize)..., 2 * SeqSize >; +}; + +// Recursion helper for 'make_integer_sequence'. +// 'Gen::type' is an alias for 'integer_sequence'. +template +struct Gen +{ + using type = + typename Extend < typename Gen < T, N / 2 >::type, N / 2, N % 2 >::type; +}; + +template +struct Gen +{ + using type = integer_sequence; +}; + +} // namespace utility_internal + +// Compile-time sequences of integers + +// make_integer_sequence +// +// This template alias is equivalent to +// `integer_sequence`, and is designed to be a drop-in +// replacement for C++14's `std::make_integer_sequence`. +template +using make_integer_sequence = typename utility_internal::Gen::type; + +// make_index_sequence +// +// This template alias is equivalent to `index_sequence<0, 1, ..., N-1>`, +// and is designed to be a drop-in replacement for C++14's +// `std::make_index_sequence`. +template +using make_index_sequence = make_integer_sequence; + +// index_sequence_for +// +// Converts a typename pack into an index sequence of the same length, and +// is designed to be a drop-in replacement for C++14's +// `std::index_sequence_for()` +template +using index_sequence_for = make_index_sequence; + +//// END OF CODE FROM GOOGLE ABSEIL + +#endif + +// dispatch utility (taken from ranges-v3) +template struct priority_tag : priority_tag < N - 1 > {}; +template<> struct priority_tag<0> {}; + +// taken from ranges-v3 +template +struct static_const +{ + static constexpr T value{}; +}; + +template +constexpr T static_const::value; + +} // namespace detail +} // namespace nlohmann + +// #include + + +namespace nlohmann +{ +namespace detail +{ +// dispatching helper struct +template struct identity_tag {}; +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // numeric_limits +#include // false_type, is_constructible, is_integral, is_same, true_type +#include // declval +#include // tuple + +// #include + + +// #include + + +#include // random_access_iterator_tag + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template +struct iterator_types {}; + +template +struct iterator_types < + It, + void_t> +{ + using difference_type = typename It::difference_type; + using value_type = typename It::value_type; + using pointer = typename It::pointer; + using reference = typename It::reference; + using iterator_category = typename It::iterator_category; +}; + +// This is required as some compilers implement std::iterator_traits in a way that +// doesn't work with SFINAE. See https://github.com/nlohmann/json/issues/1341. +template +struct iterator_traits +{ +}; + +template +struct iterator_traits < T, enable_if_t < !std::is_pointer::value >> + : iterator_types +{ +}; + +template +struct iterator_traits::value>> +{ + using iterator_category = std::random_access_iterator_tag; + using value_type = T; + using difference_type = ptrdiff_t; + using pointer = T*; + using reference = T&; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +// #include + + +namespace nlohmann +{ +NLOHMANN_CAN_CALL_STD_FUNC_IMPL(begin); +} // namespace nlohmann + +// #include + + +// #include + + +namespace nlohmann +{ +NLOHMANN_CAN_CALL_STD_FUNC_IMPL(end); +} // namespace nlohmann + +// #include + +// #include + +// #include +#ifndef INCLUDE_NLOHMANN_JSON_FWD_HPP_ +#define INCLUDE_NLOHMANN_JSON_FWD_HPP_ + +#include // int64_t, uint64_t +#include // map +#include // allocator +#include // string +#include // vector + +/*! +@brief namespace for Niels Lohmann +@see https://github.com/nlohmann +@since version 1.0.0 +*/ +namespace nlohmann +{ +/*! +@brief default JSONSerializer template argument + +This serializer ignores the template arguments and uses ADL +([argument-dependent lookup](https://en.cppreference.com/w/cpp/language/adl)) +for serialization. +*/ +template +struct adl_serializer; + +template class ObjectType = + std::map, + template class ArrayType = std::vector, + class StringType = std::string, class BooleanType = bool, + class NumberIntegerType = std::int64_t, + class NumberUnsignedType = std::uint64_t, + class NumberFloatType = double, + template class AllocatorType = std::allocator, + template class JSONSerializer = + adl_serializer, + class BinaryType = std::vector> +class basic_json; + +/*! +@brief JSON Pointer + +A JSON pointer defines a string syntax for identifying a specific value +within a JSON document. It can be used with functions `at` and +`operator[]`. Furthermore, JSON pointers are the base for JSON patches. + +@sa [RFC 6901](https://tools.ietf.org/html/rfc6901) + +@since version 2.0.0 +*/ +template +class json_pointer; + +/*! +@brief default JSON class + +This type is the default specialization of the @ref basic_json class which +uses the standard template types. + +@since version 1.0.0 +*/ +using json = basic_json<>; + +template +struct ordered_map; + +/*! +@brief ordered JSON class + +This type preserves the insertion order of object keys. + +@since version 3.9.0 +*/ +using ordered_json = basic_json; + +} // namespace nlohmann + +#endif // INCLUDE_NLOHMANN_JSON_FWD_HPP_ + + +namespace nlohmann +{ +/*! +@brief detail namespace with internal helper functions + +This namespace collects functions that should not be exposed, +implementations of some @ref basic_json methods, and meta-programming helpers. + +@since version 2.1.0 +*/ +namespace detail +{ +///////////// +// helpers // +///////////// + +// Note to maintainers: +// +// Every trait in this file expects a non CV-qualified type. +// The only exceptions are in the 'aliases for detected' section +// (i.e. those of the form: decltype(T::member_function(std::declval()))) +// +// In this case, T has to be properly CV-qualified to constraint the function arguments +// (e.g. to_json(BasicJsonType&, const T&)) + +template struct is_basic_json : std::false_type {}; + +NLOHMANN_BASIC_JSON_TPL_DECLARATION +struct is_basic_json : std::true_type {}; + +////////////////////// +// json_ref helpers // +////////////////////// + +template +class json_ref; + +template +struct is_json_ref : std::false_type {}; + +template +struct is_json_ref> : std::true_type {}; + +////////////////////////// +// aliases for detected // +////////////////////////// + +template +using mapped_type_t = typename T::mapped_type; + +template +using key_type_t = typename T::key_type; + +template +using value_type_t = typename T::value_type; + +template +using difference_type_t = typename T::difference_type; + +template +using pointer_t = typename T::pointer; + +template +using reference_t = typename T::reference; + +template +using iterator_category_t = typename T::iterator_category; + +template +using to_json_function = decltype(T::to_json(std::declval()...)); + +template +using from_json_function = decltype(T::from_json(std::declval()...)); + +template +using get_template_function = decltype(std::declval().template get()); + +// trait checking if JSONSerializer::from_json(json const&, udt&) exists +template +struct has_from_json : std::false_type {}; + +// trait checking if j.get is valid +// use this trait instead of std::is_constructible or std::is_convertible, +// both rely on, or make use of implicit conversions, and thus fail when T +// has several constructors/operator= (see https://github.com/nlohmann/json/issues/958) +template +struct is_getable +{ + static constexpr bool value = is_detected::value; +}; + +template +struct has_from_json < BasicJsonType, T, enable_if_t < !is_basic_json::value >> +{ + using serializer = typename BasicJsonType::template json_serializer; + + static constexpr bool value = + is_detected_exact::value; +}; + +// This trait checks if JSONSerializer::from_json(json const&) exists +// this overload is used for non-default-constructible user-defined-types +template +struct has_non_default_from_json : std::false_type {}; + +template +struct has_non_default_from_json < BasicJsonType, T, enable_if_t < !is_basic_json::value >> +{ + using serializer = typename BasicJsonType::template json_serializer; + + static constexpr bool value = + is_detected_exact::value; +}; + +// This trait checks if BasicJsonType::json_serializer::to_json exists +// Do not evaluate the trait when T is a basic_json type, to avoid template instantiation infinite recursion. +template +struct has_to_json : std::false_type {}; + +template +struct has_to_json < BasicJsonType, T, enable_if_t < !is_basic_json::value >> +{ + using serializer = typename BasicJsonType::template json_serializer; + + static constexpr bool value = + is_detected_exact::value; +}; + + +/////////////////// +// is_ functions // +/////////////////// + +// https://en.cppreference.com/w/cpp/types/conjunction +template struct conjunction : std::true_type { }; +template struct conjunction : B1 { }; +template +struct conjunction +: std::conditional, B1>::type {}; + +// https://en.cppreference.com/w/cpp/types/negation +template struct negation : std::integral_constant < bool, !B::value > { }; + +// Reimplementation of is_constructible and is_default_constructible, due to them being broken for +// std::pair and std::tuple until LWG 2367 fix (see https://cplusplus.github.io/LWG/lwg-defects.html#2367). +// This causes compile errors in e.g. clang 3.5 or gcc 4.9. +template +struct is_default_constructible : std::is_default_constructible {}; + +template +struct is_default_constructible> + : conjunction, is_default_constructible> {}; + +template +struct is_default_constructible> + : conjunction, is_default_constructible> {}; + +template +struct is_default_constructible> + : conjunction...> {}; + +template +struct is_default_constructible> + : conjunction...> {}; + + +template +struct is_constructible : std::is_constructible {}; + +template +struct is_constructible> : is_default_constructible> {}; + +template +struct is_constructible> : is_default_constructible> {}; + +template +struct is_constructible> : is_default_constructible> {}; + +template +struct is_constructible> : is_default_constructible> {}; + + +template +struct is_iterator_traits : std::false_type {}; + +template +struct is_iterator_traits> +{ + private: + using traits = iterator_traits; + + public: + static constexpr auto value = + is_detected::value && + is_detected::value && + is_detected::value && + is_detected::value && + is_detected::value; +}; + +template +struct is_range +{ + private: + using t_ref = typename std::add_lvalue_reference::type; + + using iterator = detected_t; + using sentinel = detected_t; + + // to be 100% correct, it should use https://en.cppreference.com/w/cpp/iterator/input_or_output_iterator + // and https://en.cppreference.com/w/cpp/iterator/sentinel_for + // but reimplementing these would be too much work, as a lot of other concepts are used underneath + static constexpr auto is_iterator_begin = + is_iterator_traits>::value; + + public: + static constexpr bool value = !std::is_same::value && !std::is_same::value && is_iterator_begin; +}; + +template +using iterator_t = enable_if_t::value, result_of_begin())>>; + +template +using range_value_t = value_type_t>>; + +// The following implementation of is_complete_type is taken from +// https://blogs.msdn.microsoft.com/vcblog/2015/12/02/partial-support-for-expression-sfinae-in-vs-2015-update-1/ +// and is written by Xiang Fan who agreed to using it in this library. + +template +struct is_complete_type : std::false_type {}; + +template +struct is_complete_type : std::true_type {}; + +template +struct is_compatible_object_type_impl : std::false_type {}; + +template +struct is_compatible_object_type_impl < + BasicJsonType, CompatibleObjectType, + enable_if_t < is_detected::value&& + is_detected::value >> +{ + using object_t = typename BasicJsonType::object_t; + + // macOS's is_constructible does not play well with nonesuch... + static constexpr bool value = + is_constructible::value && + is_constructible::value; +}; + +template +struct is_compatible_object_type + : is_compatible_object_type_impl {}; + +template +struct is_constructible_object_type_impl : std::false_type {}; + +template +struct is_constructible_object_type_impl < + BasicJsonType, ConstructibleObjectType, + enable_if_t < is_detected::value&& + is_detected::value >> +{ + using object_t = typename BasicJsonType::object_t; + + static constexpr bool value = + (is_default_constructible::value && + (std::is_move_assignable::value || + std::is_copy_assignable::value) && + (is_constructible::value && + std::is_same < + typename object_t::mapped_type, + typename ConstructibleObjectType::mapped_type >::value)) || + (has_from_json::value || + has_non_default_from_json < + BasicJsonType, + typename ConstructibleObjectType::mapped_type >::value); +}; + +template +struct is_constructible_object_type + : is_constructible_object_type_impl {}; + +template +struct is_compatible_string_type +{ + static constexpr auto value = + is_constructible::value; +}; + +template +struct is_constructible_string_type +{ + static constexpr auto value = + is_constructible::value; +}; + +template +struct is_compatible_array_type_impl : std::false_type {}; + +template +struct is_compatible_array_type_impl < + BasicJsonType, CompatibleArrayType, + enable_if_t < + is_detected::value&& + is_iterator_traits>>::value&& +// special case for types like std::filesystem::path whose iterator's value_type are themselves +// c.f. https://github.com/nlohmann/json/pull/3073 + !std::is_same>::value >> +{ + static constexpr bool value = + is_constructible>::value; +}; + +template +struct is_compatible_array_type + : is_compatible_array_type_impl {}; + +template +struct is_constructible_array_type_impl : std::false_type {}; + +template +struct is_constructible_array_type_impl < + BasicJsonType, ConstructibleArrayType, + enable_if_t::value >> + : std::true_type {}; + +template +struct is_constructible_array_type_impl < + BasicJsonType, ConstructibleArrayType, + enable_if_t < !std::is_same::value&& + !is_compatible_string_type::value&& + is_default_constructible::value&& +(std::is_move_assignable::value || + std::is_copy_assignable::value)&& +is_detected::value&& +is_iterator_traits>>::value&& +is_detected::value&& +// special case for types like std::filesystem::path whose iterator's value_type are themselves +// c.f. https://github.com/nlohmann/json/pull/3073 +!std::is_same>::value&& + is_complete_type < + detected_t>::value >> +{ + using value_type = range_value_t; + + static constexpr bool value = + std::is_same::value || + has_from_json::value || + has_non_default_from_json < + BasicJsonType, + value_type >::value; +}; + +template +struct is_constructible_array_type + : is_constructible_array_type_impl {}; + +template +struct is_compatible_integer_type_impl : std::false_type {}; + +template +struct is_compatible_integer_type_impl < + RealIntegerType, CompatibleNumberIntegerType, + enable_if_t < std::is_integral::value&& + std::is_integral::value&& + !std::is_same::value >> +{ + // is there an assert somewhere on overflows? + using RealLimits = std::numeric_limits; + using CompatibleLimits = std::numeric_limits; + + static constexpr auto value = + is_constructible::value && + CompatibleLimits::is_integer && + RealLimits::is_signed == CompatibleLimits::is_signed; +}; + +template +struct is_compatible_integer_type + : is_compatible_integer_type_impl {}; + +template +struct is_compatible_type_impl: std::false_type {}; + +template +struct is_compatible_type_impl < + BasicJsonType, CompatibleType, + enable_if_t::value >> +{ + static constexpr bool value = + has_to_json::value; +}; + +template +struct is_compatible_type + : is_compatible_type_impl {}; + +template +struct is_constructible_tuple : std::false_type {}; + +template +struct is_constructible_tuple> : conjunction...> {}; + +// a naive helper to check if a type is an ordered_map (exploits the fact that +// ordered_map inherits capacity() from std::vector) +template +struct is_ordered_map +{ + using one = char; + + struct two + { + char x[2]; // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) + }; + + template static one test( decltype(&C::capacity) ) ; + template static two test(...); + + enum { value = sizeof(test(nullptr)) == sizeof(char) }; // NOLINT(cppcoreguidelines-pro-type-vararg,hicpp-vararg) +}; + +// to avoid useless casts (see https://github.com/nlohmann/json/issues/2893#issuecomment-889152324) +template < typename T, typename U, enable_if_t < !std::is_same::value, int > = 0 > +T conditional_static_cast(U value) +{ + return static_cast(value); +} + +template::value, int> = 0> +T conditional_static_cast(U value) +{ + return value; +} + +} // namespace detail +} // namespace nlohmann + +// #include + + +#ifdef JSON_HAS_CPP_17 + #include +#endif + +namespace nlohmann +{ +namespace detail +{ +template +void from_json(const BasicJsonType& j, typename std::nullptr_t& n) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_null())) + { + JSON_THROW(type_error::create(302, "type must be null, but is " + std::string(j.type_name()), j)); + } + n = nullptr; +} + +// overloads for basic_json template parameters +template < typename BasicJsonType, typename ArithmeticType, + enable_if_t < std::is_arithmetic::value&& + !std::is_same::value, + int > = 0 > +void get_arithmetic_value(const BasicJsonType& j, ArithmeticType& val) +{ + switch (static_cast(j)) + { + case value_t::number_unsigned: + { + val = static_cast(*j.template get_ptr()); + break; + } + case value_t::number_integer: + { + val = static_cast(*j.template get_ptr()); + break; + } + case value_t::number_float: + { + val = static_cast(*j.template get_ptr()); + break; + } + + case value_t::null: + case value_t::object: + case value_t::array: + case value_t::string: + case value_t::boolean: + case value_t::binary: + case value_t::discarded: + default: + JSON_THROW(type_error::create(302, "type must be number, but is " + std::string(j.type_name()), j)); + } +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::boolean_t& b) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_boolean())) + { + JSON_THROW(type_error::create(302, "type must be boolean, but is " + std::string(j.type_name()), j)); + } + b = *j.template get_ptr(); +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::string_t& s) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_string())) + { + JSON_THROW(type_error::create(302, "type must be string, but is " + std::string(j.type_name()), j)); + } + s = *j.template get_ptr(); +} + +template < + typename BasicJsonType, typename ConstructibleStringType, + enable_if_t < + is_constructible_string_type::value&& + !std::is_same::value, + int > = 0 > +void from_json(const BasicJsonType& j, ConstructibleStringType& s) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_string())) + { + JSON_THROW(type_error::create(302, "type must be string, but is " + std::string(j.type_name()), j)); + } + + s = *j.template get_ptr(); +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::number_float_t& val) +{ + get_arithmetic_value(j, val); +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::number_unsigned_t& val) +{ + get_arithmetic_value(j, val); +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::number_integer_t& val) +{ + get_arithmetic_value(j, val); +} + +template::value, int> = 0> +void from_json(const BasicJsonType& j, EnumType& e) +{ + typename std::underlying_type::type val; + get_arithmetic_value(j, val); + e = static_cast(val); +} + +// forward_list doesn't have an insert method +template::value, int> = 0> +void from_json(const BasicJsonType& j, std::forward_list& l) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + l.clear(); + std::transform(j.rbegin(), j.rend(), + std::front_inserter(l), [](const BasicJsonType & i) + { + return i.template get(); + }); +} + +// valarray doesn't have an insert method +template::value, int> = 0> +void from_json(const BasicJsonType& j, std::valarray& l) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + l.resize(j.size()); + std::transform(j.begin(), j.end(), std::begin(l), + [](const BasicJsonType & elem) + { + return elem.template get(); + }); +} + +template +auto from_json(const BasicJsonType& j, T (&arr)[N]) // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) +-> decltype(j.template get(), void()) +{ + for (std::size_t i = 0; i < N; ++i) + { + arr[i] = j.at(i).template get(); + } +} + +template +void from_json_array_impl(const BasicJsonType& j, typename BasicJsonType::array_t& arr, priority_tag<3> /*unused*/) +{ + arr = *j.template get_ptr(); +} + +template +auto from_json_array_impl(const BasicJsonType& j, std::array& arr, + priority_tag<2> /*unused*/) +-> decltype(j.template get(), void()) +{ + for (std::size_t i = 0; i < N; ++i) + { + arr[i] = j.at(i).template get(); + } +} + +template::value, + int> = 0> +auto from_json_array_impl(const BasicJsonType& j, ConstructibleArrayType& arr, priority_tag<1> /*unused*/) +-> decltype( + arr.reserve(std::declval()), + j.template get(), + void()) +{ + using std::end; + + ConstructibleArrayType ret; + ret.reserve(j.size()); + std::transform(j.begin(), j.end(), + std::inserter(ret, end(ret)), [](const BasicJsonType & i) + { + // get() returns *this, this won't call a from_json + // method when value_type is BasicJsonType + return i.template get(); + }); + arr = std::move(ret); +} + +template::value, + int> = 0> +void from_json_array_impl(const BasicJsonType& j, ConstructibleArrayType& arr, + priority_tag<0> /*unused*/) +{ + using std::end; + + ConstructibleArrayType ret; + std::transform( + j.begin(), j.end(), std::inserter(ret, end(ret)), + [](const BasicJsonType & i) + { + // get() returns *this, this won't call a from_json + // method when value_type is BasicJsonType + return i.template get(); + }); + arr = std::move(ret); +} + +template < typename BasicJsonType, typename ConstructibleArrayType, + enable_if_t < + is_constructible_array_type::value&& + !is_constructible_object_type::value&& + !is_constructible_string_type::value&& + !std::is_same::value&& + !is_basic_json::value, + int > = 0 > +auto from_json(const BasicJsonType& j, ConstructibleArrayType& arr) +-> decltype(from_json_array_impl(j, arr, priority_tag<3> {}), +j.template get(), +void()) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + + from_json_array_impl(j, arr, priority_tag<3> {}); +} + +template < typename BasicJsonType, typename T, std::size_t... Idx > +std::array from_json_inplace_array_impl(BasicJsonType&& j, + identity_tag> /*unused*/, index_sequence /*unused*/) +{ + return { { std::forward(j).at(Idx).template get()... } }; +} + +template < typename BasicJsonType, typename T, std::size_t N > +auto from_json(BasicJsonType&& j, identity_tag> tag) +-> decltype(from_json_inplace_array_impl(std::forward(j), tag, make_index_sequence {})) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + + return from_json_inplace_array_impl(std::forward(j), tag, make_index_sequence {}); +} + +template +void from_json(const BasicJsonType& j, typename BasicJsonType::binary_t& bin) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_binary())) + { + JSON_THROW(type_error::create(302, "type must be binary, but is " + std::string(j.type_name()), j)); + } + + bin = *j.template get_ptr(); +} + +template::value, int> = 0> +void from_json(const BasicJsonType& j, ConstructibleObjectType& obj) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_object())) + { + JSON_THROW(type_error::create(302, "type must be object, but is " + std::string(j.type_name()), j)); + } + + ConstructibleObjectType ret; + const auto* inner_object = j.template get_ptr(); + using value_type = typename ConstructibleObjectType::value_type; + std::transform( + inner_object->begin(), inner_object->end(), + std::inserter(ret, ret.begin()), + [](typename BasicJsonType::object_t::value_type const & p) + { + return value_type(p.first, p.second.template get()); + }); + obj = std::move(ret); +} + +// overload for arithmetic types, not chosen for basic_json template arguments +// (BooleanType, etc..); note: Is it really necessary to provide explicit +// overloads for boolean_t etc. in case of a custom BooleanType which is not +// an arithmetic type? +template < typename BasicJsonType, typename ArithmeticType, + enable_if_t < + std::is_arithmetic::value&& + !std::is_same::value&& + !std::is_same::value&& + !std::is_same::value&& + !std::is_same::value, + int > = 0 > +void from_json(const BasicJsonType& j, ArithmeticType& val) +{ + switch (static_cast(j)) + { + case value_t::number_unsigned: + { + val = static_cast(*j.template get_ptr()); + break; + } + case value_t::number_integer: + { + val = static_cast(*j.template get_ptr()); + break; + } + case value_t::number_float: + { + val = static_cast(*j.template get_ptr()); + break; + } + case value_t::boolean: + { + val = static_cast(*j.template get_ptr()); + break; + } + + case value_t::null: + case value_t::object: + case value_t::array: + case value_t::string: + case value_t::binary: + case value_t::discarded: + default: + JSON_THROW(type_error::create(302, "type must be number, but is " + std::string(j.type_name()), j)); + } +} + +template +std::tuple from_json_tuple_impl_base(BasicJsonType&& j, index_sequence /*unused*/) +{ + return std::make_tuple(std::forward(j).at(Idx).template get()...); +} + +template < typename BasicJsonType, class A1, class A2 > +std::pair from_json_tuple_impl(BasicJsonType&& j, identity_tag> /*unused*/, priority_tag<0> /*unused*/) +{ + return {std::forward(j).at(0).template get(), + std::forward(j).at(1).template get()}; +} + +template +void from_json_tuple_impl(BasicJsonType&& j, std::pair& p, priority_tag<1> /*unused*/) +{ + p = from_json_tuple_impl(std::forward(j), identity_tag> {}, priority_tag<0> {}); +} + +template +std::tuple from_json_tuple_impl(BasicJsonType&& j, identity_tag> /*unused*/, priority_tag<2> /*unused*/) +{ + return from_json_tuple_impl_base(std::forward(j), index_sequence_for {}); +} + +template +void from_json_tuple_impl(BasicJsonType&& j, std::tuple& t, priority_tag<3> /*unused*/) +{ + t = from_json_tuple_impl_base(std::forward(j), index_sequence_for {}); +} + +template +auto from_json(BasicJsonType&& j, TupleRelated&& t) +-> decltype(from_json_tuple_impl(std::forward(j), std::forward(t), priority_tag<3> {})) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + + return from_json_tuple_impl(std::forward(j), std::forward(t), priority_tag<3> {}); +} + +template < typename BasicJsonType, typename Key, typename Value, typename Compare, typename Allocator, + typename = enable_if_t < !std::is_constructible < + typename BasicJsonType::string_t, Key >::value >> +void from_json(const BasicJsonType& j, std::map& m) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + m.clear(); + for (const auto& p : j) + { + if (JSON_HEDLEY_UNLIKELY(!p.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(p.type_name()), j)); + } + m.emplace(p.at(0).template get(), p.at(1).template get()); + } +} + +template < typename BasicJsonType, typename Key, typename Value, typename Hash, typename KeyEqual, typename Allocator, + typename = enable_if_t < !std::is_constructible < + typename BasicJsonType::string_t, Key >::value >> +void from_json(const BasicJsonType& j, std::unordered_map& m) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(j.type_name()), j)); + } + m.clear(); + for (const auto& p : j) + { + if (JSON_HEDLEY_UNLIKELY(!p.is_array())) + { + JSON_THROW(type_error::create(302, "type must be array, but is " + std::string(p.type_name()), j)); + } + m.emplace(p.at(0).template get(), p.at(1).template get()); + } +} + +#ifdef JSON_HAS_CPP_17 +template +void from_json(const BasicJsonType& j, std::filesystem::path& p) +{ + if (JSON_HEDLEY_UNLIKELY(!j.is_string())) + { + JSON_THROW(type_error::create(302, "type must be string, but is " + std::string(j.type_name()), j)); + } + p = *j.template get_ptr(); +} +#endif + +struct from_json_fn +{ + template + auto operator()(const BasicJsonType& j, T&& val) const + noexcept(noexcept(from_json(j, std::forward(val)))) + -> decltype(from_json(j, std::forward(val))) + { + return from_json(j, std::forward(val)); + } +}; +} // namespace detail + +/// namespace to hold default `from_json` function +/// to see why this is required: +/// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2015/n4381.html +namespace // NOLINT(cert-dcl59-cpp,fuchsia-header-anon-namespaces,google-build-namespaces) +{ +constexpr const auto& from_json = detail::static_const::value; // NOLINT(misc-definitions-in-headers) +} // namespace +} // namespace nlohmann + +// #include + + +#include // copy +#include // begin, end +#include // string +#include // tuple, get +#include // is_same, is_constructible, is_floating_point, is_enum, underlying_type +#include // move, forward, declval, pair +#include // valarray +#include // vector + +// #include + +// #include + + +#include // size_t +#include // input_iterator_tag +#include // string, to_string +#include // tuple_size, get, tuple_element +#include // move + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template +void int_to_string( string_type& target, std::size_t value ) +{ + // For ADL + using std::to_string; + target = to_string(value); +} +template class iteration_proxy_value +{ + public: + using difference_type = std::ptrdiff_t; + using value_type = iteration_proxy_value; + using pointer = value_type * ; + using reference = value_type & ; + using iterator_category = std::input_iterator_tag; + using string_type = typename std::remove_cv< typename std::remove_reference().key() ) >::type >::type; + + private: + /// the iterator + IteratorType anchor; + /// an index for arrays (used to create key names) + std::size_t array_index = 0; + /// last stringified array index + mutable std::size_t array_index_last = 0; + /// a string representation of the array index + mutable string_type array_index_str = "0"; + /// an empty string (to return a reference for primitive values) + const string_type empty_str{}; + + public: + explicit iteration_proxy_value(IteratorType it) noexcept + : anchor(std::move(it)) + {} + + /// dereference operator (needed for range-based for) + iteration_proxy_value& operator*() + { + return *this; + } + + /// increment operator (needed for range-based for) + iteration_proxy_value& operator++() + { + ++anchor; + ++array_index; + + return *this; + } + + /// equality operator (needed for InputIterator) + bool operator==(const iteration_proxy_value& o) const + { + return anchor == o.anchor; + } + + /// inequality operator (needed for range-based for) + bool operator!=(const iteration_proxy_value& o) const + { + return anchor != o.anchor; + } + + /// return key of the iterator + const string_type& key() const + { + JSON_ASSERT(anchor.m_object != nullptr); + + switch (anchor.m_object->type()) + { + // use integer array index as key + case value_t::array: + { + if (array_index != array_index_last) + { + int_to_string( array_index_str, array_index ); + array_index_last = array_index; + } + return array_index_str; + } + + // use key from the object + case value_t::object: + return anchor.key(); + + // use an empty key for all primitive types + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + return empty_str; + } + } + + /// return value of the iterator + typename IteratorType::reference value() const + { + return anchor.value(); + } +}; + +/// proxy class for the items() function +template class iteration_proxy +{ + private: + /// the container to iterate + typename IteratorType::reference container; + + public: + /// construct iteration proxy from a container + explicit iteration_proxy(typename IteratorType::reference cont) noexcept + : container(cont) {} + + /// return iterator begin (needed for range-based for) + iteration_proxy_value begin() noexcept + { + return iteration_proxy_value(container.begin()); + } + + /// return iterator end (needed for range-based for) + iteration_proxy_value end() noexcept + { + return iteration_proxy_value(container.end()); + } +}; +// Structured Bindings Support +// For further reference see https://blog.tartanllama.xyz/structured-bindings/ +// And see https://github.com/nlohmann/json/pull/1391 +template = 0> +auto get(const nlohmann::detail::iteration_proxy_value& i) -> decltype(i.key()) +{ + return i.key(); +} +// Structured Bindings Support +// For further reference see https://blog.tartanllama.xyz/structured-bindings/ +// And see https://github.com/nlohmann/json/pull/1391 +template = 0> +auto get(const nlohmann::detail::iteration_proxy_value& i) -> decltype(i.value()) +{ + return i.value(); +} +} // namespace detail +} // namespace nlohmann + +// The Addition to the STD Namespace is required to add +// Structured Bindings Support to the iteration_proxy_value class +// For further reference see https://blog.tartanllama.xyz/structured-bindings/ +// And see https://github.com/nlohmann/json/pull/1391 +namespace std +{ +#if defined(__clang__) + // Fix: https://github.com/nlohmann/json/issues/1401 + #pragma clang diagnostic push + #pragma clang diagnostic ignored "-Wmismatched-tags" +#endif +template +class tuple_size<::nlohmann::detail::iteration_proxy_value> + : public std::integral_constant {}; + +template +class tuple_element> +{ + public: + using type = decltype( + get(std::declval < + ::nlohmann::detail::iteration_proxy_value> ())); +}; +#if defined(__clang__) + #pragma clang diagnostic pop +#endif +} // namespace std + +// #include + +// #include + +// #include + + +#ifdef JSON_HAS_CPP_17 + #include +#endif + +namespace nlohmann +{ +namespace detail +{ +////////////////// +// constructors // +////////////////// + +/* + * Note all external_constructor<>::construct functions need to call + * j.m_value.destroy(j.m_type) to avoid a memory leak in case j contains an + * allocated value (e.g., a string). See bug issue + * https://github.com/nlohmann/json/issues/2865 for more information. + */ + +template struct external_constructor; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, typename BasicJsonType::boolean_t b) noexcept + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::boolean; + j.m_value = b; + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, const typename BasicJsonType::string_t& s) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::string; + j.m_value = s; + j.assert_invariant(); + } + + template + static void construct(BasicJsonType& j, typename BasicJsonType::string_t&& s) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::string; + j.m_value = std::move(s); + j.assert_invariant(); + } + + template < typename BasicJsonType, typename CompatibleStringType, + enable_if_t < !std::is_same::value, + int > = 0 > + static void construct(BasicJsonType& j, const CompatibleStringType& str) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::string; + j.m_value.string = j.template create(str); + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, const typename BasicJsonType::binary_t& b) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::binary; + j.m_value = typename BasicJsonType::binary_t(b); + j.assert_invariant(); + } + + template + static void construct(BasicJsonType& j, typename BasicJsonType::binary_t&& b) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::binary; + j.m_value = typename BasicJsonType::binary_t(std::move(b)); + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, typename BasicJsonType::number_float_t val) noexcept + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::number_float; + j.m_value = val; + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, typename BasicJsonType::number_unsigned_t val) noexcept + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::number_unsigned; + j.m_value = val; + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, typename BasicJsonType::number_integer_t val) noexcept + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::number_integer; + j.m_value = val; + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, const typename BasicJsonType::array_t& arr) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::array; + j.m_value = arr; + j.set_parents(); + j.assert_invariant(); + } + + template + static void construct(BasicJsonType& j, typename BasicJsonType::array_t&& arr) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::array; + j.m_value = std::move(arr); + j.set_parents(); + j.assert_invariant(); + } + + template < typename BasicJsonType, typename CompatibleArrayType, + enable_if_t < !std::is_same::value, + int > = 0 > + static void construct(BasicJsonType& j, const CompatibleArrayType& arr) + { + using std::begin; + using std::end; + + j.m_value.destroy(j.m_type); + j.m_type = value_t::array; + j.m_value.array = j.template create(begin(arr), end(arr)); + j.set_parents(); + j.assert_invariant(); + } + + template + static void construct(BasicJsonType& j, const std::vector& arr) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::array; + j.m_value = value_t::array; + j.m_value.array->reserve(arr.size()); + for (const bool x : arr) + { + j.m_value.array->push_back(x); + j.set_parent(j.m_value.array->back()); + } + j.assert_invariant(); + } + + template::value, int> = 0> + static void construct(BasicJsonType& j, const std::valarray& arr) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::array; + j.m_value = value_t::array; + j.m_value.array->resize(arr.size()); + if (arr.size() > 0) + { + std::copy(std::begin(arr), std::end(arr), j.m_value.array->begin()); + } + j.set_parents(); + j.assert_invariant(); + } +}; + +template<> +struct external_constructor +{ + template + static void construct(BasicJsonType& j, const typename BasicJsonType::object_t& obj) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::object; + j.m_value = obj; + j.set_parents(); + j.assert_invariant(); + } + + template + static void construct(BasicJsonType& j, typename BasicJsonType::object_t&& obj) + { + j.m_value.destroy(j.m_type); + j.m_type = value_t::object; + j.m_value = std::move(obj); + j.set_parents(); + j.assert_invariant(); + } + + template < typename BasicJsonType, typename CompatibleObjectType, + enable_if_t < !std::is_same::value, int > = 0 > + static void construct(BasicJsonType& j, const CompatibleObjectType& obj) + { + using std::begin; + using std::end; + + j.m_value.destroy(j.m_type); + j.m_type = value_t::object; + j.m_value.object = j.template create(begin(obj), end(obj)); + j.set_parents(); + j.assert_invariant(); + } +}; + +///////////// +// to_json // +///////////// + +template::value, int> = 0> +void to_json(BasicJsonType& j, T b) noexcept +{ + external_constructor::construct(j, b); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, const CompatibleString& s) +{ + external_constructor::construct(j, s); +} + +template +void to_json(BasicJsonType& j, typename BasicJsonType::string_t&& s) +{ + external_constructor::construct(j, std::move(s)); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, FloatType val) noexcept +{ + external_constructor::construct(j, static_cast(val)); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, CompatibleNumberUnsignedType val) noexcept +{ + external_constructor::construct(j, static_cast(val)); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, CompatibleNumberIntegerType val) noexcept +{ + external_constructor::construct(j, static_cast(val)); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, EnumType e) noexcept +{ + using underlying_type = typename std::underlying_type::type; + external_constructor::construct(j, static_cast(e)); +} + +template +void to_json(BasicJsonType& j, const std::vector& e) +{ + external_constructor::construct(j, e); +} + +template < typename BasicJsonType, typename CompatibleArrayType, + enable_if_t < is_compatible_array_type::value&& + !is_compatible_object_type::value&& + !is_compatible_string_type::value&& + !std::is_same::value&& + !is_basic_json::value, + int > = 0 > +void to_json(BasicJsonType& j, const CompatibleArrayType& arr) +{ + external_constructor::construct(j, arr); +} + +template +void to_json(BasicJsonType& j, const typename BasicJsonType::binary_t& bin) +{ + external_constructor::construct(j, bin); +} + +template::value, int> = 0> +void to_json(BasicJsonType& j, const std::valarray& arr) +{ + external_constructor::construct(j, std::move(arr)); +} + +template +void to_json(BasicJsonType& j, typename BasicJsonType::array_t&& arr) +{ + external_constructor::construct(j, std::move(arr)); +} + +template < typename BasicJsonType, typename CompatibleObjectType, + enable_if_t < is_compatible_object_type::value&& !is_basic_json::value, int > = 0 > +void to_json(BasicJsonType& j, const CompatibleObjectType& obj) +{ + external_constructor::construct(j, obj); +} + +template +void to_json(BasicJsonType& j, typename BasicJsonType::object_t&& obj) +{ + external_constructor::construct(j, std::move(obj)); +} + +template < + typename BasicJsonType, typename T, std::size_t N, + enable_if_t < !std::is_constructible::value, // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) + int > = 0 > +void to_json(BasicJsonType& j, const T(&arr)[N]) // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) +{ + external_constructor::construct(j, arr); +} + +template < typename BasicJsonType, typename T1, typename T2, enable_if_t < std::is_constructible::value&& std::is_constructible::value, int > = 0 > +void to_json(BasicJsonType& j, const std::pair& p) +{ + j = { p.first, p.second }; +} + +// for https://github.com/nlohmann/json/pull/1134 +template>::value, int> = 0> +void to_json(BasicJsonType& j, const T& b) +{ + j = { {b.key(), b.value()} }; +} + +template +void to_json_tuple_impl(BasicJsonType& j, const Tuple& t, index_sequence /*unused*/) +{ + j = { std::get(t)... }; +} + +template::value, int > = 0> +void to_json(BasicJsonType& j, const T& t) +{ + to_json_tuple_impl(j, t, make_index_sequence::value> {}); +} + +#ifdef JSON_HAS_CPP_17 +template +void to_json(BasicJsonType& j, const std::filesystem::path& p) +{ + j = p.string(); +} +#endif + +struct to_json_fn +{ + template + auto operator()(BasicJsonType& j, T&& val) const noexcept(noexcept(to_json(j, std::forward(val)))) + -> decltype(to_json(j, std::forward(val)), void()) + { + return to_json(j, std::forward(val)); + } +}; +} // namespace detail + +/// namespace to hold default `to_json` function +/// to see why this is required: +/// http://www.open-std.org/jtc1/sc22/wg21/docs/papers/2015/n4381.html +namespace // NOLINT(cert-dcl59-cpp,fuchsia-header-anon-namespaces,google-build-namespaces) +{ +constexpr const auto& to_json = detail::static_const::value; // NOLINT(misc-definitions-in-headers) +} // namespace +} // namespace nlohmann + +// #include + +// #include + + +namespace nlohmann +{ + +template +struct adl_serializer +{ + /*! + @brief convert a JSON value to any value type + + This function is usually called by the `get()` function of the + @ref basic_json class (either explicit or via conversion operators). + + @note This function is chosen for default-constructible value types. + + @param[in] j JSON value to read from + @param[in,out] val value to write to + */ + template + static auto from_json(BasicJsonType && j, TargetType& val) noexcept( + noexcept(::nlohmann::from_json(std::forward(j), val))) + -> decltype(::nlohmann::from_json(std::forward(j), val), void()) + { + ::nlohmann::from_json(std::forward(j), val); + } + + /*! + @brief convert a JSON value to any value type + + This function is usually called by the `get()` function of the + @ref basic_json class (either explicit or via conversion operators). + + @note This function is chosen for value types which are not default-constructible. + + @param[in] j JSON value to read from + + @return copy of the JSON value, converted to @a ValueType + */ + template + static auto from_json(BasicJsonType && j) noexcept( + noexcept(::nlohmann::from_json(std::forward(j), detail::identity_tag {}))) + -> decltype(::nlohmann::from_json(std::forward(j), detail::identity_tag {})) + { + return ::nlohmann::from_json(std::forward(j), detail::identity_tag {}); + } + + /*! + @brief convert any value type to a JSON value + + This function is usually called by the constructors of the @ref basic_json + class. + + @param[in,out] j JSON value to write to + @param[in] val value to read from + */ + template + static auto to_json(BasicJsonType& j, TargetType && val) noexcept( + noexcept(::nlohmann::to_json(j, std::forward(val)))) + -> decltype(::nlohmann::to_json(j, std::forward(val)), void()) + { + ::nlohmann::to_json(j, std::forward(val)); + } +}; +} // namespace nlohmann + +// #include + + +#include // uint8_t, uint64_t +#include // tie +#include // move + +namespace nlohmann +{ + +/*! +@brief an internal type for a backed binary type + +This type extends the template parameter @a BinaryType provided to `basic_json` +with a subtype used by BSON and MessagePack. This type exists so that the user +does not have to specify a type themselves with a specific naming scheme in +order to override the binary type. + +@tparam BinaryType container to store bytes (`std::vector` by + default) + +@since version 3.8.0; changed type of subtypes to std::uint64_t in 3.10.0. +*/ +template +class byte_container_with_subtype : public BinaryType +{ + public: + /// the type of the underlying container + using container_type = BinaryType; + /// the type of the subtype + using subtype_type = std::uint64_t; + + byte_container_with_subtype() noexcept(noexcept(container_type())) + : container_type() + {} + + byte_container_with_subtype(const container_type& b) noexcept(noexcept(container_type(b))) + : container_type(b) + {} + + byte_container_with_subtype(container_type&& b) noexcept(noexcept(container_type(std::move(b)))) + : container_type(std::move(b)) + {} + + byte_container_with_subtype(const container_type& b, subtype_type subtype_) noexcept(noexcept(container_type(b))) + : container_type(b) + , m_subtype(subtype_) + , m_has_subtype(true) + {} + + byte_container_with_subtype(container_type&& b, subtype_type subtype_) noexcept(noexcept(container_type(std::move(b)))) + : container_type(std::move(b)) + , m_subtype(subtype_) + , m_has_subtype(true) + {} + + bool operator==(const byte_container_with_subtype& rhs) const + { + return std::tie(static_cast(*this), m_subtype, m_has_subtype) == + std::tie(static_cast(rhs), rhs.m_subtype, rhs.m_has_subtype); + } + + bool operator!=(const byte_container_with_subtype& rhs) const + { + return !(rhs == *this); + } + + /*! + @brief sets the binary subtype + + Sets the binary subtype of the value, also flags a binary JSON value as + having a subtype, which has implications for serialization. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @sa see @ref subtype() -- return the binary subtype + @sa see @ref clear_subtype() -- clears the binary subtype + @sa see @ref has_subtype() -- returns whether or not the binary value has a + subtype + + @since version 3.8.0 + */ + void set_subtype(subtype_type subtype_) noexcept + { + m_subtype = subtype_; + m_has_subtype = true; + } + + /*! + @brief return the binary subtype + + Returns the numerical subtype of the value if it has a subtype. If it does + not have a subtype, this function will return subtype_type(-1) as a sentinel + value. + + @return the numerical subtype of the binary value + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @sa see @ref set_subtype() -- sets the binary subtype + @sa see @ref clear_subtype() -- clears the binary subtype + @sa see @ref has_subtype() -- returns whether or not the binary value has a + subtype + + @since version 3.8.0; fixed return value to properly return + subtype_type(-1) as documented in version 3.10.0 + */ + constexpr subtype_type subtype() const noexcept + { + return m_has_subtype ? m_subtype : subtype_type(-1); + } + + /*! + @brief return whether the value has a subtype + + @return whether the value has a subtype + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @sa see @ref subtype() -- return the binary subtype + @sa see @ref set_subtype() -- sets the binary subtype + @sa see @ref clear_subtype() -- clears the binary subtype + + @since version 3.8.0 + */ + constexpr bool has_subtype() const noexcept + { + return m_has_subtype; + } + + /*! + @brief clears the binary subtype + + Clears the binary subtype and flags the value as not having a subtype, which + has implications for serialization; for instance MessagePack will prefer the + bin family over the ext family. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @sa see @ref subtype() -- return the binary subtype + @sa see @ref set_subtype() -- sets the binary subtype + @sa see @ref has_subtype() -- returns whether or not the binary value has a + subtype + + @since version 3.8.0 + */ + void clear_subtype() noexcept + { + m_subtype = 0; + m_has_subtype = false; + } + + private: + subtype_type m_subtype = 0; + bool m_has_subtype = false; +}; + +} // namespace nlohmann + +// #include + +// #include + +// #include + +// #include + + +#include // uint8_t +#include // size_t +#include // hash + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ + +// boost::hash_combine +inline std::size_t combine(std::size_t seed, std::size_t h) noexcept +{ + seed ^= h + 0x9e3779b9 + (seed << 6U) + (seed >> 2U); + return seed; +} + +/*! +@brief hash a JSON value + +The hash function tries to rely on std::hash where possible. Furthermore, the +type of the JSON value is taken into account to have different hash values for +null, 0, 0U, and false, etc. + +@tparam BasicJsonType basic_json specialization +@param j JSON value to hash +@return hash value of j +*/ +template +std::size_t hash(const BasicJsonType& j) +{ + using string_t = typename BasicJsonType::string_t; + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + + const auto type = static_cast(j.type()); + switch (j.type()) + { + case BasicJsonType::value_t::null: + case BasicJsonType::value_t::discarded: + { + return combine(type, 0); + } + + case BasicJsonType::value_t::object: + { + auto seed = combine(type, j.size()); + for (const auto& element : j.items()) + { + const auto h = std::hash {}(element.key()); + seed = combine(seed, h); + seed = combine(seed, hash(element.value())); + } + return seed; + } + + case BasicJsonType::value_t::array: + { + auto seed = combine(type, j.size()); + for (const auto& element : j) + { + seed = combine(seed, hash(element)); + } + return seed; + } + + case BasicJsonType::value_t::string: + { + const auto h = std::hash {}(j.template get_ref()); + return combine(type, h); + } + + case BasicJsonType::value_t::boolean: + { + const auto h = std::hash {}(j.template get()); + return combine(type, h); + } + + case BasicJsonType::value_t::number_integer: + { + const auto h = std::hash {}(j.template get()); + return combine(type, h); + } + + case BasicJsonType::value_t::number_unsigned: + { + const auto h = std::hash {}(j.template get()); + return combine(type, h); + } + + case BasicJsonType::value_t::number_float: + { + const auto h = std::hash {}(j.template get()); + return combine(type, h); + } + + case BasicJsonType::value_t::binary: + { + auto seed = combine(type, j.get_binary().size()); + const auto h = std::hash {}(j.get_binary().has_subtype()); + seed = combine(seed, h); + seed = combine(seed, static_cast(j.get_binary().subtype())); + for (const auto byte : j.get_binary()) + { + seed = combine(seed, std::hash {}(byte)); + } + return seed; + } + + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + return 0; // LCOV_EXCL_LINE + } +} + +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // generate_n +#include // array +#include // ldexp +#include // size_t +#include // uint8_t, uint16_t, uint32_t, uint64_t +#include // snprintf +#include // memcpy +#include // back_inserter +#include // numeric_limits +#include // char_traits, string +#include // make_pair, move +#include // vector + +// #include + +// #include + + +#include // array +#include // size_t +#include // strlen +#include // begin, end, iterator_traits, random_access_iterator_tag, distance, next +#include // shared_ptr, make_shared, addressof +#include // accumulate +#include // string, char_traits +#include // enable_if, is_base_of, is_pointer, is_integral, remove_pointer +#include // pair, declval + +#ifndef JSON_NO_IO + #include // FILE * + #include // istream +#endif // JSON_NO_IO + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/// the supported input formats +enum class input_format_t { json, cbor, msgpack, ubjson, bson }; + +//////////////////// +// input adapters // +//////////////////// + +#ifndef JSON_NO_IO +/*! +Input adapter for stdio file access. This adapter read only 1 byte and do not use any + buffer. This adapter is a very low level adapter. +*/ +class file_input_adapter +{ + public: + using char_type = char; + + JSON_HEDLEY_NON_NULL(2) + explicit file_input_adapter(std::FILE* f) noexcept + : m_file(f) + {} + + // make class move-only + file_input_adapter(const file_input_adapter&) = delete; + file_input_adapter(file_input_adapter&&) noexcept = default; + file_input_adapter& operator=(const file_input_adapter&) = delete; + file_input_adapter& operator=(file_input_adapter&&) = delete; + ~file_input_adapter() = default; + + std::char_traits::int_type get_character() noexcept + { + return std::fgetc(m_file); + } + + private: + /// the file pointer to read from + std::FILE* m_file; +}; + + +/*! +Input adapter for a (caching) istream. Ignores a UFT Byte Order Mark at +beginning of input. Does not support changing the underlying std::streambuf +in mid-input. Maintains underlying std::istream and std::streambuf to support +subsequent use of standard std::istream operations to process any input +characters following those used in parsing the JSON input. Clears the +std::istream flags; any input errors (e.g., EOF) will be detected by the first +subsequent call for input from the std::istream. +*/ +class input_stream_adapter +{ + public: + using char_type = char; + + ~input_stream_adapter() + { + // clear stream flags; we use underlying streambuf I/O, do not + // maintain ifstream flags, except eof + if (is != nullptr) + { + is->clear(is->rdstate() & std::ios::eofbit); + } + } + + explicit input_stream_adapter(std::istream& i) + : is(&i), sb(i.rdbuf()) + {} + + // delete because of pointer members + input_stream_adapter(const input_stream_adapter&) = delete; + input_stream_adapter& operator=(input_stream_adapter&) = delete; + input_stream_adapter& operator=(input_stream_adapter&&) = delete; + + input_stream_adapter(input_stream_adapter&& rhs) noexcept + : is(rhs.is), sb(rhs.sb) + { + rhs.is = nullptr; + rhs.sb = nullptr; + } + + // std::istream/std::streambuf use std::char_traits::to_int_type, to + // ensure that std::char_traits::eof() and the character 0xFF do not + // end up as the same value, eg. 0xFFFFFFFF. + std::char_traits::int_type get_character() + { + auto res = sb->sbumpc(); + // set eof manually, as we don't use the istream interface. + if (JSON_HEDLEY_UNLIKELY(res == std::char_traits::eof())) + { + is->clear(is->rdstate() | std::ios::eofbit); + } + return res; + } + + private: + /// the associated input stream + std::istream* is = nullptr; + std::streambuf* sb = nullptr; +}; +#endif // JSON_NO_IO + +// General-purpose iterator-based adapter. It might not be as fast as +// theoretically possible for some containers, but it is extremely versatile. +template +class iterator_input_adapter +{ + public: + using char_type = typename std::iterator_traits::value_type; + + iterator_input_adapter(IteratorType first, IteratorType last) + : current(std::move(first)), end(std::move(last)) + {} + + typename std::char_traits::int_type get_character() + { + if (JSON_HEDLEY_LIKELY(current != end)) + { + auto result = std::char_traits::to_int_type(*current); + std::advance(current, 1); + return result; + } + + return std::char_traits::eof(); + } + + private: + IteratorType current; + IteratorType end; + + template + friend struct wide_string_input_helper; + + bool empty() const + { + return current == end; + } +}; + + +template +struct wide_string_input_helper; + +template +struct wide_string_input_helper +{ + // UTF-32 + static void fill_buffer(BaseInputAdapter& input, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) + { + utf8_bytes_index = 0; + + if (JSON_HEDLEY_UNLIKELY(input.empty())) + { + utf8_bytes[0] = std::char_traits::eof(); + utf8_bytes_filled = 1; + } + else + { + // get the current character + const auto wc = input.get_character(); + + // UTF-32 to UTF-8 encoding + if (wc < 0x80) + { + utf8_bytes[0] = static_cast::int_type>(wc); + utf8_bytes_filled = 1; + } + else if (wc <= 0x7FF) + { + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((static_cast(wc) >> 6u) & 0x1Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | (static_cast(wc) & 0x3Fu)); + utf8_bytes_filled = 2; + } + else if (wc <= 0xFFFF) + { + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((static_cast(wc) >> 12u) & 0x0Fu)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((static_cast(wc) >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (static_cast(wc) & 0x3Fu)); + utf8_bytes_filled = 3; + } + else if (wc <= 0x10FFFF) + { + utf8_bytes[0] = static_cast::int_type>(0xF0u | ((static_cast(wc) >> 18u) & 0x07u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((static_cast(wc) >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((static_cast(wc) >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (static_cast(wc) & 0x3Fu)); + utf8_bytes_filled = 4; + } + else + { + // unknown character + utf8_bytes[0] = static_cast::int_type>(wc); + utf8_bytes_filled = 1; + } + } + } +}; + +template +struct wide_string_input_helper +{ + // UTF-16 + static void fill_buffer(BaseInputAdapter& input, + std::array::int_type, 4>& utf8_bytes, + size_t& utf8_bytes_index, + size_t& utf8_bytes_filled) + { + utf8_bytes_index = 0; + + if (JSON_HEDLEY_UNLIKELY(input.empty())) + { + utf8_bytes[0] = std::char_traits::eof(); + utf8_bytes_filled = 1; + } + else + { + // get the current character + const auto wc = input.get_character(); + + // UTF-16 to UTF-8 encoding + if (wc < 0x80) + { + utf8_bytes[0] = static_cast::int_type>(wc); + utf8_bytes_filled = 1; + } + else if (wc <= 0x7FF) + { + utf8_bytes[0] = static_cast::int_type>(0xC0u | ((static_cast(wc) >> 6u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | (static_cast(wc) & 0x3Fu)); + utf8_bytes_filled = 2; + } + else if (0xD800 > wc || wc >= 0xE000) + { + utf8_bytes[0] = static_cast::int_type>(0xE0u | ((static_cast(wc) >> 12u))); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((static_cast(wc) >> 6u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | (static_cast(wc) & 0x3Fu)); + utf8_bytes_filled = 3; + } + else + { + if (JSON_HEDLEY_UNLIKELY(!input.empty())) + { + const auto wc2 = static_cast(input.get_character()); + const auto charcode = 0x10000u + (((static_cast(wc) & 0x3FFu) << 10u) | (wc2 & 0x3FFu)); + utf8_bytes[0] = static_cast::int_type>(0xF0u | (charcode >> 18u)); + utf8_bytes[1] = static_cast::int_type>(0x80u | ((charcode >> 12u) & 0x3Fu)); + utf8_bytes[2] = static_cast::int_type>(0x80u | ((charcode >> 6u) & 0x3Fu)); + utf8_bytes[3] = static_cast::int_type>(0x80u | (charcode & 0x3Fu)); + utf8_bytes_filled = 4; + } + else + { + utf8_bytes[0] = static_cast::int_type>(wc); + utf8_bytes_filled = 1; + } + } + } + } +}; + +// Wraps another input apdater to convert wide character types into individual bytes. +template +class wide_string_input_adapter +{ + public: + using char_type = char; + + wide_string_input_adapter(BaseInputAdapter base) + : base_adapter(base) {} + + typename std::char_traits::int_type get_character() noexcept + { + // check if buffer needs to be filled + if (utf8_bytes_index == utf8_bytes_filled) + { + fill_buffer(); + + JSON_ASSERT(utf8_bytes_filled > 0); + JSON_ASSERT(utf8_bytes_index == 0); + } + + // use buffer + JSON_ASSERT(utf8_bytes_filled > 0); + JSON_ASSERT(utf8_bytes_index < utf8_bytes_filled); + return utf8_bytes[utf8_bytes_index++]; + } + + private: + BaseInputAdapter base_adapter; + + template + void fill_buffer() + { + wide_string_input_helper::fill_buffer(base_adapter, utf8_bytes, utf8_bytes_index, utf8_bytes_filled); + } + + /// a buffer for UTF-8 bytes + std::array::int_type, 4> utf8_bytes = {{0, 0, 0, 0}}; + + /// index to the utf8_codes array for the next valid byte + std::size_t utf8_bytes_index = 0; + /// number of valid bytes in the utf8_codes array + std::size_t utf8_bytes_filled = 0; +}; + + +template +struct iterator_input_adapter_factory +{ + using iterator_type = IteratorType; + using char_type = typename std::iterator_traits::value_type; + using adapter_type = iterator_input_adapter; + + static adapter_type create(IteratorType first, IteratorType last) + { + return adapter_type(std::move(first), std::move(last)); + } +}; + +template +struct is_iterator_of_multibyte +{ + using value_type = typename std::iterator_traits::value_type; + enum + { + value = sizeof(value_type) > 1 + }; +}; + +template +struct iterator_input_adapter_factory::value>> +{ + using iterator_type = IteratorType; + using char_type = typename std::iterator_traits::value_type; + using base_adapter_type = iterator_input_adapter; + using adapter_type = wide_string_input_adapter; + + static adapter_type create(IteratorType first, IteratorType last) + { + return adapter_type(base_adapter_type(std::move(first), std::move(last))); + } +}; + +// General purpose iterator-based input +template +typename iterator_input_adapter_factory::adapter_type input_adapter(IteratorType first, IteratorType last) +{ + using factory_type = iterator_input_adapter_factory; + return factory_type::create(first, last); +} + +// Convenience shorthand from container to iterator +// Enables ADL on begin(container) and end(container) +// Encloses the using declarations in namespace for not to leak them to outside scope + +namespace container_input_adapter_factory_impl +{ + +using std::begin; +using std::end; + +template +struct container_input_adapter_factory {}; + +template +struct container_input_adapter_factory< ContainerType, + void_t()), end(std::declval()))>> + { + using adapter_type = decltype(input_adapter(begin(std::declval()), end(std::declval()))); + + static adapter_type create(const ContainerType& container) +{ + return input_adapter(begin(container), end(container)); +} + }; + +} // namespace container_input_adapter_factory_impl + +template +typename container_input_adapter_factory_impl::container_input_adapter_factory::adapter_type input_adapter(const ContainerType& container) +{ + return container_input_adapter_factory_impl::container_input_adapter_factory::create(container); +} + +#ifndef JSON_NO_IO +// Special cases with fast paths +inline file_input_adapter input_adapter(std::FILE* file) +{ + return file_input_adapter(file); +} + +inline input_stream_adapter input_adapter(std::istream& stream) +{ + return input_stream_adapter(stream); +} + +inline input_stream_adapter input_adapter(std::istream&& stream) +{ + return input_stream_adapter(stream); +} +#endif // JSON_NO_IO + +using contiguous_bytes_input_adapter = decltype(input_adapter(std::declval(), std::declval())); + +// Null-delimited strings, and the like. +template < typename CharT, + typename std::enable_if < + std::is_pointer::value&& + !std::is_array::value&& + std::is_integral::type>::value&& + sizeof(typename std::remove_pointer::type) == 1, + int >::type = 0 > +contiguous_bytes_input_adapter input_adapter(CharT b) +{ + auto length = std::strlen(reinterpret_cast(b)); + const auto* ptr = reinterpret_cast(b); + return input_adapter(ptr, ptr + length); +} + +template +auto input_adapter(T (&array)[N]) -> decltype(input_adapter(array, array + N)) // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) +{ + return input_adapter(array, array + N); +} + +// This class only handles inputs of input_buffer_adapter type. +// It's required so that expressions like {ptr, len} can be implicitely casted +// to the correct adapter. +class span_input_adapter +{ + public: + template < typename CharT, + typename std::enable_if < + std::is_pointer::value&& + std::is_integral::type>::value&& + sizeof(typename std::remove_pointer::type) == 1, + int >::type = 0 > + span_input_adapter(CharT b, std::size_t l) + : ia(reinterpret_cast(b), reinterpret_cast(b) + l) {} + + template::iterator_category, std::random_access_iterator_tag>::value, + int>::type = 0> + span_input_adapter(IteratorType first, IteratorType last) + : ia(input_adapter(first, last)) {} + + contiguous_bytes_input_adapter&& get() + { + return std::move(ia); // NOLINT(hicpp-move-const-arg,performance-move-const-arg) + } + + private: + contiguous_bytes_input_adapter ia; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +#include +#include // string +#include // move +#include // vector + +// #include + +// #include + + +namespace nlohmann +{ + +/*! +@brief SAX interface + +This class describes the SAX interface used by @ref nlohmann::json::sax_parse. +Each function is called in different situations while the input is parsed. The +boolean return value informs the parser whether to continue processing the +input. +*/ +template +struct json_sax +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + + /*! + @brief a null value was read + @return whether parsing should proceed + */ + virtual bool null() = 0; + + /*! + @brief a boolean value was read + @param[in] val boolean value + @return whether parsing should proceed + */ + virtual bool boolean(bool val) = 0; + + /*! + @brief an integer number was read + @param[in] val integer value + @return whether parsing should proceed + */ + virtual bool number_integer(number_integer_t val) = 0; + + /*! + @brief an unsigned integer number was read + @param[in] val unsigned integer value + @return whether parsing should proceed + */ + virtual bool number_unsigned(number_unsigned_t val) = 0; + + /*! + @brief an floating-point number was read + @param[in] val floating-point value + @param[in] s raw token value + @return whether parsing should proceed + */ + virtual bool number_float(number_float_t val, const string_t& s) = 0; + + /*! + @brief a string was read + @param[in] val string value + @return whether parsing should proceed + @note It is safe to move the passed string. + */ + virtual bool string(string_t& val) = 0; + + /*! + @brief a binary string was read + @param[in] val binary value + @return whether parsing should proceed + @note It is safe to move the passed binary. + */ + virtual bool binary(binary_t& val) = 0; + + /*! + @brief the beginning of an object was read + @param[in] elements number of object elements or -1 if unknown + @return whether parsing should proceed + @note binary formats may report the number of elements + */ + virtual bool start_object(std::size_t elements) = 0; + + /*! + @brief an object key was read + @param[in] val object key + @return whether parsing should proceed + @note It is safe to move the passed string. + */ + virtual bool key(string_t& val) = 0; + + /*! + @brief the end of an object was read + @return whether parsing should proceed + */ + virtual bool end_object() = 0; + + /*! + @brief the beginning of an array was read + @param[in] elements number of array elements or -1 if unknown + @return whether parsing should proceed + @note binary formats may report the number of elements + */ + virtual bool start_array(std::size_t elements) = 0; + + /*! + @brief the end of an array was read + @return whether parsing should proceed + */ + virtual bool end_array() = 0; + + /*! + @brief a parse error occurred + @param[in] position the position in the input where the error occurs + @param[in] last_token the last read token + @param[in] ex an exception object describing the error + @return whether parsing should proceed (must return false) + */ + virtual bool parse_error(std::size_t position, + const std::string& last_token, + const detail::exception& ex) = 0; + + json_sax() = default; + json_sax(const json_sax&) = default; + json_sax(json_sax&&) noexcept = default; + json_sax& operator=(const json_sax&) = default; + json_sax& operator=(json_sax&&) noexcept = default; + virtual ~json_sax() = default; +}; + + +namespace detail +{ +/*! +@brief SAX implementation to create a JSON value from SAX events + +This class implements the @ref json_sax interface and processes the SAX events +to create a JSON value which makes it basically a DOM parser. The structure or +hierarchy of the JSON value is managed by the stack `ref_stack` which contains +a pointer to the respective array or object for each recursion depth. + +After successful parsing, the value that is passed by reference to the +constructor contains the parsed value. + +@tparam BasicJsonType the JSON type +*/ +template +class json_sax_dom_parser +{ + public: + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + + /*! + @param[in,out] r reference to a JSON value that is manipulated while + parsing + @param[in] allow_exceptions_ whether parse errors yield exceptions + */ + explicit json_sax_dom_parser(BasicJsonType& r, const bool allow_exceptions_ = true) + : root(r), allow_exceptions(allow_exceptions_) + {} + + // make class move-only + json_sax_dom_parser(const json_sax_dom_parser&) = delete; + json_sax_dom_parser(json_sax_dom_parser&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + json_sax_dom_parser& operator=(const json_sax_dom_parser&) = delete; + json_sax_dom_parser& operator=(json_sax_dom_parser&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + ~json_sax_dom_parser() = default; + + bool null() + { + handle_value(nullptr); + return true; + } + + bool boolean(bool val) + { + handle_value(val); + return true; + } + + bool number_integer(number_integer_t val) + { + handle_value(val); + return true; + } + + bool number_unsigned(number_unsigned_t val) + { + handle_value(val); + return true; + } + + bool number_float(number_float_t val, const string_t& /*unused*/) + { + handle_value(val); + return true; + } + + bool string(string_t& val) + { + handle_value(val); + return true; + } + + bool binary(binary_t& val) + { + handle_value(std::move(val)); + return true; + } + + bool start_object(std::size_t len) + { + ref_stack.push_back(handle_value(BasicJsonType::value_t::object)); + + if (JSON_HEDLEY_UNLIKELY(len != std::size_t(-1) && len > ref_stack.back()->max_size())) + { + JSON_THROW(out_of_range::create(408, "excessive object size: " + std::to_string(len), *ref_stack.back())); + } + + return true; + } + + bool key(string_t& val) + { + // add null at given key and store the reference for later + object_element = &(ref_stack.back()->m_value.object->operator[](val)); + return true; + } + + bool end_object() + { + ref_stack.back()->set_parents(); + ref_stack.pop_back(); + return true; + } + + bool start_array(std::size_t len) + { + ref_stack.push_back(handle_value(BasicJsonType::value_t::array)); + + if (JSON_HEDLEY_UNLIKELY(len != std::size_t(-1) && len > ref_stack.back()->max_size())) + { + JSON_THROW(out_of_range::create(408, "excessive array size: " + std::to_string(len), *ref_stack.back())); + } + + return true; + } + + bool end_array() + { + ref_stack.back()->set_parents(); + ref_stack.pop_back(); + return true; + } + + template + bool parse_error(std::size_t /*unused*/, const std::string& /*unused*/, + const Exception& ex) + { + errored = true; + static_cast(ex); + if (allow_exceptions) + { + JSON_THROW(ex); + } + return false; + } + + constexpr bool is_errored() const + { + return errored; + } + + private: + /*! + @invariant If the ref stack is empty, then the passed value will be the new + root. + @invariant If the ref stack contains a value, then it is an array or an + object to which we can add elements + */ + template + JSON_HEDLEY_RETURNS_NON_NULL + BasicJsonType* handle_value(Value&& v) + { + if (ref_stack.empty()) + { + root = BasicJsonType(std::forward(v)); + return &root; + } + + JSON_ASSERT(ref_stack.back()->is_array() || ref_stack.back()->is_object()); + + if (ref_stack.back()->is_array()) + { + ref_stack.back()->m_value.array->emplace_back(std::forward(v)); + return &(ref_stack.back()->m_value.array->back()); + } + + JSON_ASSERT(ref_stack.back()->is_object()); + JSON_ASSERT(object_element); + *object_element = BasicJsonType(std::forward(v)); + return object_element; + } + + /// the parsed JSON value + BasicJsonType& root; + /// stack to model hierarchy of values + std::vector ref_stack {}; + /// helper to hold the reference for the next object element + BasicJsonType* object_element = nullptr; + /// whether a syntax error occurred + bool errored = false; + /// whether to throw exceptions in case of errors + const bool allow_exceptions = true; +}; + +template +class json_sax_dom_callback_parser +{ + public: + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + using parser_callback_t = typename BasicJsonType::parser_callback_t; + using parse_event_t = typename BasicJsonType::parse_event_t; + + json_sax_dom_callback_parser(BasicJsonType& r, + const parser_callback_t cb, + const bool allow_exceptions_ = true) + : root(r), callback(cb), allow_exceptions(allow_exceptions_) + { + keep_stack.push_back(true); + } + + // make class move-only + json_sax_dom_callback_parser(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser(json_sax_dom_callback_parser&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + json_sax_dom_callback_parser& operator=(const json_sax_dom_callback_parser&) = delete; + json_sax_dom_callback_parser& operator=(json_sax_dom_callback_parser&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + ~json_sax_dom_callback_parser() = default; + + bool null() + { + handle_value(nullptr); + return true; + } + + bool boolean(bool val) + { + handle_value(val); + return true; + } + + bool number_integer(number_integer_t val) + { + handle_value(val); + return true; + } + + bool number_unsigned(number_unsigned_t val) + { + handle_value(val); + return true; + } + + bool number_float(number_float_t val, const string_t& /*unused*/) + { + handle_value(val); + return true; + } + + bool string(string_t& val) + { + handle_value(val); + return true; + } + + bool binary(binary_t& val) + { + handle_value(std::move(val)); + return true; + } + + bool start_object(std::size_t len) + { + // check callback for object start + const bool keep = callback(static_cast(ref_stack.size()), parse_event_t::object_start, discarded); + keep_stack.push_back(keep); + + auto val = handle_value(BasicJsonType::value_t::object, true); + ref_stack.push_back(val.second); + + // check object limit + if (ref_stack.back() && JSON_HEDLEY_UNLIKELY(len != std::size_t(-1) && len > ref_stack.back()->max_size())) + { + JSON_THROW(out_of_range::create(408, "excessive object size: " + std::to_string(len), *ref_stack.back())); + } + + return true; + } + + bool key(string_t& val) + { + BasicJsonType k = BasicJsonType(val); + + // check callback for key + const bool keep = callback(static_cast(ref_stack.size()), parse_event_t::key, k); + key_keep_stack.push_back(keep); + + // add discarded value at given key and store the reference for later + if (keep && ref_stack.back()) + { + object_element = &(ref_stack.back()->m_value.object->operator[](val) = discarded); + } + + return true; + } + + bool end_object() + { + if (ref_stack.back()) + { + if (!callback(static_cast(ref_stack.size()) - 1, parse_event_t::object_end, *ref_stack.back())) + { + // discard object + *ref_stack.back() = discarded; + } + else + { + ref_stack.back()->set_parents(); + } + } + + JSON_ASSERT(!ref_stack.empty()); + JSON_ASSERT(!keep_stack.empty()); + ref_stack.pop_back(); + keep_stack.pop_back(); + + if (!ref_stack.empty() && ref_stack.back() && ref_stack.back()->is_structured()) + { + // remove discarded value + for (auto it = ref_stack.back()->begin(); it != ref_stack.back()->end(); ++it) + { + if (it->is_discarded()) + { + ref_stack.back()->erase(it); + break; + } + } + } + + return true; + } + + bool start_array(std::size_t len) + { + const bool keep = callback(static_cast(ref_stack.size()), parse_event_t::array_start, discarded); + keep_stack.push_back(keep); + + auto val = handle_value(BasicJsonType::value_t::array, true); + ref_stack.push_back(val.second); + + // check array limit + if (ref_stack.back() && JSON_HEDLEY_UNLIKELY(len != std::size_t(-1) && len > ref_stack.back()->max_size())) + { + JSON_THROW(out_of_range::create(408, "excessive array size: " + std::to_string(len), *ref_stack.back())); + } + + return true; + } + + bool end_array() + { + bool keep = true; + + if (ref_stack.back()) + { + keep = callback(static_cast(ref_stack.size()) - 1, parse_event_t::array_end, *ref_stack.back()); + if (keep) + { + ref_stack.back()->set_parents(); + } + else + { + // discard array + *ref_stack.back() = discarded; + } + } + + JSON_ASSERT(!ref_stack.empty()); + JSON_ASSERT(!keep_stack.empty()); + ref_stack.pop_back(); + keep_stack.pop_back(); + + // remove discarded value + if (!keep && !ref_stack.empty() && ref_stack.back()->is_array()) + { + ref_stack.back()->m_value.array->pop_back(); + } + + return true; + } + + template + bool parse_error(std::size_t /*unused*/, const std::string& /*unused*/, + const Exception& ex) + { + errored = true; + static_cast(ex); + if (allow_exceptions) + { + JSON_THROW(ex); + } + return false; + } + + constexpr bool is_errored() const + { + return errored; + } + + private: + /*! + @param[in] v value to add to the JSON value we build during parsing + @param[in] skip_callback whether we should skip calling the callback + function; this is required after start_array() and + start_object() SAX events, because otherwise we would call the + callback function with an empty array or object, respectively. + + @invariant If the ref stack is empty, then the passed value will be the new + root. + @invariant If the ref stack contains a value, then it is an array or an + object to which we can add elements + + @return pair of boolean (whether value should be kept) and pointer (to the + passed value in the ref_stack hierarchy; nullptr if not kept) + */ + template + std::pair handle_value(Value&& v, const bool skip_callback = false) + { + JSON_ASSERT(!keep_stack.empty()); + + // do not handle this value if we know it would be added to a discarded + // container + if (!keep_stack.back()) + { + return {false, nullptr}; + } + + // create value + auto value = BasicJsonType(std::forward(v)); + + // check callback + const bool keep = skip_callback || callback(static_cast(ref_stack.size()), parse_event_t::value, value); + + // do not handle this value if we just learnt it shall be discarded + if (!keep) + { + return {false, nullptr}; + } + + if (ref_stack.empty()) + { + root = std::move(value); + return {true, &root}; + } + + // skip this value if we already decided to skip the parent + // (https://github.com/nlohmann/json/issues/971#issuecomment-413678360) + if (!ref_stack.back()) + { + return {false, nullptr}; + } + + // we now only expect arrays and objects + JSON_ASSERT(ref_stack.back()->is_array() || ref_stack.back()->is_object()); + + // array + if (ref_stack.back()->is_array()) + { + ref_stack.back()->m_value.array->emplace_back(std::move(value)); + return {true, &(ref_stack.back()->m_value.array->back())}; + } + + // object + JSON_ASSERT(ref_stack.back()->is_object()); + // check if we should store an element for the current key + JSON_ASSERT(!key_keep_stack.empty()); + const bool store_element = key_keep_stack.back(); + key_keep_stack.pop_back(); + + if (!store_element) + { + return {false, nullptr}; + } + + JSON_ASSERT(object_element); + *object_element = std::move(value); + return {true, object_element}; + } + + /// the parsed JSON value + BasicJsonType& root; + /// stack to model hierarchy of values + std::vector ref_stack {}; + /// stack to manage which values to keep + std::vector keep_stack {}; + /// stack to manage which object keys to keep + std::vector key_keep_stack {}; + /// helper to hold the reference for the next object element + BasicJsonType* object_element = nullptr; + /// whether a syntax error occurred + bool errored = false; + /// callback function + const parser_callback_t callback = nullptr; + /// whether to throw exceptions in case of errors + const bool allow_exceptions = true; + /// a discarded value for the callback + BasicJsonType discarded = BasicJsonType::value_t::discarded; +}; + +template +class json_sax_acceptor +{ + public: + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + + bool null() + { + return true; + } + + bool boolean(bool /*unused*/) + { + return true; + } + + bool number_integer(number_integer_t /*unused*/) + { + return true; + } + + bool number_unsigned(number_unsigned_t /*unused*/) + { + return true; + } + + bool number_float(number_float_t /*unused*/, const string_t& /*unused*/) + { + return true; + } + + bool string(string_t& /*unused*/) + { + return true; + } + + bool binary(binary_t& /*unused*/) + { + return true; + } + + bool start_object(std::size_t /*unused*/ = std::size_t(-1)) + { + return true; + } + + bool key(string_t& /*unused*/) + { + return true; + } + + bool end_object() + { + return true; + } + + bool start_array(std::size_t /*unused*/ = std::size_t(-1)) + { + return true; + } + + bool end_array() + { + return true; + } + + bool parse_error(std::size_t /*unused*/, const std::string& /*unused*/, const detail::exception& /*unused*/) + { + return false; + } +}; +} // namespace detail + +} // namespace nlohmann + +// #include + + +#include // array +#include // localeconv +#include // size_t +#include // snprintf +#include // strtof, strtod, strtold, strtoll, strtoull +#include // initializer_list +#include // char_traits, string +#include // move +#include // vector + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/////////// +// lexer // +/////////// + +template +class lexer_base +{ + public: + /// token types for the parser + enum class token_type + { + uninitialized, ///< indicating the scanner is uninitialized + literal_true, ///< the `true` literal + literal_false, ///< the `false` literal + literal_null, ///< the `null` literal + value_string, ///< a string -- use get_string() for actual value + value_unsigned, ///< an unsigned integer -- use get_number_unsigned() for actual value + value_integer, ///< a signed integer -- use get_number_integer() for actual value + value_float, ///< an floating point number -- use get_number_float() for actual value + begin_array, ///< the character for array begin `[` + begin_object, ///< the character for object begin `{` + end_array, ///< the character for array end `]` + end_object, ///< the character for object end `}` + name_separator, ///< the name separator `:` + value_separator, ///< the value separator `,` + parse_error, ///< indicating a parse error + end_of_input, ///< indicating the end of the input buffer + literal_or_value ///< a literal or the begin of a value (only for diagnostics) + }; + + /// return name of values of type token_type (only used for errors) + JSON_HEDLEY_RETURNS_NON_NULL + JSON_HEDLEY_CONST + static const char* token_type_name(const token_type t) noexcept + { + switch (t) + { + case token_type::uninitialized: + return ""; + case token_type::literal_true: + return "true literal"; + case token_type::literal_false: + return "false literal"; + case token_type::literal_null: + return "null literal"; + case token_type::value_string: + return "string literal"; + case token_type::value_unsigned: + case token_type::value_integer: + case token_type::value_float: + return "number literal"; + case token_type::begin_array: + return "'['"; + case token_type::begin_object: + return "'{'"; + case token_type::end_array: + return "']'"; + case token_type::end_object: + return "'}'"; + case token_type::name_separator: + return "':'"; + case token_type::value_separator: + return "','"; + case token_type::parse_error: + return ""; + case token_type::end_of_input: + return "end of input"; + case token_type::literal_or_value: + return "'[', '{', or a literal"; + // LCOV_EXCL_START + default: // catch non-enum values + return "unknown token"; + // LCOV_EXCL_STOP + } + } +}; +/*! +@brief lexical analysis + +This class organizes the lexical analysis during JSON deserialization. +*/ +template +class lexer : public lexer_base +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using char_type = typename InputAdapterType::char_type; + using char_int_type = typename std::char_traits::int_type; + + public: + using token_type = typename lexer_base::token_type; + + explicit lexer(InputAdapterType&& adapter, bool ignore_comments_ = false) noexcept + : ia(std::move(adapter)) + , ignore_comments(ignore_comments_) + , decimal_point_char(static_cast(get_decimal_point())) + {} + + // delete because of pointer members + lexer(const lexer&) = delete; + lexer(lexer&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + lexer& operator=(lexer&) = delete; + lexer& operator=(lexer&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + ~lexer() = default; + + private: + ///////////////////// + // locales + ///////////////////// + + /// return the locale-dependent decimal point + JSON_HEDLEY_PURE + static char get_decimal_point() noexcept + { + const auto* loc = localeconv(); + JSON_ASSERT(loc != nullptr); + return (loc->decimal_point == nullptr) ? '.' : *(loc->decimal_point); + } + + ///////////////////// + // scan functions + ///////////////////// + + /*! + @brief get codepoint from 4 hex characters following `\u` + + For input "\u c1 c2 c3 c4" the codepoint is: + (c1 * 0x1000) + (c2 * 0x0100) + (c3 * 0x0010) + c4 + = (c1 << 12) + (c2 << 8) + (c3 << 4) + (c4 << 0) + + Furthermore, the possible characters '0'..'9', 'A'..'F', and 'a'..'f' + must be converted to the integers 0x0..0x9, 0xA..0xF, 0xA..0xF, resp. The + conversion is done by subtracting the offset (0x30, 0x37, and 0x57) + between the ASCII value of the character and the desired integer value. + + @return codepoint (0x0000..0xFFFF) or -1 in case of an error (e.g. EOF or + non-hex character) + */ + int get_codepoint() + { + // this function only makes sense after reading `\u` + JSON_ASSERT(current == 'u'); + int codepoint = 0; + + const auto factors = { 12u, 8u, 4u, 0u }; + for (const auto factor : factors) + { + get(); + + if (current >= '0' && current <= '9') + { + codepoint += static_cast((static_cast(current) - 0x30u) << factor); + } + else if (current >= 'A' && current <= 'F') + { + codepoint += static_cast((static_cast(current) - 0x37u) << factor); + } + else if (current >= 'a' && current <= 'f') + { + codepoint += static_cast((static_cast(current) - 0x57u) << factor); + } + else + { + return -1; + } + } + + JSON_ASSERT(0x0000 <= codepoint && codepoint <= 0xFFFF); + return codepoint; + } + + /*! + @brief check if the next byte(s) are inside a given range + + Adds the current byte and, for each passed range, reads a new byte and + checks if it is inside the range. If a violation was detected, set up an + error message and return false. Otherwise, return true. + + @param[in] ranges list of integers; interpreted as list of pairs of + inclusive lower and upper bound, respectively + + @pre The passed list @a ranges must have 2, 4, or 6 elements; that is, + 1, 2, or 3 pairs. This precondition is enforced by an assertion. + + @return true if and only if no range violation was detected + */ + bool next_byte_in_range(std::initializer_list ranges) + { + JSON_ASSERT(ranges.size() == 2 || ranges.size() == 4 || ranges.size() == 6); + add(current); + + for (auto range = ranges.begin(); range != ranges.end(); ++range) + { + get(); + if (JSON_HEDLEY_LIKELY(*range <= current && current <= *(++range))) + { + add(current); + } + else + { + error_message = "invalid string: ill-formed UTF-8 byte"; + return false; + } + } + + return true; + } + + /*! + @brief scan a string literal + + This function scans a string according to Sect. 7 of RFC 8259. While + scanning, bytes are escaped and copied into buffer token_buffer. Then the + function returns successfully, token_buffer is *not* null-terminated (as it + may contain \0 bytes), and token_buffer.size() is the number of bytes in the + string. + + @return token_type::value_string if string could be successfully scanned, + token_type::parse_error otherwise + + @note In case of errors, variable error_message contains a textual + description. + */ + token_type scan_string() + { + // reset token_buffer (ignore opening quote) + reset(); + + // we entered the function by reading an open quote + JSON_ASSERT(current == '\"'); + + while (true) + { + // get next character + switch (get()) + { + // end of file while parsing string + case std::char_traits::eof(): + { + error_message = "invalid string: missing closing quote"; + return token_type::parse_error; + } + + // closing quote + case '\"': + { + return token_type::value_string; + } + + // escapes + case '\\': + { + switch (get()) + { + // quotation mark + case '\"': + add('\"'); + break; + // reverse solidus + case '\\': + add('\\'); + break; + // solidus + case '/': + add('/'); + break; + // backspace + case 'b': + add('\b'); + break; + // form feed + case 'f': + add('\f'); + break; + // line feed + case 'n': + add('\n'); + break; + // carriage return + case 'r': + add('\r'); + break; + // tab + case 't': + add('\t'); + break; + + // unicode escapes + case 'u': + { + const int codepoint1 = get_codepoint(); + int codepoint = codepoint1; // start with codepoint1 + + if (JSON_HEDLEY_UNLIKELY(codepoint1 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if code point is a high surrogate + if (0xD800 <= codepoint1 && codepoint1 <= 0xDBFF) + { + // expect next \uxxxx entry + if (JSON_HEDLEY_LIKELY(get() == '\\' && get() == 'u')) + { + const int codepoint2 = get_codepoint(); + + if (JSON_HEDLEY_UNLIKELY(codepoint2 == -1)) + { + error_message = "invalid string: '\\u' must be followed by 4 hex digits"; + return token_type::parse_error; + } + + // check if codepoint2 is a low surrogate + if (JSON_HEDLEY_LIKELY(0xDC00 <= codepoint2 && codepoint2 <= 0xDFFF)) + { + // overwrite codepoint + codepoint = static_cast( + // high surrogate occupies the most significant 22 bits + (static_cast(codepoint1) << 10u) + // low surrogate occupies the least significant 15 bits + + static_cast(codepoint2) + // there is still the 0xD800, 0xDC00 and 0x10000 noise + // in the result so we have to subtract with: + // (0xD800 << 10) + DC00 - 0x10000 = 0x35FDC00 + - 0x35FDC00u); + } + else + { + error_message = "invalid string: surrogate U+D800..U+DBFF must be followed by U+DC00..U+DFFF"; + return token_type::parse_error; + } + } + else + { + error_message = "invalid string: surrogate U+D800..U+DBFF must be followed by U+DC00..U+DFFF"; + return token_type::parse_error; + } + } + else + { + if (JSON_HEDLEY_UNLIKELY(0xDC00 <= codepoint1 && codepoint1 <= 0xDFFF)) + { + error_message = "invalid string: surrogate U+DC00..U+DFFF must follow U+D800..U+DBFF"; + return token_type::parse_error; + } + } + + // result of the above calculation yields a proper codepoint + JSON_ASSERT(0x00 <= codepoint && codepoint <= 0x10FFFF); + + // translate codepoint into bytes + if (codepoint < 0x80) + { + // 1-byte characters: 0xxxxxxx (ASCII) + add(static_cast(codepoint)); + } + else if (codepoint <= 0x7FF) + { + // 2-byte characters: 110xxxxx 10xxxxxx + add(static_cast(0xC0u | (static_cast(codepoint) >> 6u))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + else if (codepoint <= 0xFFFF) + { + // 3-byte characters: 1110xxxx 10xxxxxx 10xxxxxx + add(static_cast(0xE0u | (static_cast(codepoint) >> 12u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + else + { + // 4-byte characters: 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + add(static_cast(0xF0u | (static_cast(codepoint) >> 18u))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 12u) & 0x3Fu))); + add(static_cast(0x80u | ((static_cast(codepoint) >> 6u) & 0x3Fu))); + add(static_cast(0x80u | (static_cast(codepoint) & 0x3Fu))); + } + + break; + } + + // other characters after escape + default: + error_message = "invalid string: forbidden character after backslash"; + return token_type::parse_error; + } + + break; + } + + // invalid control characters + case 0x00: + { + error_message = "invalid string: control character U+0000 (NUL) must be escaped to \\u0000"; + return token_type::parse_error; + } + + case 0x01: + { + error_message = "invalid string: control character U+0001 (SOH) must be escaped to \\u0001"; + return token_type::parse_error; + } + + case 0x02: + { + error_message = "invalid string: control character U+0002 (STX) must be escaped to \\u0002"; + return token_type::parse_error; + } + + case 0x03: + { + error_message = "invalid string: control character U+0003 (ETX) must be escaped to \\u0003"; + return token_type::parse_error; + } + + case 0x04: + { + error_message = "invalid string: control character U+0004 (EOT) must be escaped to \\u0004"; + return token_type::parse_error; + } + + case 0x05: + { + error_message = "invalid string: control character U+0005 (ENQ) must be escaped to \\u0005"; + return token_type::parse_error; + } + + case 0x06: + { + error_message = "invalid string: control character U+0006 (ACK) must be escaped to \\u0006"; + return token_type::parse_error; + } + + case 0x07: + { + error_message = "invalid string: control character U+0007 (BEL) must be escaped to \\u0007"; + return token_type::parse_error; + } + + case 0x08: + { + error_message = "invalid string: control character U+0008 (BS) must be escaped to \\u0008 or \\b"; + return token_type::parse_error; + } + + case 0x09: + { + error_message = "invalid string: control character U+0009 (HT) must be escaped to \\u0009 or \\t"; + return token_type::parse_error; + } + + case 0x0A: + { + error_message = "invalid string: control character U+000A (LF) must be escaped to \\u000A or \\n"; + return token_type::parse_error; + } + + case 0x0B: + { + error_message = "invalid string: control character U+000B (VT) must be escaped to \\u000B"; + return token_type::parse_error; + } + + case 0x0C: + { + error_message = "invalid string: control character U+000C (FF) must be escaped to \\u000C or \\f"; + return token_type::parse_error; + } + + case 0x0D: + { + error_message = "invalid string: control character U+000D (CR) must be escaped to \\u000D or \\r"; + return token_type::parse_error; + } + + case 0x0E: + { + error_message = "invalid string: control character U+000E (SO) must be escaped to \\u000E"; + return token_type::parse_error; + } + + case 0x0F: + { + error_message = "invalid string: control character U+000F (SI) must be escaped to \\u000F"; + return token_type::parse_error; + } + + case 0x10: + { + error_message = "invalid string: control character U+0010 (DLE) must be escaped to \\u0010"; + return token_type::parse_error; + } + + case 0x11: + { + error_message = "invalid string: control character U+0011 (DC1) must be escaped to \\u0011"; + return token_type::parse_error; + } + + case 0x12: + { + error_message = "invalid string: control character U+0012 (DC2) must be escaped to \\u0012"; + return token_type::parse_error; + } + + case 0x13: + { + error_message = "invalid string: control character U+0013 (DC3) must be escaped to \\u0013"; + return token_type::parse_error; + } + + case 0x14: + { + error_message = "invalid string: control character U+0014 (DC4) must be escaped to \\u0014"; + return token_type::parse_error; + } + + case 0x15: + { + error_message = "invalid string: control character U+0015 (NAK) must be escaped to \\u0015"; + return token_type::parse_error; + } + + case 0x16: + { + error_message = "invalid string: control character U+0016 (SYN) must be escaped to \\u0016"; + return token_type::parse_error; + } + + case 0x17: + { + error_message = "invalid string: control character U+0017 (ETB) must be escaped to \\u0017"; + return token_type::parse_error; + } + + case 0x18: + { + error_message = "invalid string: control character U+0018 (CAN) must be escaped to \\u0018"; + return token_type::parse_error; + } + + case 0x19: + { + error_message = "invalid string: control character U+0019 (EM) must be escaped to \\u0019"; + return token_type::parse_error; + } + + case 0x1A: + { + error_message = "invalid string: control character U+001A (SUB) must be escaped to \\u001A"; + return token_type::parse_error; + } + + case 0x1B: + { + error_message = "invalid string: control character U+001B (ESC) must be escaped to \\u001B"; + return token_type::parse_error; + } + + case 0x1C: + { + error_message = "invalid string: control character U+001C (FS) must be escaped to \\u001C"; + return token_type::parse_error; + } + + case 0x1D: + { + error_message = "invalid string: control character U+001D (GS) must be escaped to \\u001D"; + return token_type::parse_error; + } + + case 0x1E: + { + error_message = "invalid string: control character U+001E (RS) must be escaped to \\u001E"; + return token_type::parse_error; + } + + case 0x1F: + { + error_message = "invalid string: control character U+001F (US) must be escaped to \\u001F"; + return token_type::parse_error; + } + + // U+0020..U+007F (except U+0022 (quote) and U+005C (backspace)) + case 0x20: + case 0x21: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2A: + case 0x2B: + case 0x2C: + case 0x2D: + case 0x2E: + case 0x2F: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3A: + case 0x3B: + case 0x3C: + case 0x3D: + case 0x3E: + case 0x3F: + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4A: + case 0x4B: + case 0x4C: + case 0x4D: + case 0x4E: + case 0x4F: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: + case 0x59: + case 0x5A: + case 0x5B: + case 0x5D: + case 0x5E: + case 0x5F: + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6A: + case 0x6B: + case 0x6C: + case 0x6D: + case 0x6E: + case 0x6F: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: + case 0x79: + case 0x7A: + case 0x7B: + case 0x7C: + case 0x7D: + case 0x7E: + case 0x7F: + { + add(current); + break; + } + + // U+0080..U+07FF: bytes C2..DF 80..BF + case 0xC2: + case 0xC3: + case 0xC4: + case 0xC5: + case 0xC6: + case 0xC7: + case 0xC8: + case 0xC9: + case 0xCA: + case 0xCB: + case 0xCC: + case 0xCD: + case 0xCE: + case 0xCF: + case 0xD0: + case 0xD1: + case 0xD2: + case 0xD3: + case 0xD4: + case 0xD5: + case 0xD6: + case 0xD7: + case 0xD8: + case 0xD9: + case 0xDA: + case 0xDB: + case 0xDC: + case 0xDD: + case 0xDE: + case 0xDF: + { + if (JSON_HEDLEY_UNLIKELY(!next_byte_in_range({0x80, 0xBF}))) + { + return token_type::parse_error; + } + break; + } + + // U+0800..U+0FFF: bytes E0 A0..BF 80..BF + case 0xE0: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0xA0, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+1000..U+CFFF: bytes E1..EC 80..BF 80..BF + // U+E000..U+FFFF: bytes EE..EF 80..BF 80..BF + case 0xE1: + case 0xE2: + case 0xE3: + case 0xE4: + case 0xE5: + case 0xE6: + case 0xE7: + case 0xE8: + case 0xE9: + case 0xEA: + case 0xEB: + case 0xEC: + case 0xEE: + case 0xEF: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+D000..U+D7FF: bytes ED 80..9F 80..BF + case 0xED: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0x80, 0x9F, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+10000..U+3FFFF F0 90..BF 80..BF 80..BF + case 0xF0: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0x90, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+40000..U+FFFFF F1..F3 80..BF 80..BF 80..BF + case 0xF1: + case 0xF2: + case 0xF3: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0x80, 0xBF, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // U+100000..U+10FFFF F4 80..8F 80..BF 80..BF + case 0xF4: + { + if (JSON_HEDLEY_UNLIKELY(!(next_byte_in_range({0x80, 0x8F, 0x80, 0xBF, 0x80, 0xBF})))) + { + return token_type::parse_error; + } + break; + } + + // remaining bytes (80..C1 and F5..FF) are ill-formed + default: + { + error_message = "invalid string: ill-formed UTF-8 byte"; + return token_type::parse_error; + } + } + } + } + + /*! + * @brief scan a comment + * @return whether comment could be scanned successfully + */ + bool scan_comment() + { + switch (get()) + { + // single-line comments skip input until a newline or EOF is read + case '/': + { + while (true) + { + switch (get()) + { + case '\n': + case '\r': + case std::char_traits::eof(): + case '\0': + return true; + + default: + break; + } + } + } + + // multi-line comments skip input until */ is read + case '*': + { + while (true) + { + switch (get()) + { + case std::char_traits::eof(): + case '\0': + { + error_message = "invalid comment; missing closing '*/'"; + return false; + } + + case '*': + { + switch (get()) + { + case '/': + return true; + + default: + { + unget(); + continue; + } + } + } + + default: + continue; + } + } + } + + // unexpected character after reading '/' + default: + { + error_message = "invalid comment; expecting '/' or '*' after '/'"; + return false; + } + } + } + + JSON_HEDLEY_NON_NULL(2) + static void strtof(float& f, const char* str, char** endptr) noexcept + { + f = std::strtof(str, endptr); + } + + JSON_HEDLEY_NON_NULL(2) + static void strtof(double& f, const char* str, char** endptr) noexcept + { + f = std::strtod(str, endptr); + } + + JSON_HEDLEY_NON_NULL(2) + static void strtof(long double& f, const char* str, char** endptr) noexcept + { + f = std::strtold(str, endptr); + } + + /*! + @brief scan a number literal + + This function scans a string according to Sect. 6 of RFC 8259. + + The function is realized with a deterministic finite state machine derived + from the grammar described in RFC 8259. Starting in state "init", the + input is read and used to determined the next state. Only state "done" + accepts the number. State "error" is a trap state to model errors. In the + table below, "anything" means any character but the ones listed before. + + state | 0 | 1-9 | e E | + | - | . | anything + ---------|----------|----------|----------|---------|---------|----------|----------- + init | zero | any1 | [error] | [error] | minus | [error] | [error] + minus | zero | any1 | [error] | [error] | [error] | [error] | [error] + zero | done | done | exponent | done | done | decimal1 | done + any1 | any1 | any1 | exponent | done | done | decimal1 | done + decimal1 | decimal2 | decimal2 | [error] | [error] | [error] | [error] | [error] + decimal2 | decimal2 | decimal2 | exponent | done | done | done | done + exponent | any2 | any2 | [error] | sign | sign | [error] | [error] + sign | any2 | any2 | [error] | [error] | [error] | [error] | [error] + any2 | any2 | any2 | done | done | done | done | done + + The state machine is realized with one label per state (prefixed with + "scan_number_") and `goto` statements between them. The state machine + contains cycles, but any cycle can be left when EOF is read. Therefore, + the function is guaranteed to terminate. + + During scanning, the read bytes are stored in token_buffer. This string is + then converted to a signed integer, an unsigned integer, or a + floating-point number. + + @return token_type::value_unsigned, token_type::value_integer, or + token_type::value_float if number could be successfully scanned, + token_type::parse_error otherwise + + @note The scanner is independent of the current locale. Internally, the + locale's decimal point is used instead of `.` to work with the + locale-dependent converters. + */ + token_type scan_number() // lgtm [cpp/use-of-goto] + { + // reset token_buffer to store the number's bytes + reset(); + + // the type of the parsed number; initially set to unsigned; will be + // changed if minus sign, decimal point or exponent is read + token_type number_type = token_type::value_unsigned; + + // state (init): we just found out we need to scan a number + switch (current) + { + case '-': + { + add(current); + goto scan_number_minus; + } + + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + // all other characters are rejected outside scan_number() + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + +scan_number_minus: + // state: we just parsed a leading minus sign + number_type = token_type::value_integer; + switch (get()) + { + case '0': + { + add(current); + goto scan_number_zero; + } + + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + default: + { + error_message = "invalid number; expected digit after '-'"; + return token_type::parse_error; + } + } + +scan_number_zero: + // state: we just parse a zero (maybe with a leading minus sign) + switch (get()) + { + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_any1: + // state: we just parsed a number 0-9 (maybe with a leading minus sign) + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any1; + } + + case '.': + { + add(decimal_point_char); + goto scan_number_decimal1; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_decimal1: + // state: we just parsed a decimal point + number_type = token_type::value_float; + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + default: + { + error_message = "invalid number; expected digit after '.'"; + return token_type::parse_error; + } + } + +scan_number_decimal2: + // we just parsed at least one number after a decimal point + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_decimal2; + } + + case 'e': + case 'E': + { + add(current); + goto scan_number_exponent; + } + + default: + goto scan_number_done; + } + +scan_number_exponent: + // we just parsed an exponent + number_type = token_type::value_float; + switch (get()) + { + case '+': + case '-': + { + add(current); + goto scan_number_sign; + } + + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = + "invalid number; expected '+', '-', or digit after exponent"; + return token_type::parse_error; + } + } + +scan_number_sign: + // we just parsed an exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + { + error_message = "invalid number; expected digit after exponent sign"; + return token_type::parse_error; + } + } + +scan_number_any2: + // we just parsed a number after the exponent or exponent sign + switch (get()) + { + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + { + add(current); + goto scan_number_any2; + } + + default: + goto scan_number_done; + } + +scan_number_done: + // unget the character after the number (we only read it to know that + // we are done scanning a number) + unget(); + + char* endptr = nullptr; // NOLINT(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + errno = 0; + + // try to parse integers first and fall back to floats + if (number_type == token_type::value_unsigned) + { + const auto x = std::strtoull(token_buffer.data(), &endptr, 10); + + // we checked the number format before + JSON_ASSERT(endptr == token_buffer.data() + token_buffer.size()); + + if (errno == 0) + { + value_unsigned = static_cast(x); + if (value_unsigned == x) + { + return token_type::value_unsigned; + } + } + } + else if (number_type == token_type::value_integer) + { + const auto x = std::strtoll(token_buffer.data(), &endptr, 10); + + // we checked the number format before + JSON_ASSERT(endptr == token_buffer.data() + token_buffer.size()); + + if (errno == 0) + { + value_integer = static_cast(x); + if (value_integer == x) + { + return token_type::value_integer; + } + } + } + + // this code is reached if we parse a floating-point number or if an + // integer conversion above failed + strtof(value_float, token_buffer.data(), &endptr); + + // we checked the number format before + JSON_ASSERT(endptr == token_buffer.data() + token_buffer.size()); + + return token_type::value_float; + } + + /*! + @param[in] literal_text the literal text to expect + @param[in] length the length of the passed literal text + @param[in] return_type the token type to return on success + */ + JSON_HEDLEY_NON_NULL(2) + token_type scan_literal(const char_type* literal_text, const std::size_t length, + token_type return_type) + { + JSON_ASSERT(std::char_traits::to_char_type(current) == literal_text[0]); + for (std::size_t i = 1; i < length; ++i) + { + if (JSON_HEDLEY_UNLIKELY(std::char_traits::to_char_type(get()) != literal_text[i])) + { + error_message = "invalid literal"; + return token_type::parse_error; + } + } + return return_type; + } + + ///////////////////// + // input management + ///////////////////// + + /// reset token_buffer; current character is beginning of token + void reset() noexcept + { + token_buffer.clear(); + token_string.clear(); + token_string.push_back(std::char_traits::to_char_type(current)); + } + + /* + @brief get next character from the input + + This function provides the interface to the used input adapter. It does + not throw in case the input reached EOF, but returns a + `std::char_traits::eof()` in that case. Stores the scanned characters + for use in error messages. + + @return character read from the input + */ + char_int_type get() + { + ++position.chars_read_total; + ++position.chars_read_current_line; + + if (next_unget) + { + // just reset the next_unget variable and work with current + next_unget = false; + } + else + { + current = ia.get_character(); + } + + if (JSON_HEDLEY_LIKELY(current != std::char_traits::eof())) + { + token_string.push_back(std::char_traits::to_char_type(current)); + } + + if (current == '\n') + { + ++position.lines_read; + position.chars_read_current_line = 0; + } + + return current; + } + + /*! + @brief unget current character (read it again on next get) + + We implement unget by setting variable next_unget to true. The input is not + changed - we just simulate ungetting by modifying chars_read_total, + chars_read_current_line, and token_string. The next call to get() will + behave as if the unget character is read again. + */ + void unget() + { + next_unget = true; + + --position.chars_read_total; + + // in case we "unget" a newline, we have to also decrement the lines_read + if (position.chars_read_current_line == 0) + { + if (position.lines_read > 0) + { + --position.lines_read; + } + } + else + { + --position.chars_read_current_line; + } + + if (JSON_HEDLEY_LIKELY(current != std::char_traits::eof())) + { + JSON_ASSERT(!token_string.empty()); + token_string.pop_back(); + } + } + + /// add a character to token_buffer + void add(char_int_type c) + { + token_buffer.push_back(static_cast(c)); + } + + public: + ///////////////////// + // value getters + ///////////////////// + + /// return integer value + constexpr number_integer_t get_number_integer() const noexcept + { + return value_integer; + } + + /// return unsigned integer value + constexpr number_unsigned_t get_number_unsigned() const noexcept + { + return value_unsigned; + } + + /// return floating-point value + constexpr number_float_t get_number_float() const noexcept + { + return value_float; + } + + /// return current string value (implicitly resets the token; useful only once) + string_t& get_string() + { + return token_buffer; + } + + ///////////////////// + // diagnostics + ///////////////////// + + /// return position of last read token + constexpr position_t get_position() const noexcept + { + return position; + } + + /// return the last read token (for errors only). Will never contain EOF + /// (an arbitrary value that is not a valid char value, often -1), because + /// 255 may legitimately occur. May contain NUL, which should be escaped. + std::string get_token_string() const + { + // escape control characters + std::string result; + for (const auto c : token_string) + { + if (static_cast(c) <= '\x1F') + { + // escape control characters + std::array cs{{}}; + (std::snprintf)(cs.data(), cs.size(), "", static_cast(c)); // NOLINT(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + result += cs.data(); + } + else + { + // add character as is + result.push_back(static_cast(c)); + } + } + + return result; + } + + /// return syntax error message + JSON_HEDLEY_RETURNS_NON_NULL + constexpr const char* get_error_message() const noexcept + { + return error_message; + } + + ///////////////////// + // actual scanner + ///////////////////// + + /*! + @brief skip the UTF-8 byte order mark + @return true iff there is no BOM or the correct BOM has been skipped + */ + bool skip_bom() + { + if (get() == 0xEF) + { + // check if we completely parse the BOM + return get() == 0xBB && get() == 0xBF; + } + + // the first character is not the beginning of the BOM; unget it to + // process is later + unget(); + return true; + } + + void skip_whitespace() + { + do + { + get(); + } + while (current == ' ' || current == '\t' || current == '\n' || current == '\r'); + } + + token_type scan() + { + // initially, skip the BOM + if (position.chars_read_total == 0 && !skip_bom()) + { + error_message = "invalid BOM; must be 0xEF 0xBB 0xBF if given"; + return token_type::parse_error; + } + + // read next character and ignore whitespace + skip_whitespace(); + + // ignore comments + while (ignore_comments && current == '/') + { + if (!scan_comment()) + { + return token_type::parse_error; + } + + // skip following whitespace + skip_whitespace(); + } + + switch (current) + { + // structural characters + case '[': + return token_type::begin_array; + case ']': + return token_type::end_array; + case '{': + return token_type::begin_object; + case '}': + return token_type::end_object; + case ':': + return token_type::name_separator; + case ',': + return token_type::value_separator; + + // literals + case 't': + { + std::array true_literal = {{char_type('t'), char_type('r'), char_type('u'), char_type('e')}}; + return scan_literal(true_literal.data(), true_literal.size(), token_type::literal_true); + } + case 'f': + { + std::array false_literal = {{char_type('f'), char_type('a'), char_type('l'), char_type('s'), char_type('e')}}; + return scan_literal(false_literal.data(), false_literal.size(), token_type::literal_false); + } + case 'n': + { + std::array null_literal = {{char_type('n'), char_type('u'), char_type('l'), char_type('l')}}; + return scan_literal(null_literal.data(), null_literal.size(), token_type::literal_null); + } + + // string + case '\"': + return scan_string(); + + // number + case '-': + case '0': + case '1': + case '2': + case '3': + case '4': + case '5': + case '6': + case '7': + case '8': + case '9': + return scan_number(); + + // end of input (the null byte is needed when parsing from + // string literals) + case '\0': + case std::char_traits::eof(): + return token_type::end_of_input; + + // error + default: + error_message = "invalid literal"; + return token_type::parse_error; + } + } + + private: + /// input adapter + InputAdapterType ia; + + /// whether comments should be ignored (true) or signaled as errors (false) + const bool ignore_comments = false; + + /// the current character + char_int_type current = std::char_traits::eof(); + + /// whether the next get() call should just return current + bool next_unget = false; + + /// the start position of the current token + position_t position {}; + + /// raw input token string (for error messages) + std::vector token_string {}; + + /// buffer for variable-length tokens (numbers, strings) + string_t token_buffer {}; + + /// a description of occurred lexer errors + const char* error_message = ""; + + // number values + number_integer_t value_integer = 0; + number_unsigned_t value_unsigned = 0; + number_float_t value_float = 0; + + /// the decimal point + const char_int_type decimal_point_char = '.'; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // size_t +#include // declval +#include // string + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template +using null_function_t = decltype(std::declval().null()); + +template +using boolean_function_t = + decltype(std::declval().boolean(std::declval())); + +template +using number_integer_function_t = + decltype(std::declval().number_integer(std::declval())); + +template +using number_unsigned_function_t = + decltype(std::declval().number_unsigned(std::declval())); + +template +using number_float_function_t = decltype(std::declval().number_float( + std::declval(), std::declval())); + +template +using string_function_t = + decltype(std::declval().string(std::declval())); + +template +using binary_function_t = + decltype(std::declval().binary(std::declval())); + +template +using start_object_function_t = + decltype(std::declval().start_object(std::declval())); + +template +using key_function_t = + decltype(std::declval().key(std::declval())); + +template +using end_object_function_t = decltype(std::declval().end_object()); + +template +using start_array_function_t = + decltype(std::declval().start_array(std::declval())); + +template +using end_array_function_t = decltype(std::declval().end_array()); + +template +using parse_error_function_t = decltype(std::declval().parse_error( + std::declval(), std::declval(), + std::declval())); + +template +struct is_sax +{ + private: + static_assert(is_basic_json::value, + "BasicJsonType must be of type basic_json<...>"); + + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + using exception_t = typename BasicJsonType::exception; + + public: + static constexpr bool value = + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value && + is_detected_exact::value; +}; + +template +struct is_sax_static_asserts +{ + private: + static_assert(is_basic_json::value, + "BasicJsonType must be of type basic_json<...>"); + + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + using exception_t = typename BasicJsonType::exception; + + public: + static_assert(is_detected_exact::value, + "Missing/invalid function: bool null()"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool boolean(bool)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool boolean(bool)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool number_integer(number_integer_t)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool number_unsigned(number_unsigned_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool number_float(number_float_t, const string_t&)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool string(string_t&)"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool binary(binary_t&)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool start_object(std::size_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool key(string_t&)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool end_object()"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool start_array(std::size_t)"); + static_assert(is_detected_exact::value, + "Missing/invalid function: bool end_array()"); + static_assert( + is_detected_exact::value, + "Missing/invalid function: bool parse_error(std::size_t, const " + "std::string&, const exception&)"); +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ + +/// how to treat CBOR tags +enum class cbor_tag_handler_t +{ + error, ///< throw a parse_error exception in case of a tag + ignore, ///< ignore tags + store ///< store tags as binary type +}; + +/*! +@brief determine system byte order + +@return true if and only if system's byte order is little endian + +@note from https://stackoverflow.com/a/1001328/266378 +*/ +static inline bool little_endianess(int num = 1) noexcept +{ + return *reinterpret_cast(&num) == 1; +} + + +/////////////////// +// binary reader // +/////////////////// + +/*! +@brief deserialization of CBOR, MessagePack, and UBJSON values +*/ +template> +class binary_reader +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + using json_sax_t = SAX; + using char_type = typename InputAdapterType::char_type; + using char_int_type = typename std::char_traits::int_type; + + public: + /*! + @brief create a binary reader + + @param[in] adapter input adapter to read from + */ + explicit binary_reader(InputAdapterType&& adapter) noexcept : ia(std::move(adapter)) + { + (void)detail::is_sax_static_asserts {}; + } + + // make class move-only + binary_reader(const binary_reader&) = delete; + binary_reader(binary_reader&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + binary_reader& operator=(const binary_reader&) = delete; + binary_reader& operator=(binary_reader&&) = default; // NOLINT(hicpp-noexcept-move,performance-noexcept-move-constructor) + ~binary_reader() = default; + + /*! + @param[in] format the binary format to parse + @param[in] sax_ a SAX event processor + @param[in] strict whether to expect the input to be consumed completed + @param[in] tag_handler how to treat CBOR tags + + @return whether parsing was successful + */ + JSON_HEDLEY_NON_NULL(3) + bool sax_parse(const input_format_t format, + json_sax_t* sax_, + const bool strict = true, + const cbor_tag_handler_t tag_handler = cbor_tag_handler_t::error) + { + sax = sax_; + bool result = false; + + switch (format) + { + case input_format_t::bson: + result = parse_bson_internal(); + break; + + case input_format_t::cbor: + result = parse_cbor_internal(true, tag_handler); + break; + + case input_format_t::msgpack: + result = parse_msgpack_internal(); + break; + + case input_format_t::ubjson: + result = parse_ubjson_internal(); + break; + + case input_format_t::json: // LCOV_EXCL_LINE + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + + // strict mode: next byte must be EOF + if (result && strict) + { + if (format == input_format_t::ubjson) + { + get_ignore_noop(); + } + else + { + get(); + } + + if (JSON_HEDLEY_UNLIKELY(current != std::char_traits::eof())) + { + return sax->parse_error(chars_read, get_token_string(), + parse_error::create(110, chars_read, exception_message(format, "expected end of input; last byte: 0x" + get_token_string(), "value"), BasicJsonType())); + } + } + + return result; + } + + private: + ////////// + // BSON // + ////////// + + /*! + @brief Reads in a BSON-object and passes it to the SAX-parser. + @return whether a valid BSON-value was passed to the SAX parser + */ + bool parse_bson_internal() + { + std::int32_t document_size{}; + get_number(input_format_t::bson, document_size); + + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(std::size_t(-1)))) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_bson_element_list(/*is_array*/false))) + { + return false; + } + + return sax->end_object(); + } + + /*! + @brief Parses a C-style string from the BSON input. + @param[in,out] result A reference to the string variable where the read + string is to be stored. + @return `true` if the \x00-byte indicating the end of the string was + encountered before the EOF; false` indicates an unexpected EOF. + */ + bool get_bson_cstr(string_t& result) + { + auto out = std::back_inserter(result); + while (true) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::bson, "cstring"))) + { + return false; + } + if (current == 0x00) + { + return true; + } + *out++ = static_cast(current); + } + } + + /*! + @brief Parses a zero-terminated string of length @a len from the BSON + input. + @param[in] len The length (including the zero-byte at the end) of the + string to be read. + @param[in,out] result A reference to the string variable where the read + string is to be stored. + @tparam NumberType The type of the length @a len + @pre len >= 1 + @return `true` if the string was successfully parsed + */ + template + bool get_bson_string(const NumberType len, string_t& result) + { + if (JSON_HEDLEY_UNLIKELY(len < 1)) + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "string length must be at least 1, is " + std::to_string(len), "string"), BasicJsonType())); + } + + return get_string(input_format_t::bson, len - static_cast(1), result) && get() != std::char_traits::eof(); + } + + /*! + @brief Parses a byte array input of length @a len from the BSON input. + @param[in] len The length of the byte array to be read. + @param[in,out] result A reference to the binary variable where the read + array is to be stored. + @tparam NumberType The type of the length @a len + @pre len >= 0 + @return `true` if the byte array was successfully parsed + */ + template + bool get_bson_binary(const NumberType len, binary_t& result) + { + if (JSON_HEDLEY_UNLIKELY(len < 0)) + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::bson, "byte array length cannot be negative, is " + std::to_string(len), "binary"), BasicJsonType())); + } + + // All BSON binary values have a subtype + std::uint8_t subtype{}; + get_number(input_format_t::bson, subtype); + result.set_subtype(subtype); + + return get_binary(input_format_t::bson, len, result); + } + + /*! + @brief Read a BSON document element of the given @a element_type. + @param[in] element_type The BSON element type, c.f. http://bsonspec.org/spec.html + @param[in] element_type_parse_position The position in the input stream, + where the `element_type` was read. + @warning Not all BSON element types are supported yet. An unsupported + @a element_type will give rise to a parse_error.114: + Unsupported BSON record type 0x... + @return whether a valid BSON-object/array was passed to the SAX parser + */ + bool parse_bson_element_internal(const char_int_type element_type, + const std::size_t element_type_parse_position) + { + switch (element_type) + { + case 0x01: // double + { + double number{}; + return get_number(input_format_t::bson, number) && sax->number_float(static_cast(number), ""); + } + + case 0x02: // string + { + std::int32_t len{}; + string_t value; + return get_number(input_format_t::bson, len) && get_bson_string(len, value) && sax->string(value); + } + + case 0x03: // object + { + return parse_bson_internal(); + } + + case 0x04: // array + { + return parse_bson_array(); + } + + case 0x05: // binary + { + std::int32_t len{}; + binary_t value; + return get_number(input_format_t::bson, len) && get_bson_binary(len, value) && sax->binary(value); + } + + case 0x08: // boolean + { + return sax->boolean(get() != 0); + } + + case 0x0A: // null + { + return sax->null(); + } + + case 0x10: // int32 + { + std::int32_t value{}; + return get_number(input_format_t::bson, value) && sax->number_integer(value); + } + + case 0x12: // int64 + { + std::int64_t value{}; + return get_number(input_format_t::bson, value) && sax->number_integer(value); + } + + default: // anything else not supported (yet) + { + std::array cr{{}}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(element_type)); // NOLINT(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + return sax->parse_error(element_type_parse_position, std::string(cr.data()), parse_error::create(114, element_type_parse_position, "Unsupported BSON record type 0x" + std::string(cr.data()), BasicJsonType())); + } + } + } + + /*! + @brief Read a BSON element list (as specified in the BSON-spec) + + The same binary layout is used for objects and arrays, hence it must be + indicated with the argument @a is_array which one is expected + (true --> array, false --> object). + + @param[in] is_array Determines if the element list being read is to be + treated as an object (@a is_array == false), or as an + array (@a is_array == true). + @return whether a valid BSON-object/array was passed to the SAX parser + */ + bool parse_bson_element_list(const bool is_array) + { + string_t key; + + while (auto element_type = get()) + { + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::bson, "element list"))) + { + return false; + } + + const std::size_t element_type_parse_position = chars_read; + if (JSON_HEDLEY_UNLIKELY(!get_bson_cstr(key))) + { + return false; + } + + if (!is_array && !sax->key(key)) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_bson_element_internal(element_type, element_type_parse_position))) + { + return false; + } + + // get_bson_cstr only appends + key.clear(); + } + + return true; + } + + /*! + @brief Reads an array from the BSON input and passes it to the SAX-parser. + @return whether a valid BSON-array was passed to the SAX parser + */ + bool parse_bson_array() + { + std::int32_t document_size{}; + get_number(input_format_t::bson, document_size); + + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(std::size_t(-1)))) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_bson_element_list(/*is_array*/true))) + { + return false; + } + + return sax->end_array(); + } + + ////////// + // CBOR // + ////////// + + /*! + @param[in] get_char whether a new character should be retrieved from the + input (true) or whether the last read character should + be considered instead (false) + @param[in] tag_handler how CBOR tags should be treated + + @return whether a valid CBOR value was passed to the SAX parser + */ + bool parse_cbor_internal(const bool get_char, + const cbor_tag_handler_t tag_handler) + { + switch (get_char ? get() : current) + { + // EOF + case std::char_traits::eof(): + return unexpect_eof(input_format_t::cbor, "value"); + + // Integer 0x00..0x17 (0..23) + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0A: + case 0x0B: + case 0x0C: + case 0x0D: + case 0x0E: + case 0x0F: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + return sax->number_unsigned(static_cast(current)); + + case 0x18: // Unsigned integer (one-byte uint8_t follows) + { + std::uint8_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_unsigned(number); + } + + case 0x19: // Unsigned integer (two-byte uint16_t follows) + { + std::uint16_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_unsigned(number); + } + + case 0x1A: // Unsigned integer (four-byte uint32_t follows) + { + std::uint32_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_unsigned(number); + } + + case 0x1B: // Unsigned integer (eight-byte uint64_t follows) + { + std::uint64_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_unsigned(number); + } + + // Negative integer -1-0x00..-1-0x17 (-1..-24) + case 0x20: + case 0x21: + case 0x22: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2A: + case 0x2B: + case 0x2C: + case 0x2D: + case 0x2E: + case 0x2F: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + return sax->number_integer(static_cast(0x20 - 1 - current)); + + case 0x38: // Negative integer (one-byte uint8_t follows) + { + std::uint8_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_integer(static_cast(-1) - number); + } + + case 0x39: // Negative integer -1-n (two-byte uint16_t follows) + { + std::uint16_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_integer(static_cast(-1) - number); + } + + case 0x3A: // Negative integer -1-n (four-byte uint32_t follows) + { + std::uint32_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_integer(static_cast(-1) - number); + } + + case 0x3B: // Negative integer -1-n (eight-byte uint64_t follows) + { + std::uint64_t number{}; + return get_number(input_format_t::cbor, number) && sax->number_integer(static_cast(-1) + - static_cast(number)); + } + + // Binary data (0x00..0x17 bytes follow) + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4A: + case 0x4B: + case 0x4C: + case 0x4D: + case 0x4E: + case 0x4F: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: // Binary data (one-byte uint8_t for n follows) + case 0x59: // Binary data (two-byte uint16_t for n follow) + case 0x5A: // Binary data (four-byte uint32_t for n follow) + case 0x5B: // Binary data (eight-byte uint64_t for n follow) + case 0x5F: // Binary data (indefinite length) + { + binary_t b; + return get_cbor_binary(b) && sax->binary(b); + } + + // UTF-8 string (0x00..0x17 bytes follow) + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6A: + case 0x6B: + case 0x6C: + case 0x6D: + case 0x6E: + case 0x6F: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: // UTF-8 string (one-byte uint8_t for n follows) + case 0x79: // UTF-8 string (two-byte uint16_t for n follow) + case 0x7A: // UTF-8 string (four-byte uint32_t for n follow) + case 0x7B: // UTF-8 string (eight-byte uint64_t for n follow) + case 0x7F: // UTF-8 string (indefinite length) + { + string_t s; + return get_cbor_string(s) && sax->string(s); + } + + // array (0x00..0x17 data items follow) + case 0x80: + case 0x81: + case 0x82: + case 0x83: + case 0x84: + case 0x85: + case 0x86: + case 0x87: + case 0x88: + case 0x89: + case 0x8A: + case 0x8B: + case 0x8C: + case 0x8D: + case 0x8E: + case 0x8F: + case 0x90: + case 0x91: + case 0x92: + case 0x93: + case 0x94: + case 0x95: + case 0x96: + case 0x97: + return get_cbor_array(static_cast(static_cast(current) & 0x1Fu), tag_handler); + + case 0x98: // array (one-byte uint8_t for n follows) + { + std::uint8_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_array(static_cast(len), tag_handler); + } + + case 0x99: // array (two-byte uint16_t for n follow) + { + std::uint16_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_array(static_cast(len), tag_handler); + } + + case 0x9A: // array (four-byte uint32_t for n follow) + { + std::uint32_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_array(static_cast(len), tag_handler); + } + + case 0x9B: // array (eight-byte uint64_t for n follow) + { + std::uint64_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_array(detail::conditional_static_cast(len), tag_handler); + } + + case 0x9F: // array (indefinite length) + return get_cbor_array(std::size_t(-1), tag_handler); + + // map (0x00..0x17 pairs of data items follow) + case 0xA0: + case 0xA1: + case 0xA2: + case 0xA3: + case 0xA4: + case 0xA5: + case 0xA6: + case 0xA7: + case 0xA8: + case 0xA9: + case 0xAA: + case 0xAB: + case 0xAC: + case 0xAD: + case 0xAE: + case 0xAF: + case 0xB0: + case 0xB1: + case 0xB2: + case 0xB3: + case 0xB4: + case 0xB5: + case 0xB6: + case 0xB7: + return get_cbor_object(static_cast(static_cast(current) & 0x1Fu), tag_handler); + + case 0xB8: // map (one-byte uint8_t for n follows) + { + std::uint8_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_object(static_cast(len), tag_handler); + } + + case 0xB9: // map (two-byte uint16_t for n follow) + { + std::uint16_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_object(static_cast(len), tag_handler); + } + + case 0xBA: // map (four-byte uint32_t for n follow) + { + std::uint32_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_object(static_cast(len), tag_handler); + } + + case 0xBB: // map (eight-byte uint64_t for n follow) + { + std::uint64_t len{}; + return get_number(input_format_t::cbor, len) && get_cbor_object(detail::conditional_static_cast(len), tag_handler); + } + + case 0xBF: // map (indefinite length) + return get_cbor_object(std::size_t(-1), tag_handler); + + case 0xC6: // tagged item + case 0xC7: + case 0xC8: + case 0xC9: + case 0xCA: + case 0xCB: + case 0xCC: + case 0xCD: + case 0xCE: + case 0xCF: + case 0xD0: + case 0xD1: + case 0xD2: + case 0xD3: + case 0xD4: + case 0xD8: // tagged item (1 bytes follow) + case 0xD9: // tagged item (2 bytes follow) + case 0xDA: // tagged item (4 bytes follow) + case 0xDB: // tagged item (8 bytes follow) + { + switch (tag_handler) + { + case cbor_tag_handler_t::error: + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::cbor, "invalid byte: 0x" + last_token, "value"), BasicJsonType())); + } + + case cbor_tag_handler_t::ignore: + { + // ignore binary subtype + switch (current) + { + case 0xD8: + { + std::uint8_t subtype_to_ignore{}; + get_number(input_format_t::cbor, subtype_to_ignore); + break; + } + case 0xD9: + { + std::uint16_t subtype_to_ignore{}; + get_number(input_format_t::cbor, subtype_to_ignore); + break; + } + case 0xDA: + { + std::uint32_t subtype_to_ignore{}; + get_number(input_format_t::cbor, subtype_to_ignore); + break; + } + case 0xDB: + { + std::uint64_t subtype_to_ignore{}; + get_number(input_format_t::cbor, subtype_to_ignore); + break; + } + default: + break; + } + return parse_cbor_internal(true, tag_handler); + } + + case cbor_tag_handler_t::store: + { + binary_t b; + // use binary subtype and store in binary container + switch (current) + { + case 0xD8: + { + std::uint8_t subtype{}; + get_number(input_format_t::cbor, subtype); + b.set_subtype(detail::conditional_static_cast(subtype)); + break; + } + case 0xD9: + { + std::uint16_t subtype{}; + get_number(input_format_t::cbor, subtype); + b.set_subtype(detail::conditional_static_cast(subtype)); + break; + } + case 0xDA: + { + std::uint32_t subtype{}; + get_number(input_format_t::cbor, subtype); + b.set_subtype(detail::conditional_static_cast(subtype)); + break; + } + case 0xDB: + { + std::uint64_t subtype{}; + get_number(input_format_t::cbor, subtype); + b.set_subtype(detail::conditional_static_cast(subtype)); + break; + } + default: + return parse_cbor_internal(true, tag_handler); + } + get(); + return get_cbor_binary(b) && sax->binary(b); + } + + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + return false; // LCOV_EXCL_LINE + } + } + + case 0xF4: // false + return sax->boolean(false); + + case 0xF5: // true + return sax->boolean(true); + + case 0xF6: // null + return sax->null(); + + case 0xF9: // Half-Precision Float (two-byte IEEE 754) + { + const auto byte1_raw = get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::cbor, "number"))) + { + return false; + } + const auto byte2_raw = get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::cbor, "number"))) + { + return false; + } + + const auto byte1 = static_cast(byte1_raw); + const auto byte2 = static_cast(byte2_raw); + + // code from RFC 7049, Appendix D, Figure 3: + // As half-precision floating-point numbers were only added + // to IEEE 754 in 2008, today's programming platforms often + // still only have limited support for them. It is very + // easy to include at least decoding support for them even + // without such support. An example of a small decoder for + // half-precision floating-point numbers in the C language + // is shown in Fig. 3. + const auto half = static_cast((byte1 << 8u) + byte2); + const double val = [&half] + { + const int exp = (half >> 10u) & 0x1Fu; + const unsigned int mant = half & 0x3FFu; + JSON_ASSERT(0 <= exp&& exp <= 32); + JSON_ASSERT(mant <= 1024); + switch (exp) + { + case 0: + return std::ldexp(mant, -24); + case 31: + return (mant == 0) + ? std::numeric_limits::infinity() + : std::numeric_limits::quiet_NaN(); + default: + return std::ldexp(mant + 1024, exp - 25); + } + }(); + return sax->number_float((half & 0x8000u) != 0 + ? static_cast(-val) + : static_cast(val), ""); + } + + case 0xFA: // Single-Precision Float (four-byte IEEE 754) + { + float number{}; + return get_number(input_format_t::cbor, number) && sax->number_float(static_cast(number), ""); + } + + case 0xFB: // Double-Precision Float (eight-byte IEEE 754) + { + double number{}; + return get_number(input_format_t::cbor, number) && sax->number_float(static_cast(number), ""); + } + + default: // anything else (0xFF is handled inside the other types) + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::cbor, "invalid byte: 0x" + last_token, "value"), BasicJsonType())); + } + } + } + + /*! + @brief reads a CBOR string + + This function first reads starting bytes to determine the expected + string length and then copies this number of bytes into a string. + Additionally, CBOR's strings with indefinite lengths are supported. + + @param[out] result created string + + @return whether string creation completed + */ + bool get_cbor_string(string_t& result) + { + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::cbor, "string"))) + { + return false; + } + + switch (current) + { + // UTF-8 string (0x00..0x17 bytes follow) + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6A: + case 0x6B: + case 0x6C: + case 0x6D: + case 0x6E: + case 0x6F: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + { + return get_string(input_format_t::cbor, static_cast(current) & 0x1Fu, result); + } + + case 0x78: // UTF-8 string (one-byte uint8_t for n follows) + { + std::uint8_t len{}; + return get_number(input_format_t::cbor, len) && get_string(input_format_t::cbor, len, result); + } + + case 0x79: // UTF-8 string (two-byte uint16_t for n follow) + { + std::uint16_t len{}; + return get_number(input_format_t::cbor, len) && get_string(input_format_t::cbor, len, result); + } + + case 0x7A: // UTF-8 string (four-byte uint32_t for n follow) + { + std::uint32_t len{}; + return get_number(input_format_t::cbor, len) && get_string(input_format_t::cbor, len, result); + } + + case 0x7B: // UTF-8 string (eight-byte uint64_t for n follow) + { + std::uint64_t len{}; + return get_number(input_format_t::cbor, len) && get_string(input_format_t::cbor, len, result); + } + + case 0x7F: // UTF-8 string (indefinite length) + { + while (get() != 0xFF) + { + string_t chunk; + if (!get_cbor_string(chunk)) + { + return false; + } + result.append(chunk); + } + return true; + } + + default: + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::cbor, "expected length specification (0x60-0x7B) or indefinite string type (0x7F); last byte: 0x" + last_token, "string"), BasicJsonType())); + } + } + } + + /*! + @brief reads a CBOR byte array + + This function first reads starting bytes to determine the expected + byte array length and then copies this number of bytes into the byte array. + Additionally, CBOR's byte arrays with indefinite lengths are supported. + + @param[out] result created byte array + + @return whether byte array creation completed + */ + bool get_cbor_binary(binary_t& result) + { + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::cbor, "binary"))) + { + return false; + } + + switch (current) + { + // Binary data (0x00..0x17 bytes follow) + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4A: + case 0x4B: + case 0x4C: + case 0x4D: + case 0x4E: + case 0x4F: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + { + return get_binary(input_format_t::cbor, static_cast(current) & 0x1Fu, result); + } + + case 0x58: // Binary data (one-byte uint8_t for n follows) + { + std::uint8_t len{}; + return get_number(input_format_t::cbor, len) && + get_binary(input_format_t::cbor, len, result); + } + + case 0x59: // Binary data (two-byte uint16_t for n follow) + { + std::uint16_t len{}; + return get_number(input_format_t::cbor, len) && + get_binary(input_format_t::cbor, len, result); + } + + case 0x5A: // Binary data (four-byte uint32_t for n follow) + { + std::uint32_t len{}; + return get_number(input_format_t::cbor, len) && + get_binary(input_format_t::cbor, len, result); + } + + case 0x5B: // Binary data (eight-byte uint64_t for n follow) + { + std::uint64_t len{}; + return get_number(input_format_t::cbor, len) && + get_binary(input_format_t::cbor, len, result); + } + + case 0x5F: // Binary data (indefinite length) + { + while (get() != 0xFF) + { + binary_t chunk; + if (!get_cbor_binary(chunk)) + { + return false; + } + result.insert(result.end(), chunk.begin(), chunk.end()); + } + return true; + } + + default: + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::cbor, "expected length specification (0x40-0x5B) or indefinite binary array type (0x5F); last byte: 0x" + last_token, "binary"), BasicJsonType())); + } + } + } + + /*! + @param[in] len the length of the array or std::size_t(-1) for an + array of indefinite size + @param[in] tag_handler how CBOR tags should be treated + @return whether array creation completed + */ + bool get_cbor_array(const std::size_t len, + const cbor_tag_handler_t tag_handler) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(len))) + { + return false; + } + + if (len != std::size_t(-1)) + { + for (std::size_t i = 0; i < len; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!parse_cbor_internal(true, tag_handler))) + { + return false; + } + } + } + else + { + while (get() != 0xFF) + { + if (JSON_HEDLEY_UNLIKELY(!parse_cbor_internal(false, tag_handler))) + { + return false; + } + } + } + + return sax->end_array(); + } + + /*! + @param[in] len the length of the object or std::size_t(-1) for an + object of indefinite size + @param[in] tag_handler how CBOR tags should be treated + @return whether object creation completed + */ + bool get_cbor_object(const std::size_t len, + const cbor_tag_handler_t tag_handler) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(len))) + { + return false; + } + + if (len != 0) + { + string_t key; + if (len != std::size_t(-1)) + { + for (std::size_t i = 0; i < len; ++i) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!get_cbor_string(key) || !sax->key(key))) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_cbor_internal(true, tag_handler))) + { + return false; + } + key.clear(); + } + } + else + { + while (get() != 0xFF) + { + if (JSON_HEDLEY_UNLIKELY(!get_cbor_string(key) || !sax->key(key))) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_cbor_internal(true, tag_handler))) + { + return false; + } + key.clear(); + } + } + } + + return sax->end_object(); + } + + ///////////// + // MsgPack // + ///////////// + + /*! + @return whether a valid MessagePack value was passed to the SAX parser + */ + bool parse_msgpack_internal() + { + switch (get()) + { + // EOF + case std::char_traits::eof(): + return unexpect_eof(input_format_t::msgpack, "value"); + + // positive fixint + case 0x00: + case 0x01: + case 0x02: + case 0x03: + case 0x04: + case 0x05: + case 0x06: + case 0x07: + case 0x08: + case 0x09: + case 0x0A: + case 0x0B: + case 0x0C: + case 0x0D: + case 0x0E: + case 0x0F: + case 0x10: + case 0x11: + case 0x12: + case 0x13: + case 0x14: + case 0x15: + case 0x16: + case 0x17: + case 0x18: + case 0x19: + case 0x1A: + case 0x1B: + case 0x1C: + case 0x1D: + case 0x1E: + case 0x1F: + case 0x20: + case 0x21: + case 0x22: + case 0x23: + case 0x24: + case 0x25: + case 0x26: + case 0x27: + case 0x28: + case 0x29: + case 0x2A: + case 0x2B: + case 0x2C: + case 0x2D: + case 0x2E: + case 0x2F: + case 0x30: + case 0x31: + case 0x32: + case 0x33: + case 0x34: + case 0x35: + case 0x36: + case 0x37: + case 0x38: + case 0x39: + case 0x3A: + case 0x3B: + case 0x3C: + case 0x3D: + case 0x3E: + case 0x3F: + case 0x40: + case 0x41: + case 0x42: + case 0x43: + case 0x44: + case 0x45: + case 0x46: + case 0x47: + case 0x48: + case 0x49: + case 0x4A: + case 0x4B: + case 0x4C: + case 0x4D: + case 0x4E: + case 0x4F: + case 0x50: + case 0x51: + case 0x52: + case 0x53: + case 0x54: + case 0x55: + case 0x56: + case 0x57: + case 0x58: + case 0x59: + case 0x5A: + case 0x5B: + case 0x5C: + case 0x5D: + case 0x5E: + case 0x5F: + case 0x60: + case 0x61: + case 0x62: + case 0x63: + case 0x64: + case 0x65: + case 0x66: + case 0x67: + case 0x68: + case 0x69: + case 0x6A: + case 0x6B: + case 0x6C: + case 0x6D: + case 0x6E: + case 0x6F: + case 0x70: + case 0x71: + case 0x72: + case 0x73: + case 0x74: + case 0x75: + case 0x76: + case 0x77: + case 0x78: + case 0x79: + case 0x7A: + case 0x7B: + case 0x7C: + case 0x7D: + case 0x7E: + case 0x7F: + return sax->number_unsigned(static_cast(current)); + + // fixmap + case 0x80: + case 0x81: + case 0x82: + case 0x83: + case 0x84: + case 0x85: + case 0x86: + case 0x87: + case 0x88: + case 0x89: + case 0x8A: + case 0x8B: + case 0x8C: + case 0x8D: + case 0x8E: + case 0x8F: + return get_msgpack_object(static_cast(static_cast(current) & 0x0Fu)); + + // fixarray + case 0x90: + case 0x91: + case 0x92: + case 0x93: + case 0x94: + case 0x95: + case 0x96: + case 0x97: + case 0x98: + case 0x99: + case 0x9A: + case 0x9B: + case 0x9C: + case 0x9D: + case 0x9E: + case 0x9F: + return get_msgpack_array(static_cast(static_cast(current) & 0x0Fu)); + + // fixstr + case 0xA0: + case 0xA1: + case 0xA2: + case 0xA3: + case 0xA4: + case 0xA5: + case 0xA6: + case 0xA7: + case 0xA8: + case 0xA9: + case 0xAA: + case 0xAB: + case 0xAC: + case 0xAD: + case 0xAE: + case 0xAF: + case 0xB0: + case 0xB1: + case 0xB2: + case 0xB3: + case 0xB4: + case 0xB5: + case 0xB6: + case 0xB7: + case 0xB8: + case 0xB9: + case 0xBA: + case 0xBB: + case 0xBC: + case 0xBD: + case 0xBE: + case 0xBF: + case 0xD9: // str 8 + case 0xDA: // str 16 + case 0xDB: // str 32 + { + string_t s; + return get_msgpack_string(s) && sax->string(s); + } + + case 0xC0: // nil + return sax->null(); + + case 0xC2: // false + return sax->boolean(false); + + case 0xC3: // true + return sax->boolean(true); + + case 0xC4: // bin 8 + case 0xC5: // bin 16 + case 0xC6: // bin 32 + case 0xC7: // ext 8 + case 0xC8: // ext 16 + case 0xC9: // ext 32 + case 0xD4: // fixext 1 + case 0xD5: // fixext 2 + case 0xD6: // fixext 4 + case 0xD7: // fixext 8 + case 0xD8: // fixext 16 + { + binary_t b; + return get_msgpack_binary(b) && sax->binary(b); + } + + case 0xCA: // float 32 + { + float number{}; + return get_number(input_format_t::msgpack, number) && sax->number_float(static_cast(number), ""); + } + + case 0xCB: // float 64 + { + double number{}; + return get_number(input_format_t::msgpack, number) && sax->number_float(static_cast(number), ""); + } + + case 0xCC: // uint 8 + { + std::uint8_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_unsigned(number); + } + + case 0xCD: // uint 16 + { + std::uint16_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_unsigned(number); + } + + case 0xCE: // uint 32 + { + std::uint32_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_unsigned(number); + } + + case 0xCF: // uint 64 + { + std::uint64_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_unsigned(number); + } + + case 0xD0: // int 8 + { + std::int8_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_integer(number); + } + + case 0xD1: // int 16 + { + std::int16_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_integer(number); + } + + case 0xD2: // int 32 + { + std::int32_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_integer(number); + } + + case 0xD3: // int 64 + { + std::int64_t number{}; + return get_number(input_format_t::msgpack, number) && sax->number_integer(number); + } + + case 0xDC: // array 16 + { + std::uint16_t len{}; + return get_number(input_format_t::msgpack, len) && get_msgpack_array(static_cast(len)); + } + + case 0xDD: // array 32 + { + std::uint32_t len{}; + return get_number(input_format_t::msgpack, len) && get_msgpack_array(static_cast(len)); + } + + case 0xDE: // map 16 + { + std::uint16_t len{}; + return get_number(input_format_t::msgpack, len) && get_msgpack_object(static_cast(len)); + } + + case 0xDF: // map 32 + { + std::uint32_t len{}; + return get_number(input_format_t::msgpack, len) && get_msgpack_object(static_cast(len)); + } + + // negative fixint + case 0xE0: + case 0xE1: + case 0xE2: + case 0xE3: + case 0xE4: + case 0xE5: + case 0xE6: + case 0xE7: + case 0xE8: + case 0xE9: + case 0xEA: + case 0xEB: + case 0xEC: + case 0xED: + case 0xEE: + case 0xEF: + case 0xF0: + case 0xF1: + case 0xF2: + case 0xF3: + case 0xF4: + case 0xF5: + case 0xF6: + case 0xF7: + case 0xF8: + case 0xF9: + case 0xFA: + case 0xFB: + case 0xFC: + case 0xFD: + case 0xFE: + case 0xFF: + return sax->number_integer(static_cast(current)); + + default: // anything else + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::msgpack, "invalid byte: 0x" + last_token, "value"), BasicJsonType())); + } + } + } + + /*! + @brief reads a MessagePack string + + This function first reads starting bytes to determine the expected + string length and then copies this number of bytes into a string. + + @param[out] result created string + + @return whether string creation completed + */ + bool get_msgpack_string(string_t& result) + { + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::msgpack, "string"))) + { + return false; + } + + switch (current) + { + // fixstr + case 0xA0: + case 0xA1: + case 0xA2: + case 0xA3: + case 0xA4: + case 0xA5: + case 0xA6: + case 0xA7: + case 0xA8: + case 0xA9: + case 0xAA: + case 0xAB: + case 0xAC: + case 0xAD: + case 0xAE: + case 0xAF: + case 0xB0: + case 0xB1: + case 0xB2: + case 0xB3: + case 0xB4: + case 0xB5: + case 0xB6: + case 0xB7: + case 0xB8: + case 0xB9: + case 0xBA: + case 0xBB: + case 0xBC: + case 0xBD: + case 0xBE: + case 0xBF: + { + return get_string(input_format_t::msgpack, static_cast(current) & 0x1Fu, result); + } + + case 0xD9: // str 8 + { + std::uint8_t len{}; + return get_number(input_format_t::msgpack, len) && get_string(input_format_t::msgpack, len, result); + } + + case 0xDA: // str 16 + { + std::uint16_t len{}; + return get_number(input_format_t::msgpack, len) && get_string(input_format_t::msgpack, len, result); + } + + case 0xDB: // str 32 + { + std::uint32_t len{}; + return get_number(input_format_t::msgpack, len) && get_string(input_format_t::msgpack, len, result); + } + + default: + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::msgpack, "expected length specification (0xA0-0xBF, 0xD9-0xDB); last byte: 0x" + last_token, "string"), BasicJsonType())); + } + } + } + + /*! + @brief reads a MessagePack byte array + + This function first reads starting bytes to determine the expected + byte array length and then copies this number of bytes into a byte array. + + @param[out] result created byte array + + @return whether byte array creation completed + */ + bool get_msgpack_binary(binary_t& result) + { + // helper function to set the subtype + auto assign_and_return_true = [&result](std::int8_t subtype) + { + result.set_subtype(static_cast(subtype)); + return true; + }; + + switch (current) + { + case 0xC4: // bin 8 + { + std::uint8_t len{}; + return get_number(input_format_t::msgpack, len) && + get_binary(input_format_t::msgpack, len, result); + } + + case 0xC5: // bin 16 + { + std::uint16_t len{}; + return get_number(input_format_t::msgpack, len) && + get_binary(input_format_t::msgpack, len, result); + } + + case 0xC6: // bin 32 + { + std::uint32_t len{}; + return get_number(input_format_t::msgpack, len) && + get_binary(input_format_t::msgpack, len, result); + } + + case 0xC7: // ext 8 + { + std::uint8_t len{}; + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, len) && + get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, len, result) && + assign_and_return_true(subtype); + } + + case 0xC8: // ext 16 + { + std::uint16_t len{}; + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, len) && + get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, len, result) && + assign_and_return_true(subtype); + } + + case 0xC9: // ext 32 + { + std::uint32_t len{}; + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, len) && + get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, len, result) && + assign_and_return_true(subtype); + } + + case 0xD4: // fixext 1 + { + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, 1, result) && + assign_and_return_true(subtype); + } + + case 0xD5: // fixext 2 + { + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, 2, result) && + assign_and_return_true(subtype); + } + + case 0xD6: // fixext 4 + { + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, 4, result) && + assign_and_return_true(subtype); + } + + case 0xD7: // fixext 8 + { + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, 8, result) && + assign_and_return_true(subtype); + } + + case 0xD8: // fixext 16 + { + std::int8_t subtype{}; + return get_number(input_format_t::msgpack, subtype) && + get_binary(input_format_t::msgpack, 16, result) && + assign_and_return_true(subtype); + } + + default: // LCOV_EXCL_LINE + return false; // LCOV_EXCL_LINE + } + } + + /*! + @param[in] len the length of the array + @return whether array creation completed + */ + bool get_msgpack_array(const std::size_t len) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(len))) + { + return false; + } + + for (std::size_t i = 0; i < len; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!parse_msgpack_internal())) + { + return false; + } + } + + return sax->end_array(); + } + + /*! + @param[in] len the length of the object + @return whether object creation completed + */ + bool get_msgpack_object(const std::size_t len) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(len))) + { + return false; + } + + string_t key; + for (std::size_t i = 0; i < len; ++i) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!get_msgpack_string(key) || !sax->key(key))) + { + return false; + } + + if (JSON_HEDLEY_UNLIKELY(!parse_msgpack_internal())) + { + return false; + } + key.clear(); + } + + return sax->end_object(); + } + + //////////// + // UBJSON // + //////////// + + /*! + @param[in] get_char whether a new character should be retrieved from the + input (true, default) or whether the last read + character should be considered instead + + @return whether a valid UBJSON value was passed to the SAX parser + */ + bool parse_ubjson_internal(const bool get_char = true) + { + return get_ubjson_value(get_char ? get_ignore_noop() : current); + } + + /*! + @brief reads a UBJSON string + + This function is either called after reading the 'S' byte explicitly + indicating a string, or in case of an object key where the 'S' byte can be + left out. + + @param[out] result created string + @param[in] get_char whether a new character should be retrieved from the + input (true, default) or whether the last read + character should be considered instead + + @return whether string creation completed + */ + bool get_ubjson_string(string_t& result, const bool get_char = true) + { + if (get_char) + { + get(); // TODO(niels): may we ignore N here? + } + + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::ubjson, "value"))) + { + return false; + } + + switch (current) + { + case 'U': + { + std::uint8_t len{}; + return get_number(input_format_t::ubjson, len) && get_string(input_format_t::ubjson, len, result); + } + + case 'i': + { + std::int8_t len{}; + return get_number(input_format_t::ubjson, len) && get_string(input_format_t::ubjson, len, result); + } + + case 'I': + { + std::int16_t len{}; + return get_number(input_format_t::ubjson, len) && get_string(input_format_t::ubjson, len, result); + } + + case 'l': + { + std::int32_t len{}; + return get_number(input_format_t::ubjson, len) && get_string(input_format_t::ubjson, len, result); + } + + case 'L': + { + std::int64_t len{}; + return get_number(input_format_t::ubjson, len) && get_string(input_format_t::ubjson, len, result); + } + + default: + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::ubjson, "expected length type specification (U, i, I, l, L); last byte: 0x" + last_token, "string"), BasicJsonType())); + } + } + + /*! + @param[out] result determined size + @return whether size determination completed + */ + bool get_ubjson_size_value(std::size_t& result) + { + switch (get_ignore_noop()) + { + case 'U': + { + std::uint8_t number{}; + if (JSON_HEDLEY_UNLIKELY(!get_number(input_format_t::ubjson, number))) + { + return false; + } + result = static_cast(number); + return true; + } + + case 'i': + { + std::int8_t number{}; + if (JSON_HEDLEY_UNLIKELY(!get_number(input_format_t::ubjson, number))) + { + return false; + } + result = static_cast(number); // NOLINT(bugprone-signed-char-misuse,cert-str34-c): number is not a char + return true; + } + + case 'I': + { + std::int16_t number{}; + if (JSON_HEDLEY_UNLIKELY(!get_number(input_format_t::ubjson, number))) + { + return false; + } + result = static_cast(number); + return true; + } + + case 'l': + { + std::int32_t number{}; + if (JSON_HEDLEY_UNLIKELY(!get_number(input_format_t::ubjson, number))) + { + return false; + } + result = static_cast(number); + return true; + } + + case 'L': + { + std::int64_t number{}; + if (JSON_HEDLEY_UNLIKELY(!get_number(input_format_t::ubjson, number))) + { + return false; + } + result = static_cast(number); + return true; + } + + default: + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::ubjson, "expected length type specification (U, i, I, l, L) after '#'; last byte: 0x" + last_token, "size"), BasicJsonType())); + } + } + } + + /*! + @brief determine the type and size for a container + + In the optimized UBJSON format, a type and a size can be provided to allow + for a more compact representation. + + @param[out] result pair of the size and the type + + @return whether pair creation completed + */ + bool get_ubjson_size_type(std::pair& result) + { + result.first = string_t::npos; // size + result.second = 0; // type + + get_ignore_noop(); + + if (current == '$') + { + result.second = get(); // must not ignore 'N', because 'N' maybe the type + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::ubjson, "type"))) + { + return false; + } + + get_ignore_noop(); + if (JSON_HEDLEY_UNLIKELY(current != '#')) + { + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::ubjson, "value"))) + { + return false; + } + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::ubjson, "expected '#' after type information; last byte: 0x" + last_token, "size"), BasicJsonType())); + } + + return get_ubjson_size_value(result.first); + } + + if (current == '#') + { + return get_ubjson_size_value(result.first); + } + + return true; + } + + /*! + @param prefix the previously read or set type prefix + @return whether value creation completed + */ + bool get_ubjson_value(const char_int_type prefix) + { + switch (prefix) + { + case std::char_traits::eof(): // EOF + return unexpect_eof(input_format_t::ubjson, "value"); + + case 'T': // true + return sax->boolean(true); + case 'F': // false + return sax->boolean(false); + + case 'Z': // null + return sax->null(); + + case 'U': + { + std::uint8_t number{}; + return get_number(input_format_t::ubjson, number) && sax->number_unsigned(number); + } + + case 'i': + { + std::int8_t number{}; + return get_number(input_format_t::ubjson, number) && sax->number_integer(number); + } + + case 'I': + { + std::int16_t number{}; + return get_number(input_format_t::ubjson, number) && sax->number_integer(number); + } + + case 'l': + { + std::int32_t number{}; + return get_number(input_format_t::ubjson, number) && sax->number_integer(number); + } + + case 'L': + { + std::int64_t number{}; + return get_number(input_format_t::ubjson, number) && sax->number_integer(number); + } + + case 'd': + { + float number{}; + return get_number(input_format_t::ubjson, number) && sax->number_float(static_cast(number), ""); + } + + case 'D': + { + double number{}; + return get_number(input_format_t::ubjson, number) && sax->number_float(static_cast(number), ""); + } + + case 'H': + { + return get_ubjson_high_precision_number(); + } + + case 'C': // char + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::ubjson, "char"))) + { + return false; + } + if (JSON_HEDLEY_UNLIKELY(current > 127)) + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(113, chars_read, exception_message(input_format_t::ubjson, "byte after 'C' must be in range 0x00..0x7F; last byte: 0x" + last_token, "char"), BasicJsonType())); + } + string_t s(1, static_cast(current)); + return sax->string(s); + } + + case 'S': // string + { + string_t s; + return get_ubjson_string(s) && sax->string(s); + } + + case '[': // array + return get_ubjson_array(); + + case '{': // object + return get_ubjson_object(); + + default: // anything else + { + auto last_token = get_token_string(); + return sax->parse_error(chars_read, last_token, parse_error::create(112, chars_read, exception_message(input_format_t::ubjson, "invalid byte: 0x" + last_token, "value"), BasicJsonType())); + } + } + } + + /*! + @return whether array creation completed + */ + bool get_ubjson_array() + { + std::pair size_and_type; + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_size_type(size_and_type))) + { + return false; + } + + if (size_and_type.first != string_t::npos) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(size_and_type.first))) + { + return false; + } + + if (size_and_type.second != 0) + { + if (size_and_type.second != 'N') + { + for (std::size_t i = 0; i < size_and_type.first; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_value(size_and_type.second))) + { + return false; + } + } + } + } + else + { + for (std::size_t i = 0; i < size_and_type.first; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!parse_ubjson_internal())) + { + return false; + } + } + } + } + else + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(std::size_t(-1)))) + { + return false; + } + + while (current != ']') + { + if (JSON_HEDLEY_UNLIKELY(!parse_ubjson_internal(false))) + { + return false; + } + get_ignore_noop(); + } + } + + return sax->end_array(); + } + + /*! + @return whether object creation completed + */ + bool get_ubjson_object() + { + std::pair size_and_type; + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_size_type(size_and_type))) + { + return false; + } + + string_t key; + if (size_and_type.first != string_t::npos) + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(size_and_type.first))) + { + return false; + } + + if (size_and_type.second != 0) + { + for (std::size_t i = 0; i < size_and_type.first; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_string(key) || !sax->key(key))) + { + return false; + } + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_value(size_and_type.second))) + { + return false; + } + key.clear(); + } + } + else + { + for (std::size_t i = 0; i < size_and_type.first; ++i) + { + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_string(key) || !sax->key(key))) + { + return false; + } + if (JSON_HEDLEY_UNLIKELY(!parse_ubjson_internal())) + { + return false; + } + key.clear(); + } + } + } + else + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(std::size_t(-1)))) + { + return false; + } + + while (current != '}') + { + if (JSON_HEDLEY_UNLIKELY(!get_ubjson_string(key, false) || !sax->key(key))) + { + return false; + } + if (JSON_HEDLEY_UNLIKELY(!parse_ubjson_internal())) + { + return false; + } + get_ignore_noop(); + key.clear(); + } + } + + return sax->end_object(); + } + + // Note, no reader for UBJSON binary types is implemented because they do + // not exist + + bool get_ubjson_high_precision_number() + { + // get size of following number string + std::size_t size{}; + auto res = get_ubjson_size_value(size); + if (JSON_HEDLEY_UNLIKELY(!res)) + { + return res; + } + + // get number string + std::vector number_vector; + for (std::size_t i = 0; i < size; ++i) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(input_format_t::ubjson, "number"))) + { + return false; + } + number_vector.push_back(static_cast(current)); + } + + // parse number string + using ia_type = decltype(detail::input_adapter(number_vector)); + auto number_lexer = detail::lexer(detail::input_adapter(number_vector), false); + const auto result_number = number_lexer.scan(); + const auto number_string = number_lexer.get_token_string(); + const auto result_remainder = number_lexer.scan(); + + using token_type = typename detail::lexer_base::token_type; + + if (JSON_HEDLEY_UNLIKELY(result_remainder != token_type::end_of_input)) + { + return sax->parse_error(chars_read, number_string, parse_error::create(115, chars_read, exception_message(input_format_t::ubjson, "invalid number text: " + number_lexer.get_token_string(), "high-precision number"), BasicJsonType())); + } + + switch (result_number) + { + case token_type::value_integer: + return sax->number_integer(number_lexer.get_number_integer()); + case token_type::value_unsigned: + return sax->number_unsigned(number_lexer.get_number_unsigned()); + case token_type::value_float: + return sax->number_float(number_lexer.get_number_float(), std::move(number_string)); + case token_type::uninitialized: + case token_type::literal_true: + case token_type::literal_false: + case token_type::literal_null: + case token_type::value_string: + case token_type::begin_array: + case token_type::begin_object: + case token_type::end_array: + case token_type::end_object: + case token_type::name_separator: + case token_type::value_separator: + case token_type::parse_error: + case token_type::end_of_input: + case token_type::literal_or_value: + default: + return sax->parse_error(chars_read, number_string, parse_error::create(115, chars_read, exception_message(input_format_t::ubjson, "invalid number text: " + number_lexer.get_token_string(), "high-precision number"), BasicJsonType())); + } + } + + /////////////////////// + // Utility functions // + /////////////////////// + + /*! + @brief get next character from the input + + This function provides the interface to the used input adapter. It does + not throw in case the input reached EOF, but returns a -'ve valued + `std::char_traits::eof()` in that case. + + @return character read from the input + */ + char_int_type get() + { + ++chars_read; + return current = ia.get_character(); + } + + /*! + @return character read from the input after ignoring all 'N' entries + */ + char_int_type get_ignore_noop() + { + do + { + get(); + } + while (current == 'N'); + + return current; + } + + /* + @brief read a number from the input + + @tparam NumberType the type of the number + @param[in] format the current format (for diagnostics) + @param[out] result number of type @a NumberType + + @return whether conversion completed + + @note This function needs to respect the system's endianess, because + bytes in CBOR, MessagePack, and UBJSON are stored in network order + (big endian) and therefore need reordering on little endian systems. + */ + template + bool get_number(const input_format_t format, NumberType& result) + { + // step 1: read input into array with system's byte order + std::array vec{}; + for (std::size_t i = 0; i < sizeof(NumberType); ++i) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(format, "number"))) + { + return false; + } + + // reverse byte order prior to conversion if necessary + if (is_little_endian != InputIsLittleEndian) + { + vec[sizeof(NumberType) - i - 1] = static_cast(current); + } + else + { + vec[i] = static_cast(current); // LCOV_EXCL_LINE + } + } + + // step 2: convert array into number of type T and return + std::memcpy(&result, vec.data(), sizeof(NumberType)); + return true; + } + + /*! + @brief create a string by reading characters from the input + + @tparam NumberType the type of the number + @param[in] format the current format (for diagnostics) + @param[in] len number of characters to read + @param[out] result string created by reading @a len bytes + + @return whether string creation completed + + @note We can not reserve @a len bytes for the result, because @a len + may be too large. Usually, @ref unexpect_eof() detects the end of + the input before we run out of string memory. + */ + template + bool get_string(const input_format_t format, + const NumberType len, + string_t& result) + { + bool success = true; + for (NumberType i = 0; i < len; i++) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(format, "string"))) + { + success = false; + break; + } + result.push_back(static_cast(current)); + } + return success; + } + + /*! + @brief create a byte array by reading bytes from the input + + @tparam NumberType the type of the number + @param[in] format the current format (for diagnostics) + @param[in] len number of bytes to read + @param[out] result byte array created by reading @a len bytes + + @return whether byte array creation completed + + @note We can not reserve @a len bytes for the result, because @a len + may be too large. Usually, @ref unexpect_eof() detects the end of + the input before we run out of memory. + */ + template + bool get_binary(const input_format_t format, + const NumberType len, + binary_t& result) + { + bool success = true; + for (NumberType i = 0; i < len; i++) + { + get(); + if (JSON_HEDLEY_UNLIKELY(!unexpect_eof(format, "binary"))) + { + success = false; + break; + } + result.push_back(static_cast(current)); + } + return success; + } + + /*! + @param[in] format the current format (for diagnostics) + @param[in] context further context information (for diagnostics) + @return whether the last read character is not EOF + */ + JSON_HEDLEY_NON_NULL(3) + bool unexpect_eof(const input_format_t format, const char* context) const + { + if (JSON_HEDLEY_UNLIKELY(current == std::char_traits::eof())) + { + return sax->parse_error(chars_read, "", + parse_error::create(110, chars_read, exception_message(format, "unexpected end of input", context), BasicJsonType())); + } + return true; + } + + /*! + @return a string representation of the last read byte + */ + std::string get_token_string() const + { + std::array cr{{}}; + (std::snprintf)(cr.data(), cr.size(), "%.2hhX", static_cast(current)); // NOLINT(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + return std::string{cr.data()}; + } + + /*! + @param[in] format the current format + @param[in] detail a detailed error message + @param[in] context further context information + @return a message string to use in the parse_error exceptions + */ + std::string exception_message(const input_format_t format, + const std::string& detail, + const std::string& context) const + { + std::string error_msg = "syntax error while parsing "; + + switch (format) + { + case input_format_t::cbor: + error_msg += "CBOR"; + break; + + case input_format_t::msgpack: + error_msg += "MessagePack"; + break; + + case input_format_t::ubjson: + error_msg += "UBJSON"; + break; + + case input_format_t::bson: + error_msg += "BSON"; + break; + + case input_format_t::json: // LCOV_EXCL_LINE + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + + return error_msg + " " + context + ": " + detail; + } + + private: + /// input adapter + InputAdapterType ia; + + /// the current character + char_int_type current = std::char_traits::eof(); + + /// the number of characters read + std::size_t chars_read = 0; + + /// whether we can assume little endianess + const bool is_little_endian = little_endianess(); + + /// the SAX parser + json_sax_t* sax = nullptr; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + +// #include + + +#include // isfinite +#include // uint8_t +#include // function +#include // string +#include // move +#include // vector + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +//////////// +// parser // +//////////// + +enum class parse_event_t : std::uint8_t +{ + /// the parser read `{` and started to process a JSON object + object_start, + /// the parser read `}` and finished processing a JSON object + object_end, + /// the parser read `[` and started to process a JSON array + array_start, + /// the parser read `]` and finished processing a JSON array + array_end, + /// the parser read a key of a value in an object + key, + /// the parser finished reading a JSON value + value +}; + +template +using parser_callback_t = + std::function; + +/*! +@brief syntax analysis + +This class implements a recursive descent parser. +*/ +template +class parser +{ + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using number_float_t = typename BasicJsonType::number_float_t; + using string_t = typename BasicJsonType::string_t; + using lexer_t = lexer; + using token_type = typename lexer_t::token_type; + + public: + /// a parser reading from an input adapter + explicit parser(InputAdapterType&& adapter, + const parser_callback_t cb = nullptr, + const bool allow_exceptions_ = true, + const bool skip_comments = false) + : callback(cb) + , m_lexer(std::move(adapter), skip_comments) + , allow_exceptions(allow_exceptions_) + { + // read first token + get_token(); + } + + /*! + @brief public parser interface + + @param[in] strict whether to expect the last token to be EOF + @param[in,out] result parsed JSON value + + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ + void parse(const bool strict, BasicJsonType& result) + { + if (callback) + { + json_sax_dom_callback_parser sdp(result, callback, allow_exceptions); + sax_parse_internal(&sdp); + + // in strict mode, input must be completely read + if (strict && (get_token() != token_type::end_of_input)) + { + sdp.parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), + exception_message(token_type::end_of_input, "value"), BasicJsonType())); + } + + // in case of an error, return discarded value + if (sdp.is_errored()) + { + result = value_t::discarded; + return; + } + + // set top-level value to null if it was discarded by the callback + // function + if (result.is_discarded()) + { + result = nullptr; + } + } + else + { + json_sax_dom_parser sdp(result, allow_exceptions); + sax_parse_internal(&sdp); + + // in strict mode, input must be completely read + if (strict && (get_token() != token_type::end_of_input)) + { + sdp.parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::end_of_input, "value"), BasicJsonType())); + } + + // in case of an error, return discarded value + if (sdp.is_errored()) + { + result = value_t::discarded; + return; + } + } + + result.assert_invariant(); + } + + /*! + @brief public accept interface + + @param[in] strict whether to expect the last token to be EOF + @return whether the input is a proper JSON text + */ + bool accept(const bool strict = true) + { + json_sax_acceptor sax_acceptor; + return sax_parse(&sax_acceptor, strict); + } + + template + JSON_HEDLEY_NON_NULL(2) + bool sax_parse(SAX* sax, const bool strict = true) + { + (void)detail::is_sax_static_asserts {}; + const bool result = sax_parse_internal(sax); + + // strict mode: next byte must be EOF + if (result && strict && (get_token() != token_type::end_of_input)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::end_of_input, "value"), BasicJsonType())); + } + + return result; + } + + private: + template + JSON_HEDLEY_NON_NULL(2) + bool sax_parse_internal(SAX* sax) + { + // stack to remember the hierarchy of structured values we are parsing + // true = array; false = object + std::vector states; + // value to avoid a goto (see comment where set to true) + bool skip_to_state_evaluation = false; + + while (true) + { + if (!skip_to_state_evaluation) + { + // invariant: get_token() was called before each iteration + switch (last_token) + { + case token_type::begin_object: + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_object(std::size_t(-1)))) + { + return false; + } + + // closing } -> we are done + if (get_token() == token_type::end_object) + { + if (JSON_HEDLEY_UNLIKELY(!sax->end_object())) + { + return false; + } + break; + } + + // parse key + if (JSON_HEDLEY_UNLIKELY(last_token != token_type::value_string)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::value_string, "object key"), BasicJsonType())); + } + if (JSON_HEDLEY_UNLIKELY(!sax->key(m_lexer.get_string()))) + { + return false; + } + + // parse separator (:) + if (JSON_HEDLEY_UNLIKELY(get_token() != token_type::name_separator)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::name_separator, "object separator"), BasicJsonType())); + } + + // remember we are now inside an object + states.push_back(false); + + // parse values + get_token(); + continue; + } + + case token_type::begin_array: + { + if (JSON_HEDLEY_UNLIKELY(!sax->start_array(std::size_t(-1)))) + { + return false; + } + + // closing ] -> we are done + if (get_token() == token_type::end_array) + { + if (JSON_HEDLEY_UNLIKELY(!sax->end_array())) + { + return false; + } + break; + } + + // remember we are now inside an array + states.push_back(true); + + // parse values (no need to call get_token) + continue; + } + + case token_type::value_float: + { + const auto res = m_lexer.get_number_float(); + + if (JSON_HEDLEY_UNLIKELY(!std::isfinite(res))) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + out_of_range::create(406, "number overflow parsing '" + m_lexer.get_token_string() + "'", BasicJsonType())); + } + + if (JSON_HEDLEY_UNLIKELY(!sax->number_float(res, m_lexer.get_string()))) + { + return false; + } + + break; + } + + case token_type::literal_false: + { + if (JSON_HEDLEY_UNLIKELY(!sax->boolean(false))) + { + return false; + } + break; + } + + case token_type::literal_null: + { + if (JSON_HEDLEY_UNLIKELY(!sax->null())) + { + return false; + } + break; + } + + case token_type::literal_true: + { + if (JSON_HEDLEY_UNLIKELY(!sax->boolean(true))) + { + return false; + } + break; + } + + case token_type::value_integer: + { + if (JSON_HEDLEY_UNLIKELY(!sax->number_integer(m_lexer.get_number_integer()))) + { + return false; + } + break; + } + + case token_type::value_string: + { + if (JSON_HEDLEY_UNLIKELY(!sax->string(m_lexer.get_string()))) + { + return false; + } + break; + } + + case token_type::value_unsigned: + { + if (JSON_HEDLEY_UNLIKELY(!sax->number_unsigned(m_lexer.get_number_unsigned()))) + { + return false; + } + break; + } + + case token_type::parse_error: + { + // using "uninitialized" to avoid "expected" message + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::uninitialized, "value"), BasicJsonType())); + } + + case token_type::uninitialized: + case token_type::end_array: + case token_type::end_object: + case token_type::name_separator: + case token_type::value_separator: + case token_type::end_of_input: + case token_type::literal_or_value: + default: // the last token was unexpected + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::literal_or_value, "value"), BasicJsonType())); + } + } + } + else + { + skip_to_state_evaluation = false; + } + + // we reached this line after we successfully parsed a value + if (states.empty()) + { + // empty stack: we reached the end of the hierarchy: done + return true; + } + + if (states.back()) // array + { + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse a new value + get_token(); + continue; + } + + // closing ] + if (JSON_HEDLEY_LIKELY(last_token == token_type::end_array)) + { + if (JSON_HEDLEY_UNLIKELY(!sax->end_array())) + { + return false; + } + + // We are done with this array. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + JSON_ASSERT(!states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::end_array, "array"), BasicJsonType())); + } + + // states.back() is false -> object + + // comma -> next value + if (get_token() == token_type::value_separator) + { + // parse key + if (JSON_HEDLEY_UNLIKELY(get_token() != token_type::value_string)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::value_string, "object key"), BasicJsonType())); + } + + if (JSON_HEDLEY_UNLIKELY(!sax->key(m_lexer.get_string()))) + { + return false; + } + + // parse separator (:) + if (JSON_HEDLEY_UNLIKELY(get_token() != token_type::name_separator)) + { + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::name_separator, "object separator"), BasicJsonType())); + } + + // parse values + get_token(); + continue; + } + + // closing } + if (JSON_HEDLEY_LIKELY(last_token == token_type::end_object)) + { + if (JSON_HEDLEY_UNLIKELY(!sax->end_object())) + { + return false; + } + + // We are done with this object. Before we can parse a + // new value, we need to evaluate the new state first. + // By setting skip_to_state_evaluation to false, we + // are effectively jumping to the beginning of this if. + JSON_ASSERT(!states.empty()); + states.pop_back(); + skip_to_state_evaluation = true; + continue; + } + + return sax->parse_error(m_lexer.get_position(), + m_lexer.get_token_string(), + parse_error::create(101, m_lexer.get_position(), exception_message(token_type::end_object, "object"), BasicJsonType())); + } + } + + /// get next token from lexer + token_type get_token() + { + return last_token = m_lexer.scan(); + } + + std::string exception_message(const token_type expected, const std::string& context) + { + std::string error_msg = "syntax error "; + + if (!context.empty()) + { + error_msg += "while parsing " + context + " "; + } + + error_msg += "- "; + + if (last_token == token_type::parse_error) + { + error_msg += std::string(m_lexer.get_error_message()) + "; last read: '" + + m_lexer.get_token_string() + "'"; + } + else + { + error_msg += "unexpected " + std::string(lexer_t::token_type_name(last_token)); + } + + if (expected != token_type::uninitialized) + { + error_msg += "; expected " + std::string(lexer_t::token_type_name(expected)); + } + + return error_msg; + } + + private: + /// callback function + const parser_callback_t callback = nullptr; + /// the type of the last read token + token_type last_token = token_type::uninitialized; + /// the lexer + lexer_t m_lexer; + /// whether to throw exceptions in case of errors + const bool allow_exceptions = true; +}; + +} // namespace detail +} // namespace nlohmann + +// #include + + +// #include + + +#include // ptrdiff_t +#include // numeric_limits + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/* +@brief an iterator for primitive JSON types + +This class models an iterator for primitive JSON types (boolean, number, +string). It's only purpose is to allow the iterator/const_iterator classes +to "iterate" over primitive values. Internally, the iterator is modeled by +a `difference_type` variable. Value begin_value (`0`) models the begin, +end_value (`1`) models past the end. +*/ +class primitive_iterator_t +{ + private: + using difference_type = std::ptrdiff_t; + static constexpr difference_type begin_value = 0; + static constexpr difference_type end_value = begin_value + 1; + + JSON_PRIVATE_UNLESS_TESTED: + /// iterator as signed integer type + difference_type m_it = (std::numeric_limits::min)(); + + public: + constexpr difference_type get_value() const noexcept + { + return m_it; + } + + /// set iterator to a defined beginning + void set_begin() noexcept + { + m_it = begin_value; + } + + /// set iterator to a defined past the end + void set_end() noexcept + { + m_it = end_value; + } + + /// return whether the iterator can be dereferenced + constexpr bool is_begin() const noexcept + { + return m_it == begin_value; + } + + /// return whether the iterator is at end + constexpr bool is_end() const noexcept + { + return m_it == end_value; + } + + friend constexpr bool operator==(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it == rhs.m_it; + } + + friend constexpr bool operator<(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it < rhs.m_it; + } + + primitive_iterator_t operator+(difference_type n) noexcept + { + auto result = *this; + result += n; + return result; + } + + friend constexpr difference_type operator-(primitive_iterator_t lhs, primitive_iterator_t rhs) noexcept + { + return lhs.m_it - rhs.m_it; + } + + primitive_iterator_t& operator++() noexcept + { + ++m_it; + return *this; + } + + primitive_iterator_t const operator++(int) noexcept // NOLINT(readability-const-return-type) + { + auto result = *this; + ++m_it; + return result; + } + + primitive_iterator_t& operator--() noexcept + { + --m_it; + return *this; + } + + primitive_iterator_t const operator--(int) noexcept // NOLINT(readability-const-return-type) + { + auto result = *this; + --m_it; + return result; + } + + primitive_iterator_t& operator+=(difference_type n) noexcept + { + m_it += n; + return *this; + } + + primitive_iterator_t& operator-=(difference_type n) noexcept + { + m_it -= n; + return *this; + } +}; +} // namespace detail +} // namespace nlohmann + + +namespace nlohmann +{ +namespace detail +{ +/*! +@brief an iterator value + +@note This structure could easily be a union, but MSVC currently does not allow +unions members with complex constructors, see https://github.com/nlohmann/json/pull/105. +*/ +template struct internal_iterator +{ + /// iterator for JSON objects + typename BasicJsonType::object_t::iterator object_iterator {}; + /// iterator for JSON arrays + typename BasicJsonType::array_t::iterator array_iterator {}; + /// generic iterator for all other types + primitive_iterator_t primitive_iterator {}; +}; +} // namespace detail +} // namespace nlohmann + +// #include + + +#include // iterator, random_access_iterator_tag, bidirectional_iterator_tag, advance, next +#include // conditional, is_const, remove_const + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +// forward declare, to be able to friend it later on +template class iteration_proxy; +template class iteration_proxy_value; + +/*! +@brief a template for a bidirectional iterator for the @ref basic_json class +This class implements a both iterators (iterator and const_iterator) for the +@ref basic_json class. +@note An iterator is called *initialized* when a pointer to a JSON value has + been set (e.g., by a constructor or a copy assignment). If the iterator is + default-constructed, it is *uninitialized* and most methods are undefined. + **The library uses assertions to detect calls on uninitialized iterators.** +@requirement The class satisfies the following concept requirements: +- +[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): + The iterator that can be moved can be moved in both directions (i.e. + incremented and decremented). +@since version 1.0.0, simplified in version 2.0.9, change to bidirectional + iterators in version 3.0.0 (see https://github.com/nlohmann/json/issues/593) +*/ +template +class iter_impl +{ + /// the iterator with BasicJsonType of different const-ness + using other_iter_impl = iter_impl::value, typename std::remove_const::type, const BasicJsonType>::type>; + /// allow basic_json to access private members + friend other_iter_impl; + friend BasicJsonType; + friend iteration_proxy; + friend iteration_proxy_value; + + using object_t = typename BasicJsonType::object_t; + using array_t = typename BasicJsonType::array_t; + // make sure BasicJsonType is basic_json or const basic_json + static_assert(is_basic_json::type>::value, + "iter_impl only accepts (const) basic_json"); + + public: + + /// The std::iterator class template (used as a base class to provide typedefs) is deprecated in C++17. + /// The C++ Standard has never required user-defined iterators to derive from std::iterator. + /// A user-defined iterator should provide publicly accessible typedefs named + /// iterator_category, value_type, difference_type, pointer, and reference. + /// Note that value_type is required to be non-const, even for constant iterators. + using iterator_category = std::bidirectional_iterator_tag; + + /// the type of the values when the iterator is dereferenced + using value_type = typename BasicJsonType::value_type; + /// a type to represent differences between iterators + using difference_type = typename BasicJsonType::difference_type; + /// defines a pointer to the type iterated over (value_type) + using pointer = typename std::conditional::value, + typename BasicJsonType::const_pointer, + typename BasicJsonType::pointer>::type; + /// defines a reference to the type iterated over (value_type) + using reference = + typename std::conditional::value, + typename BasicJsonType::const_reference, + typename BasicJsonType::reference>::type; + + iter_impl() = default; + ~iter_impl() = default; + iter_impl(iter_impl&&) noexcept = default; + iter_impl& operator=(iter_impl&&) noexcept = default; + + /*! + @brief constructor for a given JSON instance + @param[in] object pointer to a JSON object for this iterator + @pre object != nullptr + @post The iterator is initialized; i.e. `m_object != nullptr`. + */ + explicit iter_impl(pointer object) noexcept : m_object(object) + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = typename object_t::iterator(); + break; + } + + case value_t::array: + { + m_it.array_iterator = typename array_t::iterator(); + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + m_it.primitive_iterator = primitive_iterator_t(); + break; + } + } + } + + /*! + @note The conventional copy constructor and copy assignment are implicitly + defined. Combined with the following converting constructor and + assignment, they support: (1) copy from iterator to iterator, (2) + copy from const iterator to const iterator, and (3) conversion from + iterator to const iterator. However conversion from const iterator + to iterator is not defined. + */ + + /*! + @brief const copy constructor + @param[in] other const iterator to copy from + @note This copy constructor had to be defined explicitly to circumvent a bug + occurring on msvc v19.0 compiler (VS 2015) debug build. For more + information refer to: https://github.com/nlohmann/json/issues/1608 + */ + iter_impl(const iter_impl& other) noexcept + : m_object(other.m_object), m_it(other.m_it) + {} + + /*! + @brief converting assignment + @param[in] other const iterator to copy from + @return const/non-const iterator + @note It is not checked whether @a other is initialized. + */ + iter_impl& operator=(const iter_impl& other) noexcept + { + if (&other != this) + { + m_object = other.m_object; + m_it = other.m_it; + } + return *this; + } + + /*! + @brief converting constructor + @param[in] other non-const iterator to copy from + @note It is not checked whether @a other is initialized. + */ + iter_impl(const iter_impl::type>& other) noexcept + : m_object(other.m_object), m_it(other.m_it) + {} + + /*! + @brief converting assignment + @param[in] other non-const iterator to copy from + @return const/non-const iterator + @note It is not checked whether @a other is initialized. + */ + iter_impl& operator=(const iter_impl::type>& other) noexcept // NOLINT(cert-oop54-cpp) + { + m_object = other.m_object; + m_it = other.m_it; + return *this; + } + + JSON_PRIVATE_UNLESS_TESTED: + /*! + @brief set the iterator to the first value + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + void set_begin() noexcept + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = m_object->m_value.object->begin(); + break; + } + + case value_t::array: + { + m_it.array_iterator = m_object->m_value.array->begin(); + break; + } + + case value_t::null: + { + // set to end so begin()==end() is true: null is empty + m_it.primitive_iterator.set_end(); + break; + } + + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + m_it.primitive_iterator.set_begin(); + break; + } + } + } + + /*! + @brief set the iterator past the last value + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + void set_end() noexcept + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + m_it.object_iterator = m_object->m_value.object->end(); + break; + } + + case value_t::array: + { + m_it.array_iterator = m_object->m_value.array->end(); + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + m_it.primitive_iterator.set_end(); + break; + } + } + } + + public: + /*! + @brief return a reference to the value pointed to by the iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference operator*() const + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + JSON_ASSERT(m_it.object_iterator != m_object->m_value.object->end()); + return m_it.object_iterator->second; + } + + case value_t::array: + { + JSON_ASSERT(m_it.array_iterator != m_object->m_value.array->end()); + return *m_it.array_iterator; + } + + case value_t::null: + JSON_THROW(invalid_iterator::create(214, "cannot get value", *m_object)); + + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + if (JSON_HEDLEY_LIKELY(m_it.primitive_iterator.is_begin())) + { + return *m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value", *m_object)); + } + } + } + + /*! + @brief dereference the iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + pointer operator->() const + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + JSON_ASSERT(m_it.object_iterator != m_object->m_value.object->end()); + return &(m_it.object_iterator->second); + } + + case value_t::array: + { + JSON_ASSERT(m_it.array_iterator != m_object->m_value.array->end()); + return &*m_it.array_iterator; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + if (JSON_HEDLEY_LIKELY(m_it.primitive_iterator.is_begin())) + { + return m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value", *m_object)); + } + } + } + + /*! + @brief post-increment (it++) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl const operator++(int) // NOLINT(readability-const-return-type) + { + auto result = *this; + ++(*this); + return result; + } + + /*! + @brief pre-increment (++it) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator++() + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + std::advance(m_it.object_iterator, 1); + break; + } + + case value_t::array: + { + std::advance(m_it.array_iterator, 1); + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + ++m_it.primitive_iterator; + break; + } + } + + return *this; + } + + /*! + @brief post-decrement (it--) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl const operator--(int) // NOLINT(readability-const-return-type) + { + auto result = *this; + --(*this); + return result; + } + + /*! + @brief pre-decrement (--it) + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator--() + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + { + std::advance(m_it.object_iterator, -1); + break; + } + + case value_t::array: + { + std::advance(m_it.array_iterator, -1); + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + --m_it.primitive_iterator; + break; + } + } + + return *this; + } + + /*! + @brief comparison: equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + template < typename IterImpl, detail::enable_if_t < (std::is_same::value || std::is_same::value), std::nullptr_t > = nullptr > + bool operator==(const IterImpl& other) const + { + // if objects are not the same, the comparison is undefined + if (JSON_HEDLEY_UNLIKELY(m_object != other.m_object)) + { + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers", *m_object)); + } + + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + return (m_it.object_iterator == other.m_it.object_iterator); + + case value_t::array: + return (m_it.array_iterator == other.m_it.array_iterator); + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + return (m_it.primitive_iterator == other.m_it.primitive_iterator); + } + } + + /*! + @brief comparison: not equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + template < typename IterImpl, detail::enable_if_t < (std::is_same::value || std::is_same::value), std::nullptr_t > = nullptr > + bool operator!=(const IterImpl& other) const + { + return !operator==(other); + } + + /*! + @brief comparison: smaller + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator<(const iter_impl& other) const + { + // if objects are not the same, the comparison is undefined + if (JSON_HEDLEY_UNLIKELY(m_object != other.m_object)) + { + JSON_THROW(invalid_iterator::create(212, "cannot compare iterators of different containers", *m_object)); + } + + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(213, "cannot compare order of object iterators", *m_object)); + + case value_t::array: + return (m_it.array_iterator < other.m_it.array_iterator); + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + return (m_it.primitive_iterator < other.m_it.primitive_iterator); + } + } + + /*! + @brief comparison: less than or equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator<=(const iter_impl& other) const + { + return !other.operator < (*this); + } + + /*! + @brief comparison: greater than + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator>(const iter_impl& other) const + { + return !operator<=(other); + } + + /*! + @brief comparison: greater than or equal + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + bool operator>=(const iter_impl& other) const + { + return !operator<(other); + } + + /*! + @brief add to iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator+=(difference_type i) + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators", *m_object)); + + case value_t::array: + { + std::advance(m_it.array_iterator, i); + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + m_it.primitive_iterator += i; + break; + } + } + + return *this; + } + + /*! + @brief subtract from iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl& operator-=(difference_type i) + { + return operator+=(-i); + } + + /*! + @brief add to iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl operator+(difference_type i) const + { + auto result = *this; + result += i; + return result; + } + + /*! + @brief addition of distance and iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + friend iter_impl operator+(difference_type i, const iter_impl& it) + { + auto result = it; + result += i; + return result; + } + + /*! + @brief subtract from iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + iter_impl operator-(difference_type i) const + { + auto result = *this; + result -= i; + return result; + } + + /*! + @brief return difference + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + difference_type operator-(const iter_impl& other) const + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(209, "cannot use offsets with object iterators", *m_object)); + + case value_t::array: + return m_it.array_iterator - other.m_it.array_iterator; + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + return m_it.primitive_iterator - other.m_it.primitive_iterator; + } + } + + /*! + @brief access to successor + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference operator[](difference_type n) const + { + JSON_ASSERT(m_object != nullptr); + + switch (m_object->m_type) + { + case value_t::object: + JSON_THROW(invalid_iterator::create(208, "cannot use operator[] for object iterators", *m_object)); + + case value_t::array: + return *std::next(m_it.array_iterator, n); + + case value_t::null: + JSON_THROW(invalid_iterator::create(214, "cannot get value", *m_object)); + + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + if (JSON_HEDLEY_LIKELY(m_it.primitive_iterator.get_value() == -n)) + { + return *m_object; + } + + JSON_THROW(invalid_iterator::create(214, "cannot get value", *m_object)); + } + } + } + + /*! + @brief return the key of an object iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + const typename object_t::key_type& key() const + { + JSON_ASSERT(m_object != nullptr); + + if (JSON_HEDLEY_LIKELY(m_object->is_object())) + { + return m_it.object_iterator->first; + } + + JSON_THROW(invalid_iterator::create(207, "cannot use key() for non-object iterators", *m_object)); + } + + /*! + @brief return the value of an iterator + @pre The iterator is initialized; i.e. `m_object != nullptr`. + */ + reference value() const + { + return operator*(); + } + + JSON_PRIVATE_UNLESS_TESTED: + /// associated JSON instance + pointer m_object = nullptr; + /// the actual iterator of the associated instance + internal_iterator::type> m_it {}; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // ptrdiff_t +#include // reverse_iterator +#include // declval + +namespace nlohmann +{ +namespace detail +{ +////////////////////// +// reverse_iterator // +////////////////////// + +/*! +@brief a template for a reverse iterator class + +@tparam Base the base iterator type to reverse. Valid types are @ref +iterator (to create @ref reverse_iterator) and @ref const_iterator (to +create @ref const_reverse_iterator). + +@requirement The class satisfies the following concept requirements: +- +[BidirectionalIterator](https://en.cppreference.com/w/cpp/named_req/BidirectionalIterator): + The iterator that can be moved can be moved in both directions (i.e. + incremented and decremented). +- [OutputIterator](https://en.cppreference.com/w/cpp/named_req/OutputIterator): + It is possible to write to the pointed-to element (only if @a Base is + @ref iterator). + +@since version 1.0.0 +*/ +template +class json_reverse_iterator : public std::reverse_iterator +{ + public: + using difference_type = std::ptrdiff_t; + /// shortcut to the reverse iterator adapter + using base_iterator = std::reverse_iterator; + /// the reference type for the pointed-to element + using reference = typename Base::reference; + + /// create reverse iterator from iterator + explicit json_reverse_iterator(const typename base_iterator::iterator_type& it) noexcept + : base_iterator(it) {} + + /// create reverse iterator from base class + explicit json_reverse_iterator(const base_iterator& it) noexcept : base_iterator(it) {} + + /// post-increment (it++) + json_reverse_iterator const operator++(int) // NOLINT(readability-const-return-type) + { + return static_cast(base_iterator::operator++(1)); + } + + /// pre-increment (++it) + json_reverse_iterator& operator++() + { + return static_cast(base_iterator::operator++()); + } + + /// post-decrement (it--) + json_reverse_iterator const operator--(int) // NOLINT(readability-const-return-type) + { + return static_cast(base_iterator::operator--(1)); + } + + /// pre-decrement (--it) + json_reverse_iterator& operator--() + { + return static_cast(base_iterator::operator--()); + } + + /// add to iterator + json_reverse_iterator& operator+=(difference_type i) + { + return static_cast(base_iterator::operator+=(i)); + } + + /// add to iterator + json_reverse_iterator operator+(difference_type i) const + { + return static_cast(base_iterator::operator+(i)); + } + + /// subtract from iterator + json_reverse_iterator operator-(difference_type i) const + { + return static_cast(base_iterator::operator-(i)); + } + + /// return difference + difference_type operator-(const json_reverse_iterator& other) const + { + return base_iterator(*this) - base_iterator(other); + } + + /// access to successor + reference operator[](difference_type n) const + { + return *(this->operator+(n)); + } + + /// return the key of an object iterator + auto key() const -> decltype(std::declval().key()) + { + auto it = --this->base(); + return it.key(); + } + + /// return the value of an iterator + reference value() const + { + auto it = --this->base(); + return it.operator * (); + } +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // all_of +#include // isdigit +#include // max +#include // accumulate +#include // string +#include // move +#include // vector + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +template +class json_pointer +{ + // allow basic_json to access private members + NLOHMANN_BASIC_JSON_TPL_DECLARATION + friend class basic_json; + + public: + /*! + @brief create JSON pointer + + Create a JSON pointer according to the syntax described in + [Section 3 of RFC6901](https://tools.ietf.org/html/rfc6901#section-3). + + @param[in] s string representing the JSON pointer; if omitted, the empty + string is assumed which references the whole JSON value + + @throw parse_error.107 if the given JSON pointer @a s is nonempty and does + not begin with a slash (`/`); see example below + + @throw parse_error.108 if a tilde (`~`) in the given JSON pointer @a s is + not followed by `0` (representing `~`) or `1` (representing `/`); see + example below + + @liveexample{The example shows the construction several valid JSON pointers + as well as the exceptional behavior.,json_pointer} + + @since version 2.0.0 + */ + explicit json_pointer(const std::string& s = "") + : reference_tokens(split(s)) + {} + + /*! + @brief return a string representation of the JSON pointer + + @invariant For each JSON pointer `ptr`, it holds: + @code {.cpp} + ptr == json_pointer(ptr.to_string()); + @endcode + + @return a string representation of the JSON pointer + + @liveexample{The example shows the result of `to_string`.,json_pointer__to_string} + + @since version 2.0.0 + */ + std::string to_string() const + { + return std::accumulate(reference_tokens.begin(), reference_tokens.end(), + std::string{}, + [](const std::string & a, const std::string & b) + { + return a + "/" + detail::escape(b); + }); + } + + /// @copydoc to_string() + operator std::string() const + { + return to_string(); + } + + /*! + @brief append another JSON pointer at the end of this JSON pointer + + @param[in] ptr JSON pointer to append + @return JSON pointer with @a ptr appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Linear in the length of @a ptr. + + @sa see @ref operator/=(std::string) to append a reference token + @sa see @ref operator/=(std::size_t) to append an array index + @sa see @ref operator/(const json_pointer&, const json_pointer&) for a binary operator + + @since version 3.6.0 + */ + json_pointer& operator/=(const json_pointer& ptr) + { + reference_tokens.insert(reference_tokens.end(), + ptr.reference_tokens.begin(), + ptr.reference_tokens.end()); + return *this; + } + + /*! + @brief append an unescaped reference token at the end of this JSON pointer + + @param[in] token reference token to append + @return JSON pointer with @a token appended without escaping @a token + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa see @ref operator/=(const json_pointer&) to append a JSON pointer + @sa see @ref operator/=(std::size_t) to append an array index + @sa see @ref operator/(const json_pointer&, std::size_t) for a binary operator + + @since version 3.6.0 + */ + json_pointer& operator/=(std::string token) + { + push_back(std::move(token)); + return *this; + } + + /*! + @brief append an array index at the end of this JSON pointer + + @param[in] array_idx array index to append + @return JSON pointer with @a array_idx appended + + @liveexample{The example shows the usage of `operator/=`.,json_pointer__operator_add} + + @complexity Amortized constant. + + @sa see @ref operator/=(const json_pointer&) to append a JSON pointer + @sa see @ref operator/=(std::string) to append a reference token + @sa see @ref operator/(const json_pointer&, std::string) for a binary operator + + @since version 3.6.0 + */ + json_pointer& operator/=(std::size_t array_idx) + { + return *this /= std::to_string(array_idx); + } + + /*! + @brief create a new JSON pointer by appending the right JSON pointer at the end of the left JSON pointer + + @param[in] lhs JSON pointer + @param[in] rhs JSON pointer + @return a new JSON pointer with @a rhs appended to @a lhs + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a lhs and @a rhs. + + @sa see @ref operator/=(const json_pointer&) to append a JSON pointer + + @since version 3.6.0 + */ + friend json_pointer operator/(const json_pointer& lhs, + const json_pointer& rhs) + { + return json_pointer(lhs) /= rhs; + } + + /*! + @brief create a new JSON pointer by appending the unescaped token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] token reference token + @return a new JSON pointer with unescaped @a token appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa see @ref operator/=(std::string) to append a reference token + + @since version 3.6.0 + */ + friend json_pointer operator/(const json_pointer& ptr, std::string token) // NOLINT(performance-unnecessary-value-param) + { + return json_pointer(ptr) /= std::move(token); + } + + /*! + @brief create a new JSON pointer by appending the array-index-token at the end of the JSON pointer + + @param[in] ptr JSON pointer + @param[in] array_idx array index + @return a new JSON pointer with @a array_idx appended to @a ptr + + @liveexample{The example shows the usage of `operator/`.,json_pointer__operator_add_binary} + + @complexity Linear in the length of @a ptr. + + @sa see @ref operator/=(std::size_t) to append an array index + + @since version 3.6.0 + */ + friend json_pointer operator/(const json_pointer& ptr, std::size_t array_idx) + { + return json_pointer(ptr) /= array_idx; + } + + /*! + @brief returns the parent of this JSON pointer + + @return parent of this JSON pointer; in case this JSON pointer is the root, + the root itself is returned + + @complexity Linear in the length of the JSON pointer. + + @liveexample{The example shows the result of `parent_pointer` for different + JSON Pointers.,json_pointer__parent_pointer} + + @since version 3.6.0 + */ + json_pointer parent_pointer() const + { + if (empty()) + { + return *this; + } + + json_pointer res = *this; + res.pop_back(); + return res; + } + + /*! + @brief remove last reference token + + @pre not `empty()` + + @liveexample{The example shows the usage of `pop_back`.,json_pointer__pop_back} + + @complexity Constant. + + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 + */ + void pop_back() + { + if (JSON_HEDLEY_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent", BasicJsonType())); + } + + reference_tokens.pop_back(); + } + + /*! + @brief return last reference token + + @pre not `empty()` + @return last reference token + + @liveexample{The example shows the usage of `back`.,json_pointer__back} + + @complexity Constant. + + @throw out_of_range.405 if JSON pointer has no parent + + @since version 3.6.0 + */ + const std::string& back() const + { + if (JSON_HEDLEY_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent", BasicJsonType())); + } + + return reference_tokens.back(); + } + + /*! + @brief append an unescaped token at the end of the reference pointer + + @param[in] token token to add + + @complexity Amortized constant. + + @liveexample{The example shows the result of `push_back` for different + JSON Pointers.,json_pointer__push_back} + + @since version 3.6.0 + */ + void push_back(const std::string& token) + { + reference_tokens.push_back(token); + } + + /// @copydoc push_back(const std::string&) + void push_back(std::string&& token) + { + reference_tokens.push_back(std::move(token)); + } + + /*! + @brief return whether pointer points to the root document + + @return true iff the JSON pointer points to the root document + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example shows the result of `empty` for different JSON + Pointers.,json_pointer__empty} + + @since version 3.6.0 + */ + bool empty() const noexcept + { + return reference_tokens.empty(); + } + + private: + /*! + @param[in] s reference token to be converted into an array index + + @return integer representation of @a s + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index begins not with a digit + @throw out_of_range.404 if string @a s could not be converted to an integer + @throw out_of_range.410 if an array index exceeds size_type + */ + static typename BasicJsonType::size_type array_index(const std::string& s) + { + using size_type = typename BasicJsonType::size_type; + + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_HEDLEY_UNLIKELY(s.size() > 1 && s[0] == '0')) + { + JSON_THROW(detail::parse_error::create(106, 0, "array index '" + s + "' must not begin with '0'", BasicJsonType())); + } + + // error condition (cf. RFC 6901, Sect. 4) + if (JSON_HEDLEY_UNLIKELY(s.size() > 1 && !(s[0] >= '1' && s[0] <= '9'))) + { + JSON_THROW(detail::parse_error::create(109, 0, "array index '" + s + "' is not a number", BasicJsonType())); + } + + std::size_t processed_chars = 0; + unsigned long long res = 0; // NOLINT(runtime/int) + JSON_TRY + { + res = std::stoull(s, &processed_chars); + } + JSON_CATCH(std::out_of_range&) + { + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'", BasicJsonType())); + } + + // check if the string was completely read + if (JSON_HEDLEY_UNLIKELY(processed_chars != s.size())) + { + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + s + "'", BasicJsonType())); + } + + // only triggered on special platforms (like 32bit), see also + // https://github.com/nlohmann/json/pull/2203 + if (res >= static_cast((std::numeric_limits::max)())) // NOLINT(runtime/int) + { + JSON_THROW(detail::out_of_range::create(410, "array index " + s + " exceeds size_type", BasicJsonType())); // LCOV_EXCL_LINE + } + + return static_cast(res); + } + + JSON_PRIVATE_UNLESS_TESTED: + json_pointer top() const + { + if (JSON_HEDLEY_UNLIKELY(empty())) + { + JSON_THROW(detail::out_of_range::create(405, "JSON pointer has no parent", BasicJsonType())); + } + + json_pointer result = *this; + result.reference_tokens = {reference_tokens[0]}; + return result; + } + + private: + /*! + @brief create and return a reference to the pointed to value + + @complexity Linear in the number of reference tokens. + + @throw parse_error.109 if array index is not a number + @throw type_error.313 if value cannot be unflattened + */ + BasicJsonType& get_and_create(BasicJsonType& j) const + { + auto* result = &j; + + // in case no reference tokens exist, return a reference to the JSON value + // j which will be overwritten by a primitive value + for (const auto& reference_token : reference_tokens) + { + switch (result->type()) + { + case detail::value_t::null: + { + if (reference_token == "0") + { + // start a new array if reference token is 0 + result = &result->operator[](0); + } + else + { + // start a new object otherwise + result = &result->operator[](reference_token); + } + break; + } + + case detail::value_t::object: + { + // create an entry in the object + result = &result->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + // create an entry in the array + result = &result->operator[](array_index(reference_token)); + break; + } + + /* + The following code is only reached if there exists a reference + token _and_ the current value is primitive. In this case, we have + an error situation, because primitive values may only occur as + single value; that is, with an empty list of reference tokens. + */ + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + JSON_THROW(detail::type_error::create(313, "invalid value to unflatten", j)); + } + } + + return *result; + } + + /*! + @brief return a reference to the pointed to value + + @note This version does not throw if a value is not present, but tries to + create nested values instead. For instance, calling this function + with pointer `"/this/that"` on a null value is equivalent to calling + `operator[]("this").operator[]("that")` on that value, effectively + changing the null value to an object. + + @param[in] ptr a JSON value + + @return reference to the JSON value pointed to by the JSON pointer + + @complexity Linear in the length of the JSON pointer. + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + BasicJsonType& get_unchecked(BasicJsonType* ptr) const + { + for (const auto& reference_token : reference_tokens) + { + // convert null values to arrays or objects before continuing + if (ptr->is_null()) + { + // check if reference token is a number + const bool nums = + std::all_of(reference_token.begin(), reference_token.end(), + [](const unsigned char x) + { + return std::isdigit(x); + }); + + // change value to array for numbers or "-" or to object otherwise + *ptr = (nums || reference_token == "-") + ? detail::value_t::array + : detail::value_t::object; + } + + switch (ptr->type()) + { + case detail::value_t::object: + { + // use unchecked object access + ptr = &ptr->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + if (reference_token == "-") + { + // explicitly treat "-" as index beyond the end + ptr = &ptr->operator[](ptr->m_value.array->size()); + } + else + { + // convert array index to number; unchecked access + ptr = &ptr->operator[](array_index(reference_token)); + } + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'", *ptr)); + } + } + + return *ptr; + } + + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + BasicJsonType& get_checked(BasicJsonType* ptr) const + { + for (const auto& reference_token : reference_tokens) + { + switch (ptr->type()) + { + case detail::value_t::object: + { + // note: at performs range check + ptr = &ptr->at(reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_HEDLEY_UNLIKELY(reference_token == "-")) + { + // "-" always fails the range check + JSON_THROW(detail::out_of_range::create(402, + "array index '-' (" + std::to_string(ptr->m_value.array->size()) + + ") is out of range", *ptr)); + } + + // note: at performs range check + ptr = &ptr->at(array_index(reference_token)); + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'", *ptr)); + } + } + + return *ptr; + } + + /*! + @brief return a const reference to the pointed to value + + @param[in] ptr a JSON value + + @return const reference to the JSON value pointed to by the JSON + pointer + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + const BasicJsonType& get_unchecked(const BasicJsonType* ptr) const + { + for (const auto& reference_token : reference_tokens) + { + switch (ptr->type()) + { + case detail::value_t::object: + { + // use unchecked object access + ptr = &ptr->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_HEDLEY_UNLIKELY(reference_token == "-")) + { + // "-" cannot be used for const access + JSON_THROW(detail::out_of_range::create(402, "array index '-' (" + std::to_string(ptr->m_value.array->size()) + ") is out of range", *ptr)); + } + + // use unchecked array access + ptr = &ptr->operator[](array_index(reference_token)); + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'", *ptr)); + } + } + + return *ptr; + } + + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + */ + const BasicJsonType& get_checked(const BasicJsonType* ptr) const + { + for (const auto& reference_token : reference_tokens) + { + switch (ptr->type()) + { + case detail::value_t::object: + { + // note: at performs range check + ptr = &ptr->at(reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_HEDLEY_UNLIKELY(reference_token == "-")) + { + // "-" always fails the range check + JSON_THROW(detail::out_of_range::create(402, + "array index '-' (" + std::to_string(ptr->m_value.array->size()) + + ") is out of range", *ptr)); + } + + // note: at performs range check + ptr = &ptr->at(array_index(reference_token)); + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + JSON_THROW(detail::out_of_range::create(404, "unresolved reference token '" + reference_token + "'", *ptr)); + } + } + + return *ptr; + } + + /*! + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + */ + bool contains(const BasicJsonType* ptr) const + { + for (const auto& reference_token : reference_tokens) + { + switch (ptr->type()) + { + case detail::value_t::object: + { + if (!ptr->contains(reference_token)) + { + // we did not find the key in the object + return false; + } + + ptr = &ptr->operator[](reference_token); + break; + } + + case detail::value_t::array: + { + if (JSON_HEDLEY_UNLIKELY(reference_token == "-")) + { + // "-" always fails the range check + return false; + } + if (JSON_HEDLEY_UNLIKELY(reference_token.size() == 1 && !("0" <= reference_token && reference_token <= "9"))) + { + // invalid char + return false; + } + if (JSON_HEDLEY_UNLIKELY(reference_token.size() > 1)) + { + if (JSON_HEDLEY_UNLIKELY(!('1' <= reference_token[0] && reference_token[0] <= '9'))) + { + // first char should be between '1' and '9' + return false; + } + for (std::size_t i = 1; i < reference_token.size(); i++) + { + if (JSON_HEDLEY_UNLIKELY(!('0' <= reference_token[i] && reference_token[i] <= '9'))) + { + // other char should be between '0' and '9' + return false; + } + } + } + + const auto idx = array_index(reference_token); + if (idx >= ptr->size()) + { + // index out of range + return false; + } + + ptr = &ptr->operator[](idx); + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + { + // we do not expect primitive values if there is still a + // reference token to process + return false; + } + } + } + + // no reference token left means we found a primitive value + return true; + } + + /*! + @brief split the string input to reference tokens + + @note This function is only called by the json_pointer constructor. + All exceptions below are documented there. + + @throw parse_error.107 if the pointer is not empty or begins with '/' + @throw parse_error.108 if character '~' is not followed by '0' or '1' + */ + static std::vector split(const std::string& reference_string) + { + std::vector result; + + // special case: empty reference string -> no reference tokens + if (reference_string.empty()) + { + return result; + } + + // check if nonempty reference string begins with slash + if (JSON_HEDLEY_UNLIKELY(reference_string[0] != '/')) + { + JSON_THROW(detail::parse_error::create(107, 1, "JSON pointer must be empty or begin with '/' - was: '" + reference_string + "'", BasicJsonType())); + } + + // extract the reference tokens: + // - slash: position of the last read slash (or end of string) + // - start: position after the previous slash + for ( + // search for the first slash after the first character + std::size_t slash = reference_string.find_first_of('/', 1), + // set the beginning of the first reference token + start = 1; + // we can stop if start == 0 (if slash == std::string::npos) + start != 0; + // set the beginning of the next reference token + // (will eventually be 0 if slash == std::string::npos) + start = (slash == std::string::npos) ? 0 : slash + 1, + // find next slash + slash = reference_string.find_first_of('/', start)) + { + // use the text between the beginning of the reference token + // (start) and the last slash (slash). + auto reference_token = reference_string.substr(start, slash - start); + + // check reference tokens are properly escaped + for (std::size_t pos = reference_token.find_first_of('~'); + pos != std::string::npos; + pos = reference_token.find_first_of('~', pos + 1)) + { + JSON_ASSERT(reference_token[pos] == '~'); + + // ~ must be followed by 0 or 1 + if (JSON_HEDLEY_UNLIKELY(pos == reference_token.size() - 1 || + (reference_token[pos + 1] != '0' && + reference_token[pos + 1] != '1'))) + { + JSON_THROW(detail::parse_error::create(108, 0, "escape character '~' must be followed with '0' or '1'", BasicJsonType())); + } + } + + // finally, store the reference token + detail::unescape(reference_token); + result.push_back(reference_token); + } + + return result; + } + + private: + /*! + @param[in] reference_string the reference string to the current value + @param[in] value the value to consider + @param[in,out] result the result object to insert values to + + @note Empty objects or arrays are flattened to `null`. + */ + static void flatten(const std::string& reference_string, + const BasicJsonType& value, + BasicJsonType& result) + { + switch (value.type()) + { + case detail::value_t::array: + { + if (value.m_value.array->empty()) + { + // flatten empty array as null + result[reference_string] = nullptr; + } + else + { + // iterate array and use index as reference string + for (std::size_t i = 0; i < value.m_value.array->size(); ++i) + { + flatten(reference_string + "/" + std::to_string(i), + value.m_value.array->operator[](i), result); + } + } + break; + } + + case detail::value_t::object: + { + if (value.m_value.object->empty()) + { + // flatten empty object as null + result[reference_string] = nullptr; + } + else + { + // iterate object and use keys as reference string + for (const auto& element : *value.m_value.object) + { + flatten(reference_string + "/" + detail::escape(element.first), element.second, result); + } + } + break; + } + + case detail::value_t::null: + case detail::value_t::string: + case detail::value_t::boolean: + case detail::value_t::number_integer: + case detail::value_t::number_unsigned: + case detail::value_t::number_float: + case detail::value_t::binary: + case detail::value_t::discarded: + default: + { + // add primitive value with its reference string + result[reference_string] = value; + break; + } + } + } + + /*! + @param[in] value flattened JSON + + @return unflattened JSON + + @throw parse_error.109 if array index is not a number + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitive + @throw type_error.313 if value cannot be unflattened + */ + static BasicJsonType + unflatten(const BasicJsonType& value) + { + if (JSON_HEDLEY_UNLIKELY(!value.is_object())) + { + JSON_THROW(detail::type_error::create(314, "only objects can be unflattened", value)); + } + + BasicJsonType result; + + // iterate the JSON object values + for (const auto& element : *value.m_value.object) + { + if (JSON_HEDLEY_UNLIKELY(!element.second.is_primitive())) + { + JSON_THROW(detail::type_error::create(315, "values in object must be primitive", element.second)); + } + + // assign value to reference pointed to by JSON pointer; Note that if + // the JSON pointer is "" (i.e., points to the whole value), function + // get_and_create returns a reference to result itself. An assignment + // will then create a primitive value. + json_pointer(element.first).get_and_create(result) = element.second; + } + + return result; + } + + /*! + @brief compares two JSON pointers for equality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is equal to @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ + friend bool operator==(json_pointer const& lhs, + json_pointer const& rhs) noexcept + { + return lhs.reference_tokens == rhs.reference_tokens; + } + + /*! + @brief compares two JSON pointers for inequality + + @param[in] lhs JSON pointer to compare + @param[in] rhs JSON pointer to compare + @return whether @a lhs is not equal @a rhs + + @complexity Linear in the length of the JSON pointer + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + */ + friend bool operator!=(json_pointer const& lhs, + json_pointer const& rhs) noexcept + { + return !(lhs == rhs); + } + + /// the reference tokens + std::vector reference_tokens; +}; +} // namespace nlohmann + +// #include + + +#include +#include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +template +class json_ref +{ + public: + using value_type = BasicJsonType; + + json_ref(value_type&& value) + : owned_value(std::move(value)) + {} + + json_ref(const value_type& value) + : value_ref(&value) + {} + + json_ref(std::initializer_list init) + : owned_value(init) + {} + + template < + class... Args, + enable_if_t::value, int> = 0 > + json_ref(Args && ... args) + : owned_value(std::forward(args)...) + {} + + // class should be movable only + json_ref(json_ref&&) noexcept = default; + json_ref(const json_ref&) = delete; + json_ref& operator=(const json_ref&) = delete; + json_ref& operator=(json_ref&&) = delete; + ~json_ref() = default; + + value_type moved_or_copied() const + { + if (value_ref == nullptr) + { + return std::move(owned_value); + } + return *value_ref; + } + + value_type const& operator*() const + { + return value_ref ? *value_ref : owned_value; + } + + value_type const* operator->() const + { + return &** this; + } + + private: + mutable value_type owned_value = nullptr; + value_type const* value_ref = nullptr; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + +// #include + +// #include + +// #include + + +#include // reverse +#include // array +#include // isnan, isinf +#include // uint8_t, uint16_t, uint32_t, uint64_t +#include // memcpy +#include // numeric_limits +#include // string +#include // move + +// #include + +// #include + +// #include + + +#include // copy +#include // size_t +#include // back_inserter +#include // shared_ptr, make_shared +#include // basic_string +#include // vector + +#ifndef JSON_NO_IO + #include // streamsize + #include // basic_ostream +#endif // JSON_NO_IO + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/// abstract output adapter interface +template struct output_adapter_protocol +{ + virtual void write_character(CharType c) = 0; + virtual void write_characters(const CharType* s, std::size_t length) = 0; + virtual ~output_adapter_protocol() = default; + + output_adapter_protocol() = default; + output_adapter_protocol(const output_adapter_protocol&) = default; + output_adapter_protocol(output_adapter_protocol&&) noexcept = default; + output_adapter_protocol& operator=(const output_adapter_protocol&) = default; + output_adapter_protocol& operator=(output_adapter_protocol&&) noexcept = default; +}; + +/// a type to simplify interfaces +template +using output_adapter_t = std::shared_ptr>; + +/// output adapter for byte vectors +template> +class output_vector_adapter : public output_adapter_protocol +{ + public: + explicit output_vector_adapter(std::vector& vec) noexcept + : v(vec) + {} + + void write_character(CharType c) override + { + v.push_back(c); + } + + JSON_HEDLEY_NON_NULL(2) + void write_characters(const CharType* s, std::size_t length) override + { + std::copy(s, s + length, std::back_inserter(v)); + } + + private: + std::vector& v; +}; + +#ifndef JSON_NO_IO +/// output adapter for output streams +template +class output_stream_adapter : public output_adapter_protocol +{ + public: + explicit output_stream_adapter(std::basic_ostream& s) noexcept + : stream(s) + {} + + void write_character(CharType c) override + { + stream.put(c); + } + + JSON_HEDLEY_NON_NULL(2) + void write_characters(const CharType* s, std::size_t length) override + { + stream.write(s, static_cast(length)); + } + + private: + std::basic_ostream& stream; +}; +#endif // JSON_NO_IO + +/// output adapter for basic_string +template> +class output_string_adapter : public output_adapter_protocol +{ + public: + explicit output_string_adapter(StringType& s) noexcept + : str(s) + {} + + void write_character(CharType c) override + { + str.push_back(c); + } + + JSON_HEDLEY_NON_NULL(2) + void write_characters(const CharType* s, std::size_t length) override + { + str.append(s, length); + } + + private: + StringType& str; +}; + +template> +class output_adapter +{ + public: + template> + output_adapter(std::vector& vec) + : oa(std::make_shared>(vec)) {} + +#ifndef JSON_NO_IO + output_adapter(std::basic_ostream& s) + : oa(std::make_shared>(s)) {} +#endif // JSON_NO_IO + + output_adapter(StringType& s) + : oa(std::make_shared>(s)) {} + + operator output_adapter_t() + { + return oa; + } + + private: + output_adapter_t oa = nullptr; +}; +} // namespace detail +} // namespace nlohmann + + +namespace nlohmann +{ +namespace detail +{ +/////////////////// +// binary writer // +/////////////////// + +/*! +@brief serialization to CBOR and MessagePack values +*/ +template +class binary_writer +{ + using string_t = typename BasicJsonType::string_t; + using binary_t = typename BasicJsonType::binary_t; + using number_float_t = typename BasicJsonType::number_float_t; + + public: + /*! + @brief create a binary writer + + @param[in] adapter output adapter to write to + */ + explicit binary_writer(output_adapter_t adapter) : oa(std::move(adapter)) + { + JSON_ASSERT(oa); + } + + /*! + @param[in] j JSON value to serialize + @pre j.type() == value_t::object + */ + void write_bson(const BasicJsonType& j) + { + switch (j.type()) + { + case value_t::object: + { + write_bson_object(*j.m_value.object); + break; + } + + case value_t::null: + case value_t::array: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + JSON_THROW(type_error::create(317, "to serialize to BSON, top-level type must be object, but is " + std::string(j.type_name()), j)); + } + } + } + + /*! + @param[in] j JSON value to serialize + */ + void write_cbor(const BasicJsonType& j) + { + switch (j.type()) + { + case value_t::null: + { + oa->write_character(to_char_type(0xF6)); + break; + } + + case value_t::boolean: + { + oa->write_character(j.m_value.boolean + ? to_char_type(0xF5) + : to_char_type(0xF4)); + break; + } + + case value_t::number_integer: + { + if (j.m_value.number_integer >= 0) + { + // CBOR does not differentiate between positive signed + // integers and unsigned integers. Therefore, we used the + // code from the value_t::number_unsigned case here. + if (j.m_value.number_integer <= 0x17) + { + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x18)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x19)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x1A)); + write_number(static_cast(j.m_value.number_integer)); + } + else + { + oa->write_character(to_char_type(0x1B)); + write_number(static_cast(j.m_value.number_integer)); + } + } + else + { + // The conversions below encode the sign in the first + // byte, and the value is converted to a positive number. + const auto positive_number = -1 - j.m_value.number_integer; + if (j.m_value.number_integer >= -24) + { + write_number(static_cast(0x20 + positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x38)); + write_number(static_cast(positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x39)); + write_number(static_cast(positive_number)); + } + else if (positive_number <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x3A)); + write_number(static_cast(positive_number)); + } + else + { + oa->write_character(to_char_type(0x3B)); + write_number(static_cast(positive_number)); + } + } + break; + } + + case value_t::number_unsigned: + { + if (j.m_value.number_unsigned <= 0x17) + { + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x18)); + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x19)); + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x1A)); + write_number(static_cast(j.m_value.number_unsigned)); + } + else + { + oa->write_character(to_char_type(0x1B)); + write_number(static_cast(j.m_value.number_unsigned)); + } + break; + } + + case value_t::number_float: + { + if (std::isnan(j.m_value.number_float)) + { + // NaN is 0xf97e00 in CBOR + oa->write_character(to_char_type(0xF9)); + oa->write_character(to_char_type(0x7E)); + oa->write_character(to_char_type(0x00)); + } + else if (std::isinf(j.m_value.number_float)) + { + // Infinity is 0xf97c00, -Infinity is 0xf9fc00 + oa->write_character(to_char_type(0xf9)); + oa->write_character(j.m_value.number_float > 0 ? to_char_type(0x7C) : to_char_type(0xFC)); + oa->write_character(to_char_type(0x00)); + } + else + { + write_compact_float(j.m_value.number_float, detail::input_format_t::cbor); + } + break; + } + + case value_t::string: + { + // step 1: write control byte and the string length + const auto N = j.m_value.string->size(); + if (N <= 0x17) + { + write_number(static_cast(0x60 + N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x78)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x79)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x7A)); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x7B)); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // step 2: write the string + oa->write_characters( + reinterpret_cast(j.m_value.string->c_str()), + j.m_value.string->size()); + break; + } + + case value_t::array: + { + // step 1: write control byte and the array size + const auto N = j.m_value.array->size(); + if (N <= 0x17) + { + write_number(static_cast(0x80 + N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x98)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x99)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x9A)); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x9B)); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // step 2: write each element + for (const auto& el : *j.m_value.array) + { + write_cbor(el); + } + break; + } + + case value_t::binary: + { + if (j.m_value.binary->has_subtype()) + { + if (j.m_value.binary->subtype() <= (std::numeric_limits::max)()) + { + write_number(static_cast(0xd8)); + write_number(static_cast(j.m_value.binary->subtype())); + } + else if (j.m_value.binary->subtype() <= (std::numeric_limits::max)()) + { + write_number(static_cast(0xd9)); + write_number(static_cast(j.m_value.binary->subtype())); + } + else if (j.m_value.binary->subtype() <= (std::numeric_limits::max)()) + { + write_number(static_cast(0xda)); + write_number(static_cast(j.m_value.binary->subtype())); + } + else if (j.m_value.binary->subtype() <= (std::numeric_limits::max)()) + { + write_number(static_cast(0xdb)); + write_number(static_cast(j.m_value.binary->subtype())); + } + } + + // step 1: write control byte and the binary array size + const auto N = j.m_value.binary->size(); + if (N <= 0x17) + { + write_number(static_cast(0x40 + N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x58)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x59)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x5A)); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0x5B)); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // step 2: write each element + oa->write_characters( + reinterpret_cast(j.m_value.binary->data()), + N); + + break; + } + + case value_t::object: + { + // step 1: write control byte and the object size + const auto N = j.m_value.object->size(); + if (N <= 0x17) + { + write_number(static_cast(0xA0 + N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0xB8)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0xB9)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0xBA)); + write_number(static_cast(N)); + } + // LCOV_EXCL_START + else if (N <= (std::numeric_limits::max)()) + { + oa->write_character(to_char_type(0xBB)); + write_number(static_cast(N)); + } + // LCOV_EXCL_STOP + + // step 2: write each element + for (const auto& el : *j.m_value.object) + { + write_cbor(el.first); + write_cbor(el.second); + } + break; + } + + case value_t::discarded: + default: + break; + } + } + + /*! + @param[in] j JSON value to serialize + */ + void write_msgpack(const BasicJsonType& j) + { + switch (j.type()) + { + case value_t::null: // nil + { + oa->write_character(to_char_type(0xC0)); + break; + } + + case value_t::boolean: // true and false + { + oa->write_character(j.m_value.boolean + ? to_char_type(0xC3) + : to_char_type(0xC2)); + break; + } + + case value_t::number_integer: + { + if (j.m_value.number_integer >= 0) + { + // MessagePack does not differentiate between positive + // signed integers and unsigned integers. Therefore, we used + // the code from the value_t::number_unsigned case here. + if (j.m_value.number_unsigned < 128) + { + // positive fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 8 + oa->write_character(to_char_type(0xCC)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 16 + oa->write_character(to_char_type(0xCD)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 32 + oa->write_character(to_char_type(0xCE)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 64 + oa->write_character(to_char_type(0xCF)); + write_number(static_cast(j.m_value.number_integer)); + } + } + else + { + if (j.m_value.number_integer >= -32) + { + // negative fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() && + j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 8 + oa->write_character(to_char_type(0xD0)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() && + j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 16 + oa->write_character(to_char_type(0xD1)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() && + j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 32 + oa->write_character(to_char_type(0xD2)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_integer >= (std::numeric_limits::min)() && + j.m_value.number_integer <= (std::numeric_limits::max)()) + { + // int 64 + oa->write_character(to_char_type(0xD3)); + write_number(static_cast(j.m_value.number_integer)); + } + } + break; + } + + case value_t::number_unsigned: + { + if (j.m_value.number_unsigned < 128) + { + // positive fixnum + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 8 + oa->write_character(to_char_type(0xCC)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 16 + oa->write_character(to_char_type(0xCD)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 32 + oa->write_character(to_char_type(0xCE)); + write_number(static_cast(j.m_value.number_integer)); + } + else if (j.m_value.number_unsigned <= (std::numeric_limits::max)()) + { + // uint 64 + oa->write_character(to_char_type(0xCF)); + write_number(static_cast(j.m_value.number_integer)); + } + break; + } + + case value_t::number_float: + { + write_compact_float(j.m_value.number_float, detail::input_format_t::msgpack); + break; + } + + case value_t::string: + { + // step 1: write control byte and the string length + const auto N = j.m_value.string->size(); + if (N <= 31) + { + // fixstr + write_number(static_cast(0xA0 | N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // str 8 + oa->write_character(to_char_type(0xD9)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // str 16 + oa->write_character(to_char_type(0xDA)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // str 32 + oa->write_character(to_char_type(0xDB)); + write_number(static_cast(N)); + } + + // step 2: write the string + oa->write_characters( + reinterpret_cast(j.m_value.string->c_str()), + j.m_value.string->size()); + break; + } + + case value_t::array: + { + // step 1: write control byte and the array size + const auto N = j.m_value.array->size(); + if (N <= 15) + { + // fixarray + write_number(static_cast(0x90 | N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // array 16 + oa->write_character(to_char_type(0xDC)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // array 32 + oa->write_character(to_char_type(0xDD)); + write_number(static_cast(N)); + } + + // step 2: write each element + for (const auto& el : *j.m_value.array) + { + write_msgpack(el); + } + break; + } + + case value_t::binary: + { + // step 0: determine if the binary type has a set subtype to + // determine whether or not to use the ext or fixext types + const bool use_ext = j.m_value.binary->has_subtype(); + + // step 1: write control byte and the byte string length + const auto N = j.m_value.binary->size(); + if (N <= (std::numeric_limits::max)()) + { + std::uint8_t output_type{}; + bool fixed = true; + if (use_ext) + { + switch (N) + { + case 1: + output_type = 0xD4; // fixext 1 + break; + case 2: + output_type = 0xD5; // fixext 2 + break; + case 4: + output_type = 0xD6; // fixext 4 + break; + case 8: + output_type = 0xD7; // fixext 8 + break; + case 16: + output_type = 0xD8; // fixext 16 + break; + default: + output_type = 0xC7; // ext 8 + fixed = false; + break; + } + + } + else + { + output_type = 0xC4; // bin 8 + fixed = false; + } + + oa->write_character(to_char_type(output_type)); + if (!fixed) + { + write_number(static_cast(N)); + } + } + else if (N <= (std::numeric_limits::max)()) + { + std::uint8_t output_type = use_ext + ? 0xC8 // ext 16 + : 0xC5; // bin 16 + + oa->write_character(to_char_type(output_type)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + std::uint8_t output_type = use_ext + ? 0xC9 // ext 32 + : 0xC6; // bin 32 + + oa->write_character(to_char_type(output_type)); + write_number(static_cast(N)); + } + + // step 1.5: if this is an ext type, write the subtype + if (use_ext) + { + write_number(static_cast(j.m_value.binary->subtype())); + } + + // step 2: write the byte string + oa->write_characters( + reinterpret_cast(j.m_value.binary->data()), + N); + + break; + } + + case value_t::object: + { + // step 1: write control byte and the object size + const auto N = j.m_value.object->size(); + if (N <= 15) + { + // fixmap + write_number(static_cast(0x80 | (N & 0xF))); + } + else if (N <= (std::numeric_limits::max)()) + { + // map 16 + oa->write_character(to_char_type(0xDE)); + write_number(static_cast(N)); + } + else if (N <= (std::numeric_limits::max)()) + { + // map 32 + oa->write_character(to_char_type(0xDF)); + write_number(static_cast(N)); + } + + // step 2: write each element + for (const auto& el : *j.m_value.object) + { + write_msgpack(el.first); + write_msgpack(el.second); + } + break; + } + + case value_t::discarded: + default: + break; + } + } + + /*! + @param[in] j JSON value to serialize + @param[in] use_count whether to use '#' prefixes (optimized format) + @param[in] use_type whether to use '$' prefixes (optimized format) + @param[in] add_prefix whether prefixes need to be used for this value + */ + void write_ubjson(const BasicJsonType& j, const bool use_count, + const bool use_type, const bool add_prefix = true) + { + switch (j.type()) + { + case value_t::null: + { + if (add_prefix) + { + oa->write_character(to_char_type('Z')); + } + break; + } + + case value_t::boolean: + { + if (add_prefix) + { + oa->write_character(j.m_value.boolean + ? to_char_type('T') + : to_char_type('F')); + } + break; + } + + case value_t::number_integer: + { + write_number_with_ubjson_prefix(j.m_value.number_integer, add_prefix); + break; + } + + case value_t::number_unsigned: + { + write_number_with_ubjson_prefix(j.m_value.number_unsigned, add_prefix); + break; + } + + case value_t::number_float: + { + write_number_with_ubjson_prefix(j.m_value.number_float, add_prefix); + break; + } + + case value_t::string: + { + if (add_prefix) + { + oa->write_character(to_char_type('S')); + } + write_number_with_ubjson_prefix(j.m_value.string->size(), true); + oa->write_characters( + reinterpret_cast(j.m_value.string->c_str()), + j.m_value.string->size()); + break; + } + + case value_t::array: + { + if (add_prefix) + { + oa->write_character(to_char_type('[')); + } + + bool prefix_required = true; + if (use_type && !j.m_value.array->empty()) + { + JSON_ASSERT(use_count); + const CharType first_prefix = ubjson_prefix(j.front()); + const bool same_prefix = std::all_of(j.begin() + 1, j.end(), + [this, first_prefix](const BasicJsonType & v) + { + return ubjson_prefix(v) == first_prefix; + }); + + if (same_prefix) + { + prefix_required = false; + oa->write_character(to_char_type('$')); + oa->write_character(first_prefix); + } + } + + if (use_count) + { + oa->write_character(to_char_type('#')); + write_number_with_ubjson_prefix(j.m_value.array->size(), true); + } + + for (const auto& el : *j.m_value.array) + { + write_ubjson(el, use_count, use_type, prefix_required); + } + + if (!use_count) + { + oa->write_character(to_char_type(']')); + } + + break; + } + + case value_t::binary: + { + if (add_prefix) + { + oa->write_character(to_char_type('[')); + } + + if (use_type && !j.m_value.binary->empty()) + { + JSON_ASSERT(use_count); + oa->write_character(to_char_type('$')); + oa->write_character('U'); + } + + if (use_count) + { + oa->write_character(to_char_type('#')); + write_number_with_ubjson_prefix(j.m_value.binary->size(), true); + } + + if (use_type) + { + oa->write_characters( + reinterpret_cast(j.m_value.binary->data()), + j.m_value.binary->size()); + } + else + { + for (size_t i = 0; i < j.m_value.binary->size(); ++i) + { + oa->write_character(to_char_type('U')); + oa->write_character(j.m_value.binary->data()[i]); + } + } + + if (!use_count) + { + oa->write_character(to_char_type(']')); + } + + break; + } + + case value_t::object: + { + if (add_prefix) + { + oa->write_character(to_char_type('{')); + } + + bool prefix_required = true; + if (use_type && !j.m_value.object->empty()) + { + JSON_ASSERT(use_count); + const CharType first_prefix = ubjson_prefix(j.front()); + const bool same_prefix = std::all_of(j.begin(), j.end(), + [this, first_prefix](const BasicJsonType & v) + { + return ubjson_prefix(v) == first_prefix; + }); + + if (same_prefix) + { + prefix_required = false; + oa->write_character(to_char_type('$')); + oa->write_character(first_prefix); + } + } + + if (use_count) + { + oa->write_character(to_char_type('#')); + write_number_with_ubjson_prefix(j.m_value.object->size(), true); + } + + for (const auto& el : *j.m_value.object) + { + write_number_with_ubjson_prefix(el.first.size(), true); + oa->write_characters( + reinterpret_cast(el.first.c_str()), + el.first.size()); + write_ubjson(el.second, use_count, use_type, prefix_required); + } + + if (!use_count) + { + oa->write_character(to_char_type('}')); + } + + break; + } + + case value_t::discarded: + default: + break; + } + } + + private: + ////////// + // BSON // + ////////// + + /*! + @return The size of a BSON document entry header, including the id marker + and the entry name size (and its null-terminator). + */ + static std::size_t calc_bson_entry_header_size(const string_t& name, const BasicJsonType& j) + { + const auto it = name.find(static_cast(0)); + if (JSON_HEDLEY_UNLIKELY(it != BasicJsonType::string_t::npos)) + { + JSON_THROW(out_of_range::create(409, "BSON key cannot contain code point U+0000 (at byte " + std::to_string(it) + ")", j)); + static_cast(j); + } + + return /*id*/ 1ul + name.size() + /*zero-terminator*/1u; + } + + /*! + @brief Writes the given @a element_type and @a name to the output adapter + */ + void write_bson_entry_header(const string_t& name, + const std::uint8_t element_type) + { + oa->write_character(to_char_type(element_type)); // boolean + oa->write_characters( + reinterpret_cast(name.c_str()), + name.size() + 1u); + } + + /*! + @brief Writes a BSON element with key @a name and boolean value @a value + */ + void write_bson_boolean(const string_t& name, + const bool value) + { + write_bson_entry_header(name, 0x08); + oa->write_character(value ? to_char_type(0x01) : to_char_type(0x00)); + } + + /*! + @brief Writes a BSON element with key @a name and double value @a value + */ + void write_bson_double(const string_t& name, + const double value) + { + write_bson_entry_header(name, 0x01); + write_number(value); + } + + /*! + @return The size of the BSON-encoded string in @a value + */ + static std::size_t calc_bson_string_size(const string_t& value) + { + return sizeof(std::int32_t) + value.size() + 1ul; + } + + /*! + @brief Writes a BSON element with key @a name and string value @a value + */ + void write_bson_string(const string_t& name, + const string_t& value) + { + write_bson_entry_header(name, 0x02); + + write_number(static_cast(value.size() + 1ul)); + oa->write_characters( + reinterpret_cast(value.c_str()), + value.size() + 1); + } + + /*! + @brief Writes a BSON element with key @a name and null value + */ + void write_bson_null(const string_t& name) + { + write_bson_entry_header(name, 0x0A); + } + + /*! + @return The size of the BSON-encoded integer @a value + */ + static std::size_t calc_bson_integer_size(const std::int64_t value) + { + return (std::numeric_limits::min)() <= value && value <= (std::numeric_limits::max)() + ? sizeof(std::int32_t) + : sizeof(std::int64_t); + } + + /*! + @brief Writes a BSON element with key @a name and integer @a value + */ + void write_bson_integer(const string_t& name, + const std::int64_t value) + { + if ((std::numeric_limits::min)() <= value && value <= (std::numeric_limits::max)()) + { + write_bson_entry_header(name, 0x10); // int32 + write_number(static_cast(value)); + } + else + { + write_bson_entry_header(name, 0x12); // int64 + write_number(static_cast(value)); + } + } + + /*! + @return The size of the BSON-encoded unsigned integer in @a j + */ + static constexpr std::size_t calc_bson_unsigned_size(const std::uint64_t value) noexcept + { + return (value <= static_cast((std::numeric_limits::max)())) + ? sizeof(std::int32_t) + : sizeof(std::int64_t); + } + + /*! + @brief Writes a BSON element with key @a name and unsigned @a value + */ + void write_bson_unsigned(const string_t& name, + const BasicJsonType& j) + { + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + write_bson_entry_header(name, 0x10 /* int32 */); + write_number(static_cast(j.m_value.number_unsigned)); + } + else if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + write_bson_entry_header(name, 0x12 /* int64 */); + write_number(static_cast(j.m_value.number_unsigned)); + } + else + { + JSON_THROW(out_of_range::create(407, "integer number " + std::to_string(j.m_value.number_unsigned) + " cannot be represented by BSON as it does not fit int64", j)); + } + } + + /*! + @brief Writes a BSON element with key @a name and object @a value + */ + void write_bson_object_entry(const string_t& name, + const typename BasicJsonType::object_t& value) + { + write_bson_entry_header(name, 0x03); // object + write_bson_object(value); + } + + /*! + @return The size of the BSON-encoded array @a value + */ + static std::size_t calc_bson_array_size(const typename BasicJsonType::array_t& value) + { + std::size_t array_index = 0ul; + + const std::size_t embedded_document_size = std::accumulate(std::begin(value), std::end(value), std::size_t(0), [&array_index](std::size_t result, const typename BasicJsonType::array_t::value_type & el) + { + return result + calc_bson_element_size(std::to_string(array_index++), el); + }); + + return sizeof(std::int32_t) + embedded_document_size + 1ul; + } + + /*! + @return The size of the BSON-encoded binary array @a value + */ + static std::size_t calc_bson_binary_size(const typename BasicJsonType::binary_t& value) + { + return sizeof(std::int32_t) + value.size() + 1ul; + } + + /*! + @brief Writes a BSON element with key @a name and array @a value + */ + void write_bson_array(const string_t& name, + const typename BasicJsonType::array_t& value) + { + write_bson_entry_header(name, 0x04); // array + write_number(static_cast(calc_bson_array_size(value))); + + std::size_t array_index = 0ul; + + for (const auto& el : value) + { + write_bson_element(std::to_string(array_index++), el); + } + + oa->write_character(to_char_type(0x00)); + } + + /*! + @brief Writes a BSON element with key @a name and binary value @a value + */ + void write_bson_binary(const string_t& name, + const binary_t& value) + { + write_bson_entry_header(name, 0x05); + + write_number(static_cast(value.size())); + write_number(value.has_subtype() ? static_cast(value.subtype()) : std::uint8_t(0x00)); + + oa->write_characters(reinterpret_cast(value.data()), value.size()); + } + + /*! + @brief Calculates the size necessary to serialize the JSON value @a j with its @a name + @return The calculated size for the BSON document entry for @a j with the given @a name. + */ + static std::size_t calc_bson_element_size(const string_t& name, + const BasicJsonType& j) + { + const auto header_size = calc_bson_entry_header_size(name, j); + switch (j.type()) + { + case value_t::object: + return header_size + calc_bson_object_size(*j.m_value.object); + + case value_t::array: + return header_size + calc_bson_array_size(*j.m_value.array); + + case value_t::binary: + return header_size + calc_bson_binary_size(*j.m_value.binary); + + case value_t::boolean: + return header_size + 1ul; + + case value_t::number_float: + return header_size + 8ul; + + case value_t::number_integer: + return header_size + calc_bson_integer_size(j.m_value.number_integer); + + case value_t::number_unsigned: + return header_size + calc_bson_unsigned_size(j.m_value.number_unsigned); + + case value_t::string: + return header_size + calc_bson_string_size(*j.m_value.string); + + case value_t::null: + return header_size + 0ul; + + // LCOV_EXCL_START + case value_t::discarded: + default: + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) + return 0ul; + // LCOV_EXCL_STOP + } + } + + /*! + @brief Serializes the JSON value @a j to BSON and associates it with the + key @a name. + @param name The name to associate with the JSON entity @a j within the + current BSON document + */ + void write_bson_element(const string_t& name, + const BasicJsonType& j) + { + switch (j.type()) + { + case value_t::object: + return write_bson_object_entry(name, *j.m_value.object); + + case value_t::array: + return write_bson_array(name, *j.m_value.array); + + case value_t::binary: + return write_bson_binary(name, *j.m_value.binary); + + case value_t::boolean: + return write_bson_boolean(name, j.m_value.boolean); + + case value_t::number_float: + return write_bson_double(name, j.m_value.number_float); + + case value_t::number_integer: + return write_bson_integer(name, j.m_value.number_integer); + + case value_t::number_unsigned: + return write_bson_unsigned(name, j); + + case value_t::string: + return write_bson_string(name, *j.m_value.string); + + case value_t::null: + return write_bson_null(name); + + // LCOV_EXCL_START + case value_t::discarded: + default: + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) + return; + // LCOV_EXCL_STOP + } + } + + /*! + @brief Calculates the size of the BSON serialization of the given + JSON-object @a j. + @param[in] value JSON value to serialize + @pre value.type() == value_t::object + */ + static std::size_t calc_bson_object_size(const typename BasicJsonType::object_t& value) + { + std::size_t document_size = std::accumulate(value.begin(), value.end(), std::size_t(0), + [](size_t result, const typename BasicJsonType::object_t::value_type & el) + { + return result += calc_bson_element_size(el.first, el.second); + }); + + return sizeof(std::int32_t) + document_size + 1ul; + } + + /*! + @param[in] value JSON value to serialize + @pre value.type() == value_t::object + */ + void write_bson_object(const typename BasicJsonType::object_t& value) + { + write_number(static_cast(calc_bson_object_size(value))); + + for (const auto& el : value) + { + write_bson_element(el.first, el.second); + } + + oa->write_character(to_char_type(0x00)); + } + + ////////// + // CBOR // + ////////// + + static constexpr CharType get_cbor_float_prefix(float /*unused*/) + { + return to_char_type(0xFA); // Single-Precision Float + } + + static constexpr CharType get_cbor_float_prefix(double /*unused*/) + { + return to_char_type(0xFB); // Double-Precision Float + } + + ///////////// + // MsgPack // + ///////////// + + static constexpr CharType get_msgpack_float_prefix(float /*unused*/) + { + return to_char_type(0xCA); // float 32 + } + + static constexpr CharType get_msgpack_float_prefix(double /*unused*/) + { + return to_char_type(0xCB); // float 64 + } + + //////////// + // UBJSON // + //////////// + + // UBJSON: write number (floating point) + template::value, int>::type = 0> + void write_number_with_ubjson_prefix(const NumberType n, + const bool add_prefix) + { + if (add_prefix) + { + oa->write_character(get_ubjson_float_prefix(n)); + } + write_number(n); + } + + // UBJSON: write number (unsigned integer) + template::value, int>::type = 0> + void write_number_with_ubjson_prefix(const NumberType n, + const bool add_prefix) + { + if (n <= static_cast((std::numeric_limits::max)())) + { + if (add_prefix) + { + oa->write_character(to_char_type('i')); // int8 + } + write_number(static_cast(n)); + } + else if (n <= (std::numeric_limits::max)()) + { + if (add_prefix) + { + oa->write_character(to_char_type('U')); // uint8 + } + write_number(static_cast(n)); + } + else if (n <= static_cast((std::numeric_limits::max)())) + { + if (add_prefix) + { + oa->write_character(to_char_type('I')); // int16 + } + write_number(static_cast(n)); + } + else if (n <= static_cast((std::numeric_limits::max)())) + { + if (add_prefix) + { + oa->write_character(to_char_type('l')); // int32 + } + write_number(static_cast(n)); + } + else if (n <= static_cast((std::numeric_limits::max)())) + { + if (add_prefix) + { + oa->write_character(to_char_type('L')); // int64 + } + write_number(static_cast(n)); + } + else + { + if (add_prefix) + { + oa->write_character(to_char_type('H')); // high-precision number + } + + const auto number = BasicJsonType(n).dump(); + write_number_with_ubjson_prefix(number.size(), true); + for (std::size_t i = 0; i < number.size(); ++i) + { + oa->write_character(to_char_type(static_cast(number[i]))); + } + } + } + + // UBJSON: write number (signed integer) + template < typename NumberType, typename std::enable_if < + std::is_signed::value&& + !std::is_floating_point::value, int >::type = 0 > + void write_number_with_ubjson_prefix(const NumberType n, + const bool add_prefix) + { + if ((std::numeric_limits::min)() <= n && n <= (std::numeric_limits::max)()) + { + if (add_prefix) + { + oa->write_character(to_char_type('i')); // int8 + } + write_number(static_cast(n)); + } + else if (static_cast((std::numeric_limits::min)()) <= n && n <= static_cast((std::numeric_limits::max)())) + { + if (add_prefix) + { + oa->write_character(to_char_type('U')); // uint8 + } + write_number(static_cast(n)); + } + else if ((std::numeric_limits::min)() <= n && n <= (std::numeric_limits::max)()) + { + if (add_prefix) + { + oa->write_character(to_char_type('I')); // int16 + } + write_number(static_cast(n)); + } + else if ((std::numeric_limits::min)() <= n && n <= (std::numeric_limits::max)()) + { + if (add_prefix) + { + oa->write_character(to_char_type('l')); // int32 + } + write_number(static_cast(n)); + } + else if ((std::numeric_limits::min)() <= n && n <= (std::numeric_limits::max)()) + { + if (add_prefix) + { + oa->write_character(to_char_type('L')); // int64 + } + write_number(static_cast(n)); + } + // LCOV_EXCL_START + else + { + if (add_prefix) + { + oa->write_character(to_char_type('H')); // high-precision number + } + + const auto number = BasicJsonType(n).dump(); + write_number_with_ubjson_prefix(number.size(), true); + for (std::size_t i = 0; i < number.size(); ++i) + { + oa->write_character(to_char_type(static_cast(number[i]))); + } + } + // LCOV_EXCL_STOP + } + + /*! + @brief determine the type prefix of container values + */ + CharType ubjson_prefix(const BasicJsonType& j) const noexcept + { + switch (j.type()) + { + case value_t::null: + return 'Z'; + + case value_t::boolean: + return j.m_value.boolean ? 'T' : 'F'; + + case value_t::number_integer: + { + if ((std::numeric_limits::min)() <= j.m_value.number_integer && j.m_value.number_integer <= (std::numeric_limits::max)()) + { + return 'i'; + } + if ((std::numeric_limits::min)() <= j.m_value.number_integer && j.m_value.number_integer <= (std::numeric_limits::max)()) + { + return 'U'; + } + if ((std::numeric_limits::min)() <= j.m_value.number_integer && j.m_value.number_integer <= (std::numeric_limits::max)()) + { + return 'I'; + } + if ((std::numeric_limits::min)() <= j.m_value.number_integer && j.m_value.number_integer <= (std::numeric_limits::max)()) + { + return 'l'; + } + if ((std::numeric_limits::min)() <= j.m_value.number_integer && j.m_value.number_integer <= (std::numeric_limits::max)()) + { + return 'L'; + } + // anything else is treated as high-precision number + return 'H'; // LCOV_EXCL_LINE + } + + case value_t::number_unsigned: + { + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + return 'i'; + } + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + return 'U'; + } + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + return 'I'; + } + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + return 'l'; + } + if (j.m_value.number_unsigned <= static_cast((std::numeric_limits::max)())) + { + return 'L'; + } + // anything else is treated as high-precision number + return 'H'; // LCOV_EXCL_LINE + } + + case value_t::number_float: + return get_ubjson_float_prefix(j.m_value.number_float); + + case value_t::string: + return 'S'; + + case value_t::array: // fallthrough + case value_t::binary: + return '['; + + case value_t::object: + return '{'; + + case value_t::discarded: + default: // discarded values + return 'N'; + } + } + + static constexpr CharType get_ubjson_float_prefix(float /*unused*/) + { + return 'd'; // float 32 + } + + static constexpr CharType get_ubjson_float_prefix(double /*unused*/) + { + return 'D'; // float 64 + } + + /////////////////////// + // Utility functions // + /////////////////////// + + /* + @brief write a number to output input + @param[in] n number of type @a NumberType + @tparam NumberType the type of the number + @tparam OutputIsLittleEndian Set to true if output data is + required to be little endian + + @note This function needs to respect the system's endianess, because bytes + in CBOR, MessagePack, and UBJSON are stored in network order (big + endian) and therefore need reordering on little endian systems. + */ + template + void write_number(const NumberType n) + { + // step 1: write number to array of length NumberType + std::array vec{}; + std::memcpy(vec.data(), &n, sizeof(NumberType)); + + // step 2: write array to output (with possible reordering) + if (is_little_endian != OutputIsLittleEndian) + { + // reverse byte order prior to conversion if necessary + std::reverse(vec.begin(), vec.end()); + } + + oa->write_characters(vec.data(), sizeof(NumberType)); + } + + void write_compact_float(const number_float_t n, detail::input_format_t format) + { +#ifdef __GNUC__ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wfloat-equal" +#endif + if (static_cast(n) >= static_cast(std::numeric_limits::lowest()) && + static_cast(n) <= static_cast((std::numeric_limits::max)()) && + static_cast(static_cast(n)) == static_cast(n)) + { + oa->write_character(format == detail::input_format_t::cbor + ? get_cbor_float_prefix(static_cast(n)) + : get_msgpack_float_prefix(static_cast(n))); + write_number(static_cast(n)); + } + else + { + oa->write_character(format == detail::input_format_t::cbor + ? get_cbor_float_prefix(n) + : get_msgpack_float_prefix(n)); + write_number(n); + } +#ifdef __GNUC__ +#pragma GCC diagnostic pop +#endif + } + + public: + // The following to_char_type functions are implement the conversion + // between uint8_t and CharType. In case CharType is not unsigned, + // such a conversion is required to allow values greater than 128. + // See for a discussion. + template < typename C = CharType, + enable_if_t < std::is_signed::value && std::is_signed::value > * = nullptr > + static constexpr CharType to_char_type(std::uint8_t x) noexcept + { + return *reinterpret_cast(&x); + } + + template < typename C = CharType, + enable_if_t < std::is_signed::value && std::is_unsigned::value > * = nullptr > + static CharType to_char_type(std::uint8_t x) noexcept + { + static_assert(sizeof(std::uint8_t) == sizeof(CharType), "size of CharType must be equal to std::uint8_t"); + static_assert(std::is_trivial::value, "CharType must be trivial"); + CharType result; + std::memcpy(&result, &x, sizeof(x)); + return result; + } + + template::value>* = nullptr> + static constexpr CharType to_char_type(std::uint8_t x) noexcept + { + return x; + } + + template < typename InputCharType, typename C = CharType, + enable_if_t < + std::is_signed::value && + std::is_signed::value && + std::is_same::type>::value + > * = nullptr > + static constexpr CharType to_char_type(InputCharType x) noexcept + { + return x; + } + + private: + /// whether we can assume little endianess + const bool is_little_endian = little_endianess(); + + /// the output + output_adapter_t oa = nullptr; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + + +#include // reverse, remove, fill, find, none_of +#include // array +#include // localeconv, lconv +#include // labs, isfinite, isnan, signbit +#include // size_t, ptrdiff_t +#include // uint8_t +#include // snprintf +#include // numeric_limits +#include // string, char_traits +#include // is_same +#include // move + +// #include + + +#include // array +#include // signbit, isfinite +#include // intN_t, uintN_t +#include // memcpy, memmove +#include // numeric_limits +#include // conditional + +// #include + + +namespace nlohmann +{ +namespace detail +{ + +/*! +@brief implements the Grisu2 algorithm for binary to decimal floating-point +conversion. + +This implementation is a slightly modified version of the reference +implementation which may be obtained from +http://florian.loitsch.com/publications (bench.tar.gz). + +The code is distributed under the MIT license, Copyright (c) 2009 Florian Loitsch. + +For a detailed description of the algorithm see: + +[1] Loitsch, "Printing Floating-Point Numbers Quickly and Accurately with + Integers", Proceedings of the ACM SIGPLAN 2010 Conference on Programming + Language Design and Implementation, PLDI 2010 +[2] Burger, Dybvig, "Printing Floating-Point Numbers Quickly and Accurately", + Proceedings of the ACM SIGPLAN 1996 Conference on Programming Language + Design and Implementation, PLDI 1996 +*/ +namespace dtoa_impl +{ + +template +Target reinterpret_bits(const Source source) +{ + static_assert(sizeof(Target) == sizeof(Source), "size mismatch"); + + Target target; + std::memcpy(&target, &source, sizeof(Source)); + return target; +} + +struct diyfp // f * 2^e +{ + static constexpr int kPrecision = 64; // = q + + std::uint64_t f = 0; + int e = 0; + + constexpr diyfp(std::uint64_t f_, int e_) noexcept : f(f_), e(e_) {} + + /*! + @brief returns x - y + @pre x.e == y.e and x.f >= y.f + */ + static diyfp sub(const diyfp& x, const diyfp& y) noexcept + { + JSON_ASSERT(x.e == y.e); + JSON_ASSERT(x.f >= y.f); + + return {x.f - y.f, x.e}; + } + + /*! + @brief returns x * y + @note The result is rounded. (Only the upper q bits are returned.) + */ + static diyfp mul(const diyfp& x, const diyfp& y) noexcept + { + static_assert(kPrecision == 64, "internal error"); + + // Computes: + // f = round((x.f * y.f) / 2^q) + // e = x.e + y.e + q + + // Emulate the 64-bit * 64-bit multiplication: + // + // p = u * v + // = (u_lo + 2^32 u_hi) (v_lo + 2^32 v_hi) + // = (u_lo v_lo ) + 2^32 ((u_lo v_hi ) + (u_hi v_lo )) + 2^64 (u_hi v_hi ) + // = (p0 ) + 2^32 ((p1 ) + (p2 )) + 2^64 (p3 ) + // = (p0_lo + 2^32 p0_hi) + 2^32 ((p1_lo + 2^32 p1_hi) + (p2_lo + 2^32 p2_hi)) + 2^64 (p3 ) + // = (p0_lo ) + 2^32 (p0_hi + p1_lo + p2_lo ) + 2^64 (p1_hi + p2_hi + p3) + // = (p0_lo ) + 2^32 (Q ) + 2^64 (H ) + // = (p0_lo ) + 2^32 (Q_lo + 2^32 Q_hi ) + 2^64 (H ) + // + // (Since Q might be larger than 2^32 - 1) + // + // = (p0_lo + 2^32 Q_lo) + 2^64 (Q_hi + H) + // + // (Q_hi + H does not overflow a 64-bit int) + // + // = p_lo + 2^64 p_hi + + const std::uint64_t u_lo = x.f & 0xFFFFFFFFu; + const std::uint64_t u_hi = x.f >> 32u; + const std::uint64_t v_lo = y.f & 0xFFFFFFFFu; + const std::uint64_t v_hi = y.f >> 32u; + + const std::uint64_t p0 = u_lo * v_lo; + const std::uint64_t p1 = u_lo * v_hi; + const std::uint64_t p2 = u_hi * v_lo; + const std::uint64_t p3 = u_hi * v_hi; + + const std::uint64_t p0_hi = p0 >> 32u; + const std::uint64_t p1_lo = p1 & 0xFFFFFFFFu; + const std::uint64_t p1_hi = p1 >> 32u; + const std::uint64_t p2_lo = p2 & 0xFFFFFFFFu; + const std::uint64_t p2_hi = p2 >> 32u; + + std::uint64_t Q = p0_hi + p1_lo + p2_lo; + + // The full product might now be computed as + // + // p_hi = p3 + p2_hi + p1_hi + (Q >> 32) + // p_lo = p0_lo + (Q << 32) + // + // But in this particular case here, the full p_lo is not required. + // Effectively we only need to add the highest bit in p_lo to p_hi (and + // Q_hi + 1 does not overflow). + + Q += std::uint64_t{1} << (64u - 32u - 1u); // round, ties up + + const std::uint64_t h = p3 + p2_hi + p1_hi + (Q >> 32u); + + return {h, x.e + y.e + 64}; + } + + /*! + @brief normalize x such that the significand is >= 2^(q-1) + @pre x.f != 0 + */ + static diyfp normalize(diyfp x) noexcept + { + JSON_ASSERT(x.f != 0); + + while ((x.f >> 63u) == 0) + { + x.f <<= 1u; + x.e--; + } + + return x; + } + + /*! + @brief normalize x such that the result has the exponent E + @pre e >= x.e and the upper e - x.e bits of x.f must be zero. + */ + static diyfp normalize_to(const diyfp& x, const int target_exponent) noexcept + { + const int delta = x.e - target_exponent; + + JSON_ASSERT(delta >= 0); + JSON_ASSERT(((x.f << delta) >> delta) == x.f); + + return {x.f << delta, target_exponent}; + } +}; + +struct boundaries +{ + diyfp w; + diyfp minus; + diyfp plus; +}; + +/*! +Compute the (normalized) diyfp representing the input number 'value' and its +boundaries. + +@pre value must be finite and positive +*/ +template +boundaries compute_boundaries(FloatType value) +{ + JSON_ASSERT(std::isfinite(value)); + JSON_ASSERT(value > 0); + + // Convert the IEEE representation into a diyfp. + // + // If v is denormal: + // value = 0.F * 2^(1 - bias) = ( F) * 2^(1 - bias - (p-1)) + // If v is normalized: + // value = 1.F * 2^(E - bias) = (2^(p-1) + F) * 2^(E - bias - (p-1)) + + static_assert(std::numeric_limits::is_iec559, + "internal error: dtoa_short requires an IEEE-754 floating-point implementation"); + + constexpr int kPrecision = std::numeric_limits::digits; // = p (includes the hidden bit) + constexpr int kBias = std::numeric_limits::max_exponent - 1 + (kPrecision - 1); + constexpr int kMinExp = 1 - kBias; + constexpr std::uint64_t kHiddenBit = std::uint64_t{1} << (kPrecision - 1); // = 2^(p-1) + + using bits_type = typename std::conditional::type; + + const auto bits = static_cast(reinterpret_bits(value)); + const std::uint64_t E = bits >> (kPrecision - 1); + const std::uint64_t F = bits & (kHiddenBit - 1); + + const bool is_denormal = E == 0; + const diyfp v = is_denormal + ? diyfp(F, kMinExp) + : diyfp(F + kHiddenBit, static_cast(E) - kBias); + + // Compute the boundaries m- and m+ of the floating-point value + // v = f * 2^e. + // + // Determine v- and v+, the floating-point predecessor and successor if v, + // respectively. + // + // v- = v - 2^e if f != 2^(p-1) or e == e_min (A) + // = v - 2^(e-1) if f == 2^(p-1) and e > e_min (B) + // + // v+ = v + 2^e + // + // Let m- = (v- + v) / 2 and m+ = (v + v+) / 2. All real numbers _strictly_ + // between m- and m+ round to v, regardless of how the input rounding + // algorithm breaks ties. + // + // ---+-------------+-------------+-------------+-------------+--- (A) + // v- m- v m+ v+ + // + // -----------------+------+------+-------------+-------------+--- (B) + // v- m- v m+ v+ + + const bool lower_boundary_is_closer = F == 0 && E > 1; + const diyfp m_plus = diyfp(2 * v.f + 1, v.e - 1); + const diyfp m_minus = lower_boundary_is_closer + ? diyfp(4 * v.f - 1, v.e - 2) // (B) + : diyfp(2 * v.f - 1, v.e - 1); // (A) + + // Determine the normalized w+ = m+. + const diyfp w_plus = diyfp::normalize(m_plus); + + // Determine w- = m- such that e_(w-) = e_(w+). + const diyfp w_minus = diyfp::normalize_to(m_minus, w_plus.e); + + return {diyfp::normalize(v), w_minus, w_plus}; +} + +// Given normalized diyfp w, Grisu needs to find a (normalized) cached +// power-of-ten c, such that the exponent of the product c * w = f * 2^e lies +// within a certain range [alpha, gamma] (Definition 3.2 from [1]) +// +// alpha <= e = e_c + e_w + q <= gamma +// +// or +// +// f_c * f_w * 2^alpha <= f_c 2^(e_c) * f_w 2^(e_w) * 2^q +// <= f_c * f_w * 2^gamma +// +// Since c and w are normalized, i.e. 2^(q-1) <= f < 2^q, this implies +// +// 2^(q-1) * 2^(q-1) * 2^alpha <= c * w * 2^q < 2^q * 2^q * 2^gamma +// +// or +// +// 2^(q - 2 + alpha) <= c * w < 2^(q + gamma) +// +// The choice of (alpha,gamma) determines the size of the table and the form of +// the digit generation procedure. Using (alpha,gamma)=(-60,-32) works out well +// in practice: +// +// The idea is to cut the number c * w = f * 2^e into two parts, which can be +// processed independently: An integral part p1, and a fractional part p2: +// +// f * 2^e = ( (f div 2^-e) * 2^-e + (f mod 2^-e) ) * 2^e +// = (f div 2^-e) + (f mod 2^-e) * 2^e +// = p1 + p2 * 2^e +// +// The conversion of p1 into decimal form requires a series of divisions and +// modulos by (a power of) 10. These operations are faster for 32-bit than for +// 64-bit integers, so p1 should ideally fit into a 32-bit integer. This can be +// achieved by choosing +// +// -e >= 32 or e <= -32 := gamma +// +// In order to convert the fractional part +// +// p2 * 2^e = p2 / 2^-e = d[-1] / 10^1 + d[-2] / 10^2 + ... +// +// into decimal form, the fraction is repeatedly multiplied by 10 and the digits +// d[-i] are extracted in order: +// +// (10 * p2) div 2^-e = d[-1] +// (10 * p2) mod 2^-e = d[-2] / 10^1 + ... +// +// The multiplication by 10 must not overflow. It is sufficient to choose +// +// 10 * p2 < 16 * p2 = 2^4 * p2 <= 2^64. +// +// Since p2 = f mod 2^-e < 2^-e, +// +// -e <= 60 or e >= -60 := alpha + +constexpr int kAlpha = -60; +constexpr int kGamma = -32; + +struct cached_power // c = f * 2^e ~= 10^k +{ + std::uint64_t f; + int e; + int k; +}; + +/*! +For a normalized diyfp w = f * 2^e, this function returns a (normalized) cached +power-of-ten c = f_c * 2^e_c, such that the exponent of the product w * c +satisfies (Definition 3.2 from [1]) + + alpha <= e_c + e + q <= gamma. +*/ +inline cached_power get_cached_power_for_binary_exponent(int e) +{ + // Now + // + // alpha <= e_c + e + q <= gamma (1) + // ==> f_c * 2^alpha <= c * 2^e * 2^q + // + // and since the c's are normalized, 2^(q-1) <= f_c, + // + // ==> 2^(q - 1 + alpha) <= c * 2^(e + q) + // ==> 2^(alpha - e - 1) <= c + // + // If c were an exact power of ten, i.e. c = 10^k, one may determine k as + // + // k = ceil( log_10( 2^(alpha - e - 1) ) ) + // = ceil( (alpha - e - 1) * log_10(2) ) + // + // From the paper: + // "In theory the result of the procedure could be wrong since c is rounded, + // and the computation itself is approximated [...]. In practice, however, + // this simple function is sufficient." + // + // For IEEE double precision floating-point numbers converted into + // normalized diyfp's w = f * 2^e, with q = 64, + // + // e >= -1022 (min IEEE exponent) + // -52 (p - 1) + // -52 (p - 1, possibly normalize denormal IEEE numbers) + // -11 (normalize the diyfp) + // = -1137 + // + // and + // + // e <= +1023 (max IEEE exponent) + // -52 (p - 1) + // -11 (normalize the diyfp) + // = 960 + // + // This binary exponent range [-1137,960] results in a decimal exponent + // range [-307,324]. One does not need to store a cached power for each + // k in this range. For each such k it suffices to find a cached power + // such that the exponent of the product lies in [alpha,gamma]. + // This implies that the difference of the decimal exponents of adjacent + // table entries must be less than or equal to + // + // floor( (gamma - alpha) * log_10(2) ) = 8. + // + // (A smaller distance gamma-alpha would require a larger table.) + + // NB: + // Actually this function returns c, such that -60 <= e_c + e + 64 <= -34. + + constexpr int kCachedPowersMinDecExp = -300; + constexpr int kCachedPowersDecStep = 8; + + static constexpr std::array kCachedPowers = + { + { + { 0xAB70FE17C79AC6CA, -1060, -300 }, + { 0xFF77B1FCBEBCDC4F, -1034, -292 }, + { 0xBE5691EF416BD60C, -1007, -284 }, + { 0x8DD01FAD907FFC3C, -980, -276 }, + { 0xD3515C2831559A83, -954, -268 }, + { 0x9D71AC8FADA6C9B5, -927, -260 }, + { 0xEA9C227723EE8BCB, -901, -252 }, + { 0xAECC49914078536D, -874, -244 }, + { 0x823C12795DB6CE57, -847, -236 }, + { 0xC21094364DFB5637, -821, -228 }, + { 0x9096EA6F3848984F, -794, -220 }, + { 0xD77485CB25823AC7, -768, -212 }, + { 0xA086CFCD97BF97F4, -741, -204 }, + { 0xEF340A98172AACE5, -715, -196 }, + { 0xB23867FB2A35B28E, -688, -188 }, + { 0x84C8D4DFD2C63F3B, -661, -180 }, + { 0xC5DD44271AD3CDBA, -635, -172 }, + { 0x936B9FCEBB25C996, -608, -164 }, + { 0xDBAC6C247D62A584, -582, -156 }, + { 0xA3AB66580D5FDAF6, -555, -148 }, + { 0xF3E2F893DEC3F126, -529, -140 }, + { 0xB5B5ADA8AAFF80B8, -502, -132 }, + { 0x87625F056C7C4A8B, -475, -124 }, + { 0xC9BCFF6034C13053, -449, -116 }, + { 0x964E858C91BA2655, -422, -108 }, + { 0xDFF9772470297EBD, -396, -100 }, + { 0xA6DFBD9FB8E5B88F, -369, -92 }, + { 0xF8A95FCF88747D94, -343, -84 }, + { 0xB94470938FA89BCF, -316, -76 }, + { 0x8A08F0F8BF0F156B, -289, -68 }, + { 0xCDB02555653131B6, -263, -60 }, + { 0x993FE2C6D07B7FAC, -236, -52 }, + { 0xE45C10C42A2B3B06, -210, -44 }, + { 0xAA242499697392D3, -183, -36 }, + { 0xFD87B5F28300CA0E, -157, -28 }, + { 0xBCE5086492111AEB, -130, -20 }, + { 0x8CBCCC096F5088CC, -103, -12 }, + { 0xD1B71758E219652C, -77, -4 }, + { 0x9C40000000000000, -50, 4 }, + { 0xE8D4A51000000000, -24, 12 }, + { 0xAD78EBC5AC620000, 3, 20 }, + { 0x813F3978F8940984, 30, 28 }, + { 0xC097CE7BC90715B3, 56, 36 }, + { 0x8F7E32CE7BEA5C70, 83, 44 }, + { 0xD5D238A4ABE98068, 109, 52 }, + { 0x9F4F2726179A2245, 136, 60 }, + { 0xED63A231D4C4FB27, 162, 68 }, + { 0xB0DE65388CC8ADA8, 189, 76 }, + { 0x83C7088E1AAB65DB, 216, 84 }, + { 0xC45D1DF942711D9A, 242, 92 }, + { 0x924D692CA61BE758, 269, 100 }, + { 0xDA01EE641A708DEA, 295, 108 }, + { 0xA26DA3999AEF774A, 322, 116 }, + { 0xF209787BB47D6B85, 348, 124 }, + { 0xB454E4A179DD1877, 375, 132 }, + { 0x865B86925B9BC5C2, 402, 140 }, + { 0xC83553C5C8965D3D, 428, 148 }, + { 0x952AB45CFA97A0B3, 455, 156 }, + { 0xDE469FBD99A05FE3, 481, 164 }, + { 0xA59BC234DB398C25, 508, 172 }, + { 0xF6C69A72A3989F5C, 534, 180 }, + { 0xB7DCBF5354E9BECE, 561, 188 }, + { 0x88FCF317F22241E2, 588, 196 }, + { 0xCC20CE9BD35C78A5, 614, 204 }, + { 0x98165AF37B2153DF, 641, 212 }, + { 0xE2A0B5DC971F303A, 667, 220 }, + { 0xA8D9D1535CE3B396, 694, 228 }, + { 0xFB9B7CD9A4A7443C, 720, 236 }, + { 0xBB764C4CA7A44410, 747, 244 }, + { 0x8BAB8EEFB6409C1A, 774, 252 }, + { 0xD01FEF10A657842C, 800, 260 }, + { 0x9B10A4E5E9913129, 827, 268 }, + { 0xE7109BFBA19C0C9D, 853, 276 }, + { 0xAC2820D9623BF429, 880, 284 }, + { 0x80444B5E7AA7CF85, 907, 292 }, + { 0xBF21E44003ACDD2D, 933, 300 }, + { 0x8E679C2F5E44FF8F, 960, 308 }, + { 0xD433179D9C8CB841, 986, 316 }, + { 0x9E19DB92B4E31BA9, 1013, 324 }, + } + }; + + // This computation gives exactly the same results for k as + // k = ceil((kAlpha - e - 1) * 0.30102999566398114) + // for |e| <= 1500, but doesn't require floating-point operations. + // NB: log_10(2) ~= 78913 / 2^18 + JSON_ASSERT(e >= -1500); + JSON_ASSERT(e <= 1500); + const int f = kAlpha - e - 1; + const int k = (f * 78913) / (1 << 18) + static_cast(f > 0); + + const int index = (-kCachedPowersMinDecExp + k + (kCachedPowersDecStep - 1)) / kCachedPowersDecStep; + JSON_ASSERT(index >= 0); + JSON_ASSERT(static_cast(index) < kCachedPowers.size()); + + const cached_power cached = kCachedPowers[static_cast(index)]; + JSON_ASSERT(kAlpha <= cached.e + e + 64); + JSON_ASSERT(kGamma >= cached.e + e + 64); + + return cached; +} + +/*! +For n != 0, returns k, such that pow10 := 10^(k-1) <= n < 10^k. +For n == 0, returns 1 and sets pow10 := 1. +*/ +inline int find_largest_pow10(const std::uint32_t n, std::uint32_t& pow10) +{ + // LCOV_EXCL_START + if (n >= 1000000000) + { + pow10 = 1000000000; + return 10; + } + // LCOV_EXCL_STOP + if (n >= 100000000) + { + pow10 = 100000000; + return 9; + } + if (n >= 10000000) + { + pow10 = 10000000; + return 8; + } + if (n >= 1000000) + { + pow10 = 1000000; + return 7; + } + if (n >= 100000) + { + pow10 = 100000; + return 6; + } + if (n >= 10000) + { + pow10 = 10000; + return 5; + } + if (n >= 1000) + { + pow10 = 1000; + return 4; + } + if (n >= 100) + { + pow10 = 100; + return 3; + } + if (n >= 10) + { + pow10 = 10; + return 2; + } + + pow10 = 1; + return 1; +} + +inline void grisu2_round(char* buf, int len, std::uint64_t dist, std::uint64_t delta, + std::uint64_t rest, std::uint64_t ten_k) +{ + JSON_ASSERT(len >= 1); + JSON_ASSERT(dist <= delta); + JSON_ASSERT(rest <= delta); + JSON_ASSERT(ten_k > 0); + + // <--------------------------- delta ----> + // <---- dist ---------> + // --------------[------------------+-------------------]-------------- + // M- w M+ + // + // ten_k + // <------> + // <---- rest ----> + // --------------[------------------+----+--------------]-------------- + // w V + // = buf * 10^k + // + // ten_k represents a unit-in-the-last-place in the decimal representation + // stored in buf. + // Decrement buf by ten_k while this takes buf closer to w. + + // The tests are written in this order to avoid overflow in unsigned + // integer arithmetic. + + while (rest < dist + && delta - rest >= ten_k + && (rest + ten_k < dist || dist - rest > rest + ten_k - dist)) + { + JSON_ASSERT(buf[len - 1] != '0'); + buf[len - 1]--; + rest += ten_k; + } +} + +/*! +Generates V = buffer * 10^decimal_exponent, such that M- <= V <= M+. +M- and M+ must be normalized and share the same exponent -60 <= e <= -32. +*/ +inline void grisu2_digit_gen(char* buffer, int& length, int& decimal_exponent, + diyfp M_minus, diyfp w, diyfp M_plus) +{ + static_assert(kAlpha >= -60, "internal error"); + static_assert(kGamma <= -32, "internal error"); + + // Generates the digits (and the exponent) of a decimal floating-point + // number V = buffer * 10^decimal_exponent in the range [M-, M+]. The diyfp's + // w, M- and M+ share the same exponent e, which satisfies alpha <= e <= gamma. + // + // <--------------------------- delta ----> + // <---- dist ---------> + // --------------[------------------+-------------------]-------------- + // M- w M+ + // + // Grisu2 generates the digits of M+ from left to right and stops as soon as + // V is in [M-,M+]. + + JSON_ASSERT(M_plus.e >= kAlpha); + JSON_ASSERT(M_plus.e <= kGamma); + + std::uint64_t delta = diyfp::sub(M_plus, M_minus).f; // (significand of (M+ - M-), implicit exponent is e) + std::uint64_t dist = diyfp::sub(M_plus, w ).f; // (significand of (M+ - w ), implicit exponent is e) + + // Split M+ = f * 2^e into two parts p1 and p2 (note: e < 0): + // + // M+ = f * 2^e + // = ((f div 2^-e) * 2^-e + (f mod 2^-e)) * 2^e + // = ((p1 ) * 2^-e + (p2 )) * 2^e + // = p1 + p2 * 2^e + + const diyfp one(std::uint64_t{1} << -M_plus.e, M_plus.e); + + auto p1 = static_cast(M_plus.f >> -one.e); // p1 = f div 2^-e (Since -e >= 32, p1 fits into a 32-bit int.) + std::uint64_t p2 = M_plus.f & (one.f - 1); // p2 = f mod 2^-e + + // 1) + // + // Generate the digits of the integral part p1 = d[n-1]...d[1]d[0] + + JSON_ASSERT(p1 > 0); + + std::uint32_t pow10{}; + const int k = find_largest_pow10(p1, pow10); + + // 10^(k-1) <= p1 < 10^k, pow10 = 10^(k-1) + // + // p1 = (p1 div 10^(k-1)) * 10^(k-1) + (p1 mod 10^(k-1)) + // = (d[k-1] ) * 10^(k-1) + (p1 mod 10^(k-1)) + // + // M+ = p1 + p2 * 2^e + // = d[k-1] * 10^(k-1) + (p1 mod 10^(k-1)) + p2 * 2^e + // = d[k-1] * 10^(k-1) + ((p1 mod 10^(k-1)) * 2^-e + p2) * 2^e + // = d[k-1] * 10^(k-1) + ( rest) * 2^e + // + // Now generate the digits d[n] of p1 from left to right (n = k-1,...,0) + // + // p1 = d[k-1]...d[n] * 10^n + d[n-1]...d[0] + // + // but stop as soon as + // + // rest * 2^e = (d[n-1]...d[0] * 2^-e + p2) * 2^e <= delta * 2^e + + int n = k; + while (n > 0) + { + // Invariants: + // M+ = buffer * 10^n + (p1 + p2 * 2^e) (buffer = 0 for n = k) + // pow10 = 10^(n-1) <= p1 < 10^n + // + const std::uint32_t d = p1 / pow10; // d = p1 div 10^(n-1) + const std::uint32_t r = p1 % pow10; // r = p1 mod 10^(n-1) + // + // M+ = buffer * 10^n + (d * 10^(n-1) + r) + p2 * 2^e + // = (buffer * 10 + d) * 10^(n-1) + (r + p2 * 2^e) + // + JSON_ASSERT(d <= 9); + buffer[length++] = static_cast('0' + d); // buffer := buffer * 10 + d + // + // M+ = buffer * 10^(n-1) + (r + p2 * 2^e) + // + p1 = r; + n--; + // + // M+ = buffer * 10^n + (p1 + p2 * 2^e) + // pow10 = 10^n + // + + // Now check if enough digits have been generated. + // Compute + // + // p1 + p2 * 2^e = (p1 * 2^-e + p2) * 2^e = rest * 2^e + // + // Note: + // Since rest and delta share the same exponent e, it suffices to + // compare the significands. + const std::uint64_t rest = (std::uint64_t{p1} << -one.e) + p2; + if (rest <= delta) + { + // V = buffer * 10^n, with M- <= V <= M+. + + decimal_exponent += n; + + // We may now just stop. But instead look if the buffer could be + // decremented to bring V closer to w. + // + // pow10 = 10^n is now 1 ulp in the decimal representation V. + // The rounding procedure works with diyfp's with an implicit + // exponent of e. + // + // 10^n = (10^n * 2^-e) * 2^e = ulp * 2^e + // + const std::uint64_t ten_n = std::uint64_t{pow10} << -one.e; + grisu2_round(buffer, length, dist, delta, rest, ten_n); + + return; + } + + pow10 /= 10; + // + // pow10 = 10^(n-1) <= p1 < 10^n + // Invariants restored. + } + + // 2) + // + // The digits of the integral part have been generated: + // + // M+ = d[k-1]...d[1]d[0] + p2 * 2^e + // = buffer + p2 * 2^e + // + // Now generate the digits of the fractional part p2 * 2^e. + // + // Note: + // No decimal point is generated: the exponent is adjusted instead. + // + // p2 actually represents the fraction + // + // p2 * 2^e + // = p2 / 2^-e + // = d[-1] / 10^1 + d[-2] / 10^2 + ... + // + // Now generate the digits d[-m] of p1 from left to right (m = 1,2,...) + // + // p2 * 2^e = d[-1]d[-2]...d[-m] * 10^-m + // + 10^-m * (d[-m-1] / 10^1 + d[-m-2] / 10^2 + ...) + // + // using + // + // 10^m * p2 = ((10^m * p2) div 2^-e) * 2^-e + ((10^m * p2) mod 2^-e) + // = ( d) * 2^-e + ( r) + // + // or + // 10^m * p2 * 2^e = d + r * 2^e + // + // i.e. + // + // M+ = buffer + p2 * 2^e + // = buffer + 10^-m * (d + r * 2^e) + // = (buffer * 10^m + d) * 10^-m + 10^-m * r * 2^e + // + // and stop as soon as 10^-m * r * 2^e <= delta * 2^e + + JSON_ASSERT(p2 > delta); + + int m = 0; + for (;;) + { + // Invariant: + // M+ = buffer * 10^-m + 10^-m * (d[-m-1] / 10 + d[-m-2] / 10^2 + ...) * 2^e + // = buffer * 10^-m + 10^-m * (p2 ) * 2^e + // = buffer * 10^-m + 10^-m * (1/10 * (10 * p2) ) * 2^e + // = buffer * 10^-m + 10^-m * (1/10 * ((10*p2 div 2^-e) * 2^-e + (10*p2 mod 2^-e)) * 2^e + // + JSON_ASSERT(p2 <= (std::numeric_limits::max)() / 10); + p2 *= 10; + const std::uint64_t d = p2 >> -one.e; // d = (10 * p2) div 2^-e + const std::uint64_t r = p2 & (one.f - 1); // r = (10 * p2) mod 2^-e + // + // M+ = buffer * 10^-m + 10^-m * (1/10 * (d * 2^-e + r) * 2^e + // = buffer * 10^-m + 10^-m * (1/10 * (d + r * 2^e)) + // = (buffer * 10 + d) * 10^(-m-1) + 10^(-m-1) * r * 2^e + // + JSON_ASSERT(d <= 9); + buffer[length++] = static_cast('0' + d); // buffer := buffer * 10 + d + // + // M+ = buffer * 10^(-m-1) + 10^(-m-1) * r * 2^e + // + p2 = r; + m++; + // + // M+ = buffer * 10^-m + 10^-m * p2 * 2^e + // Invariant restored. + + // Check if enough digits have been generated. + // + // 10^-m * p2 * 2^e <= delta * 2^e + // p2 * 2^e <= 10^m * delta * 2^e + // p2 <= 10^m * delta + delta *= 10; + dist *= 10; + if (p2 <= delta) + { + break; + } + } + + // V = buffer * 10^-m, with M- <= V <= M+. + + decimal_exponent -= m; + + // 1 ulp in the decimal representation is now 10^-m. + // Since delta and dist are now scaled by 10^m, we need to do the + // same with ulp in order to keep the units in sync. + // + // 10^m * 10^-m = 1 = 2^-e * 2^e = ten_m * 2^e + // + const std::uint64_t ten_m = one.f; + grisu2_round(buffer, length, dist, delta, p2, ten_m); + + // By construction this algorithm generates the shortest possible decimal + // number (Loitsch, Theorem 6.2) which rounds back to w. + // For an input number of precision p, at least + // + // N = 1 + ceil(p * log_10(2)) + // + // decimal digits are sufficient to identify all binary floating-point + // numbers (Matula, "In-and-Out conversions"). + // This implies that the algorithm does not produce more than N decimal + // digits. + // + // N = 17 for p = 53 (IEEE double precision) + // N = 9 for p = 24 (IEEE single precision) +} + +/*! +v = buf * 10^decimal_exponent +len is the length of the buffer (number of decimal digits) +The buffer must be large enough, i.e. >= max_digits10. +*/ +JSON_HEDLEY_NON_NULL(1) +inline void grisu2(char* buf, int& len, int& decimal_exponent, + diyfp m_minus, diyfp v, diyfp m_plus) +{ + JSON_ASSERT(m_plus.e == m_minus.e); + JSON_ASSERT(m_plus.e == v.e); + + // --------(-----------------------+-----------------------)-------- (A) + // m- v m+ + // + // --------------------(-----------+-----------------------)-------- (B) + // m- v m+ + // + // First scale v (and m- and m+) such that the exponent is in the range + // [alpha, gamma]. + + const cached_power cached = get_cached_power_for_binary_exponent(m_plus.e); + + const diyfp c_minus_k(cached.f, cached.e); // = c ~= 10^-k + + // The exponent of the products is = v.e + c_minus_k.e + q and is in the range [alpha,gamma] + const diyfp w = diyfp::mul(v, c_minus_k); + const diyfp w_minus = diyfp::mul(m_minus, c_minus_k); + const diyfp w_plus = diyfp::mul(m_plus, c_minus_k); + + // ----(---+---)---------------(---+---)---------------(---+---)---- + // w- w w+ + // = c*m- = c*v = c*m+ + // + // diyfp::mul rounds its result and c_minus_k is approximated too. w, w- and + // w+ are now off by a small amount. + // In fact: + // + // w - v * 10^k < 1 ulp + // + // To account for this inaccuracy, add resp. subtract 1 ulp. + // + // --------+---[---------------(---+---)---------------]---+-------- + // w- M- w M+ w+ + // + // Now any number in [M-, M+] (bounds included) will round to w when input, + // regardless of how the input rounding algorithm breaks ties. + // + // And digit_gen generates the shortest possible such number in [M-, M+]. + // Note that this does not mean that Grisu2 always generates the shortest + // possible number in the interval (m-, m+). + const diyfp M_minus(w_minus.f + 1, w_minus.e); + const diyfp M_plus (w_plus.f - 1, w_plus.e ); + + decimal_exponent = -cached.k; // = -(-k) = k + + grisu2_digit_gen(buf, len, decimal_exponent, M_minus, w, M_plus); +} + +/*! +v = buf * 10^decimal_exponent +len is the length of the buffer (number of decimal digits) +The buffer must be large enough, i.e. >= max_digits10. +*/ +template +JSON_HEDLEY_NON_NULL(1) +void grisu2(char* buf, int& len, int& decimal_exponent, FloatType value) +{ + static_assert(diyfp::kPrecision >= std::numeric_limits::digits + 3, + "internal error: not enough precision"); + + JSON_ASSERT(std::isfinite(value)); + JSON_ASSERT(value > 0); + + // If the neighbors (and boundaries) of 'value' are always computed for double-precision + // numbers, all float's can be recovered using strtod (and strtof). However, the resulting + // decimal representations are not exactly "short". + // + // The documentation for 'std::to_chars' (https://en.cppreference.com/w/cpp/utility/to_chars) + // says "value is converted to a string as if by std::sprintf in the default ("C") locale" + // and since sprintf promotes float's to double's, I think this is exactly what 'std::to_chars' + // does. + // On the other hand, the documentation for 'std::to_chars' requires that "parsing the + // representation using the corresponding std::from_chars function recovers value exactly". That + // indicates that single precision floating-point numbers should be recovered using + // 'std::strtof'. + // + // NB: If the neighbors are computed for single-precision numbers, there is a single float + // (7.0385307e-26f) which can't be recovered using strtod. The resulting double precision + // value is off by 1 ulp. +#if 0 + const boundaries w = compute_boundaries(static_cast(value)); +#else + const boundaries w = compute_boundaries(value); +#endif + + grisu2(buf, len, decimal_exponent, w.minus, w.w, w.plus); +} + +/*! +@brief appends a decimal representation of e to buf +@return a pointer to the element following the exponent. +@pre -1000 < e < 1000 +*/ +JSON_HEDLEY_NON_NULL(1) +JSON_HEDLEY_RETURNS_NON_NULL +inline char* append_exponent(char* buf, int e) +{ + JSON_ASSERT(e > -1000); + JSON_ASSERT(e < 1000); + + if (e < 0) + { + e = -e; + *buf++ = '-'; + } + else + { + *buf++ = '+'; + } + + auto k = static_cast(e); + if (k < 10) + { + // Always print at least two digits in the exponent. + // This is for compatibility with printf("%g"). + *buf++ = '0'; + *buf++ = static_cast('0' + k); + } + else if (k < 100) + { + *buf++ = static_cast('0' + k / 10); + k %= 10; + *buf++ = static_cast('0' + k); + } + else + { + *buf++ = static_cast('0' + k / 100); + k %= 100; + *buf++ = static_cast('0' + k / 10); + k %= 10; + *buf++ = static_cast('0' + k); + } + + return buf; +} + +/*! +@brief prettify v = buf * 10^decimal_exponent + +If v is in the range [10^min_exp, 10^max_exp) it will be printed in fixed-point +notation. Otherwise it will be printed in exponential notation. + +@pre min_exp < 0 +@pre max_exp > 0 +*/ +JSON_HEDLEY_NON_NULL(1) +JSON_HEDLEY_RETURNS_NON_NULL +inline char* format_buffer(char* buf, int len, int decimal_exponent, + int min_exp, int max_exp) +{ + JSON_ASSERT(min_exp < 0); + JSON_ASSERT(max_exp > 0); + + const int k = len; + const int n = len + decimal_exponent; + + // v = buf * 10^(n-k) + // k is the length of the buffer (number of decimal digits) + // n is the position of the decimal point relative to the start of the buffer. + + if (k <= n && n <= max_exp) + { + // digits[000] + // len <= max_exp + 2 + + std::memset(buf + k, '0', static_cast(n) - static_cast(k)); + // Make it look like a floating-point number (#362, #378) + buf[n + 0] = '.'; + buf[n + 1] = '0'; + return buf + (static_cast(n) + 2); + } + + if (0 < n && n <= max_exp) + { + // dig.its + // len <= max_digits10 + 1 + + JSON_ASSERT(k > n); + + std::memmove(buf + (static_cast(n) + 1), buf + n, static_cast(k) - static_cast(n)); + buf[n] = '.'; + return buf + (static_cast(k) + 1U); + } + + if (min_exp < n && n <= 0) + { + // 0.[000]digits + // len <= 2 + (-min_exp - 1) + max_digits10 + + std::memmove(buf + (2 + static_cast(-n)), buf, static_cast(k)); + buf[0] = '0'; + buf[1] = '.'; + std::memset(buf + 2, '0', static_cast(-n)); + return buf + (2U + static_cast(-n) + static_cast(k)); + } + + if (k == 1) + { + // dE+123 + // len <= 1 + 5 + + buf += 1; + } + else + { + // d.igitsE+123 + // len <= max_digits10 + 1 + 5 + + std::memmove(buf + 2, buf + 1, static_cast(k) - 1); + buf[1] = '.'; + buf += 1 + static_cast(k); + } + + *buf++ = 'e'; + return append_exponent(buf, n - 1); +} + +} // namespace dtoa_impl + +/*! +@brief generates a decimal representation of the floating-point number value in [first, last). + +The format of the resulting decimal representation is similar to printf's %g +format. Returns an iterator pointing past-the-end of the decimal representation. + +@note The input number must be finite, i.e. NaN's and Inf's are not supported. +@note The buffer must be large enough. +@note The result is NOT null-terminated. +*/ +template +JSON_HEDLEY_NON_NULL(1, 2) +JSON_HEDLEY_RETURNS_NON_NULL +char* to_chars(char* first, const char* last, FloatType value) +{ + static_cast(last); // maybe unused - fix warning + JSON_ASSERT(std::isfinite(value)); + + // Use signbit(value) instead of (value < 0) since signbit works for -0. + if (std::signbit(value)) + { + value = -value; + *first++ = '-'; + } + +#ifdef __GNUC__ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wfloat-equal" +#endif + if (value == 0) // +-0 + { + *first++ = '0'; + // Make it look like a floating-point number (#362, #378) + *first++ = '.'; + *first++ = '0'; + return first; + } +#ifdef __GNUC__ +#pragma GCC diagnostic pop +#endif + + JSON_ASSERT(last - first >= std::numeric_limits::max_digits10); + + // Compute v = buffer * 10^decimal_exponent. + // The decimal digits are stored in the buffer, which needs to be interpreted + // as an unsigned decimal integer. + // len is the length of the buffer, i.e. the number of decimal digits. + int len = 0; + int decimal_exponent = 0; + dtoa_impl::grisu2(first, len, decimal_exponent, value); + + JSON_ASSERT(len <= std::numeric_limits::max_digits10); + + // Format the buffer like printf("%.*g", prec, value) + constexpr int kMinExp = -4; + // Use digits10 here to increase compatibility with version 2. + constexpr int kMaxExp = std::numeric_limits::digits10; + + JSON_ASSERT(last - first >= kMaxExp + 2); + JSON_ASSERT(last - first >= 2 + (-kMinExp - 1) + std::numeric_limits::max_digits10); + JSON_ASSERT(last - first >= std::numeric_limits::max_digits10 + 6); + + return dtoa_impl::format_buffer(first, len, decimal_exponent, kMinExp, kMaxExp); +} + +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + +// #include + +// #include + +// #include + +// #include + + +namespace nlohmann +{ +namespace detail +{ +/////////////////// +// serialization // +/////////////////// + +/// how to treat decoding errors +enum class error_handler_t +{ + strict, ///< throw a type_error exception in case of invalid UTF-8 + replace, ///< replace invalid UTF-8 sequences with U+FFFD + ignore ///< ignore invalid UTF-8 sequences +}; + +template +class serializer +{ + using string_t = typename BasicJsonType::string_t; + using number_float_t = typename BasicJsonType::number_float_t; + using number_integer_t = typename BasicJsonType::number_integer_t; + using number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using binary_char_t = typename BasicJsonType::binary_t::value_type; + static constexpr std::uint8_t UTF8_ACCEPT = 0; + static constexpr std::uint8_t UTF8_REJECT = 1; + + public: + /*! + @param[in] s output stream to serialize to + @param[in] ichar indentation character to use + @param[in] error_handler_ how to react on decoding errors + */ + serializer(output_adapter_t s, const char ichar, + error_handler_t error_handler_ = error_handler_t::strict) + : o(std::move(s)) + , loc(std::localeconv()) + , thousands_sep(loc->thousands_sep == nullptr ? '\0' : std::char_traits::to_char_type(* (loc->thousands_sep))) + , decimal_point(loc->decimal_point == nullptr ? '\0' : std::char_traits::to_char_type(* (loc->decimal_point))) + , indent_char(ichar) + , indent_string(512, indent_char) + , error_handler(error_handler_) + {} + + // delete because of pointer members + serializer(const serializer&) = delete; + serializer& operator=(const serializer&) = delete; + serializer(serializer&&) = delete; + serializer& operator=(serializer&&) = delete; + ~serializer() = default; + + /*! + @brief internal implementation of the serialization function + + This function is called by the public member function dump and organizes + the serialization internally. The indentation level is propagated as + additional parameter. In case of arrays and objects, the function is + called recursively. + + - strings and object keys are escaped using `escape_string()` + - integer numbers are converted implicitly via `operator<<` + - floating-point numbers are converted to a string using `"%g"` format + - binary values are serialized as objects containing the subtype and the + byte array + + @param[in] val value to serialize + @param[in] pretty_print whether the output shall be pretty-printed + @param[in] ensure_ascii If @a ensure_ascii is true, all non-ASCII characters + in the output are escaped with `\uXXXX` sequences, and the result consists + of ASCII characters only. + @param[in] indent_step the indent level + @param[in] current_indent the current indent level (only used internally) + */ + void dump(const BasicJsonType& val, + const bool pretty_print, + const bool ensure_ascii, + const unsigned int indent_step, + const unsigned int current_indent = 0) + { + switch (val.m_type) + { + case value_t::object: + { + if (val.m_value.object->empty()) + { + o->write_characters("{}", 2); + return; + } + + if (pretty_print) + { + o->write_characters("{\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + if (JSON_HEDLEY_UNLIKELY(indent_string.size() < new_indent)) + { + indent_string.resize(indent_string.size() * 2, ' '); + } + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (std::size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o->write_characters(indent_string.c_str(), new_indent); + o->write_character('\"'); + dump_escaped(i->first, ensure_ascii); + o->write_characters("\": ", 3); + dump(i->second, true, ensure_ascii, indent_step, new_indent); + o->write_characters(",\n", 2); + } + + // last element + JSON_ASSERT(i != val.m_value.object->cend()); + JSON_ASSERT(std::next(i) == val.m_value.object->cend()); + o->write_characters(indent_string.c_str(), new_indent); + o->write_character('\"'); + dump_escaped(i->first, ensure_ascii); + o->write_characters("\": ", 3); + dump(i->second, true, ensure_ascii, indent_step, new_indent); + + o->write_character('\n'); + o->write_characters(indent_string.c_str(), current_indent); + o->write_character('}'); + } + else + { + o->write_character('{'); + + // first n-1 elements + auto i = val.m_value.object->cbegin(); + for (std::size_t cnt = 0; cnt < val.m_value.object->size() - 1; ++cnt, ++i) + { + o->write_character('\"'); + dump_escaped(i->first, ensure_ascii); + o->write_characters("\":", 2); + dump(i->second, false, ensure_ascii, indent_step, current_indent); + o->write_character(','); + } + + // last element + JSON_ASSERT(i != val.m_value.object->cend()); + JSON_ASSERT(std::next(i) == val.m_value.object->cend()); + o->write_character('\"'); + dump_escaped(i->first, ensure_ascii); + o->write_characters("\":", 2); + dump(i->second, false, ensure_ascii, indent_step, current_indent); + + o->write_character('}'); + } + + return; + } + + case value_t::array: + { + if (val.m_value.array->empty()) + { + o->write_characters("[]", 2); + return; + } + + if (pretty_print) + { + o->write_characters("[\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + if (JSON_HEDLEY_UNLIKELY(indent_string.size() < new_indent)) + { + indent_string.resize(indent_string.size() * 2, ' '); + } + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); + i != val.m_value.array->cend() - 1; ++i) + { + o->write_characters(indent_string.c_str(), new_indent); + dump(*i, true, ensure_ascii, indent_step, new_indent); + o->write_characters(",\n", 2); + } + + // last element + JSON_ASSERT(!val.m_value.array->empty()); + o->write_characters(indent_string.c_str(), new_indent); + dump(val.m_value.array->back(), true, ensure_ascii, indent_step, new_indent); + + o->write_character('\n'); + o->write_characters(indent_string.c_str(), current_indent); + o->write_character(']'); + } + else + { + o->write_character('['); + + // first n-1 elements + for (auto i = val.m_value.array->cbegin(); + i != val.m_value.array->cend() - 1; ++i) + { + dump(*i, false, ensure_ascii, indent_step, current_indent); + o->write_character(','); + } + + // last element + JSON_ASSERT(!val.m_value.array->empty()); + dump(val.m_value.array->back(), false, ensure_ascii, indent_step, current_indent); + + o->write_character(']'); + } + + return; + } + + case value_t::string: + { + o->write_character('\"'); + dump_escaped(*val.m_value.string, ensure_ascii); + o->write_character('\"'); + return; + } + + case value_t::binary: + { + if (pretty_print) + { + o->write_characters("{\n", 2); + + // variable to hold indentation for recursive calls + const auto new_indent = current_indent + indent_step; + if (JSON_HEDLEY_UNLIKELY(indent_string.size() < new_indent)) + { + indent_string.resize(indent_string.size() * 2, ' '); + } + + o->write_characters(indent_string.c_str(), new_indent); + + o->write_characters("\"bytes\": [", 10); + + if (!val.m_value.binary->empty()) + { + for (auto i = val.m_value.binary->cbegin(); + i != val.m_value.binary->cend() - 1; ++i) + { + dump_integer(*i); + o->write_characters(", ", 2); + } + dump_integer(val.m_value.binary->back()); + } + + o->write_characters("],\n", 3); + o->write_characters(indent_string.c_str(), new_indent); + + o->write_characters("\"subtype\": ", 11); + if (val.m_value.binary->has_subtype()) + { + dump_integer(val.m_value.binary->subtype()); + } + else + { + o->write_characters("null", 4); + } + o->write_character('\n'); + o->write_characters(indent_string.c_str(), current_indent); + o->write_character('}'); + } + else + { + o->write_characters("{\"bytes\":[", 10); + + if (!val.m_value.binary->empty()) + { + for (auto i = val.m_value.binary->cbegin(); + i != val.m_value.binary->cend() - 1; ++i) + { + dump_integer(*i); + o->write_character(','); + } + dump_integer(val.m_value.binary->back()); + } + + o->write_characters("],\"subtype\":", 12); + if (val.m_value.binary->has_subtype()) + { + dump_integer(val.m_value.binary->subtype()); + o->write_character('}'); + } + else + { + o->write_characters("null}", 5); + } + } + return; + } + + case value_t::boolean: + { + if (val.m_value.boolean) + { + o->write_characters("true", 4); + } + else + { + o->write_characters("false", 5); + } + return; + } + + case value_t::number_integer: + { + dump_integer(val.m_value.number_integer); + return; + } + + case value_t::number_unsigned: + { + dump_integer(val.m_value.number_unsigned); + return; + } + + case value_t::number_float: + { + dump_float(val.m_value.number_float); + return; + } + + case value_t::discarded: + { + o->write_characters("", 11); + return; + } + + case value_t::null: + { + o->write_characters("null", 4); + return; + } + + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + } + + JSON_PRIVATE_UNLESS_TESTED: + /*! + @brief dump escaped string + + Escape a string by replacing certain special characters by a sequence of an + escape character (backslash) and another character and other control + characters by a sequence of "\u" followed by a four-digit hex + representation. The escaped string is written to output stream @a o. + + @param[in] s the string to escape + @param[in] ensure_ascii whether to escape non-ASCII characters with + \uXXXX sequences + + @complexity Linear in the length of string @a s. + */ + void dump_escaped(const string_t& s, const bool ensure_ascii) + { + std::uint32_t codepoint{}; + std::uint8_t state = UTF8_ACCEPT; + std::size_t bytes = 0; // number of bytes written to string_buffer + + // number of bytes written at the point of the last valid byte + std::size_t bytes_after_last_accept = 0; + std::size_t undumped_chars = 0; + + for (std::size_t i = 0; i < s.size(); ++i) + { + const auto byte = static_cast(s[i]); + + switch (decode(state, codepoint, byte)) + { + case UTF8_ACCEPT: // decode found a new code point + { + switch (codepoint) + { + case 0x08: // backspace + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 'b'; + break; + } + + case 0x09: // horizontal tab + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 't'; + break; + } + + case 0x0A: // newline + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 'n'; + break; + } + + case 0x0C: // formfeed + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 'f'; + break; + } + + case 0x0D: // carriage return + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 'r'; + break; + } + + case 0x22: // quotation mark + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = '\"'; + break; + } + + case 0x5C: // reverse solidus + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = '\\'; + break; + } + + default: + { + // escape control characters (0x00..0x1F) or, if + // ensure_ascii parameter is used, non-ASCII characters + if ((codepoint <= 0x1F) || (ensure_ascii && (codepoint >= 0x7F))) + { + if (codepoint <= 0xFFFF) + { + // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + (std::snprintf)(string_buffer.data() + bytes, 7, "\\u%04x", + static_cast(codepoint)); + bytes += 6; + } + else + { + // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + (std::snprintf)(string_buffer.data() + bytes, 13, "\\u%04x\\u%04x", + static_cast(0xD7C0u + (codepoint >> 10u)), + static_cast(0xDC00u + (codepoint & 0x3FFu))); + bytes += 12; + } + } + else + { + // copy byte to buffer (all previous bytes + // been copied have in default case above) + string_buffer[bytes++] = s[i]; + } + break; + } + } + + // write buffer and reset index; there must be 13 bytes + // left, as this is the maximal number of bytes to be + // written ("\uxxxx\uxxxx\0") for one code point + if (string_buffer.size() - bytes < 13) + { + o->write_characters(string_buffer.data(), bytes); + bytes = 0; + } + + // remember the byte position of this accept + bytes_after_last_accept = bytes; + undumped_chars = 0; + break; + } + + case UTF8_REJECT: // decode found invalid UTF-8 byte + { + switch (error_handler) + { + case error_handler_t::strict: + { + std::string sn(9, '\0'); + // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + (std::snprintf)(&sn[0], sn.size(), "%.2X", byte); + JSON_THROW(type_error::create(316, "invalid UTF-8 byte at index " + std::to_string(i) + ": 0x" + sn, BasicJsonType())); + } + + case error_handler_t::ignore: + case error_handler_t::replace: + { + // in case we saw this character the first time, we + // would like to read it again, because the byte + // may be OK for itself, but just not OK for the + // previous sequence + if (undumped_chars > 0) + { + --i; + } + + // reset length buffer to the last accepted index; + // thus removing/ignoring the invalid characters + bytes = bytes_after_last_accept; + + if (error_handler == error_handler_t::replace) + { + // add a replacement character + if (ensure_ascii) + { + string_buffer[bytes++] = '\\'; + string_buffer[bytes++] = 'u'; + string_buffer[bytes++] = 'f'; + string_buffer[bytes++] = 'f'; + string_buffer[bytes++] = 'f'; + string_buffer[bytes++] = 'd'; + } + else + { + string_buffer[bytes++] = detail::binary_writer::to_char_type('\xEF'); + string_buffer[bytes++] = detail::binary_writer::to_char_type('\xBF'); + string_buffer[bytes++] = detail::binary_writer::to_char_type('\xBD'); + } + + // write buffer and reset index; there must be 13 bytes + // left, as this is the maximal number of bytes to be + // written ("\uxxxx\uxxxx\0") for one code point + if (string_buffer.size() - bytes < 13) + { + o->write_characters(string_buffer.data(), bytes); + bytes = 0; + } + + bytes_after_last_accept = bytes; + } + + undumped_chars = 0; + + // continue processing the string + state = UTF8_ACCEPT; + break; + } + + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + break; + } + + default: // decode found yet incomplete multi-byte code point + { + if (!ensure_ascii) + { + // code point will not be escaped - copy byte to buffer + string_buffer[bytes++] = s[i]; + } + ++undumped_chars; + break; + } + } + } + + // we finished processing the string + if (JSON_HEDLEY_LIKELY(state == UTF8_ACCEPT)) + { + // write buffer + if (bytes > 0) + { + o->write_characters(string_buffer.data(), bytes); + } + } + else + { + // we finish reading, but do not accept: string was incomplete + switch (error_handler) + { + case error_handler_t::strict: + { + std::string sn(9, '\0'); + // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + (std::snprintf)(&sn[0], sn.size(), "%.2X", static_cast(s.back())); + JSON_THROW(type_error::create(316, "incomplete UTF-8 string; last byte: 0x" + sn, BasicJsonType())); + } + + case error_handler_t::ignore: + { + // write all accepted bytes + o->write_characters(string_buffer.data(), bytes_after_last_accept); + break; + } + + case error_handler_t::replace: + { + // write all accepted bytes + o->write_characters(string_buffer.data(), bytes_after_last_accept); + // add a replacement character + if (ensure_ascii) + { + o->write_characters("\\ufffd", 6); + } + else + { + o->write_characters("\xEF\xBF\xBD", 3); + } + break; + } + + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + } + } + + private: + /*! + @brief count digits + + Count the number of decimal (base 10) digits for an input unsigned integer. + + @param[in] x unsigned integer number to count its digits + @return number of decimal digits + */ + inline unsigned int count_digits(number_unsigned_t x) noexcept + { + unsigned int n_digits = 1; + for (;;) + { + if (x < 10) + { + return n_digits; + } + if (x < 100) + { + return n_digits + 1; + } + if (x < 1000) + { + return n_digits + 2; + } + if (x < 10000) + { + return n_digits + 3; + } + x = x / 10000u; + n_digits += 4; + } + } + + /*! + @brief dump an integer + + Dump a given integer to output stream @a o. Works internally with + @a number_buffer. + + @param[in] x integer number (signed or unsigned) to dump + @tparam NumberType either @a number_integer_t or @a number_unsigned_t + */ + template < typename NumberType, detail::enable_if_t < + std::is_integral::value || + std::is_same::value || + std::is_same::value || + std::is_same::value, + int > = 0 > + void dump_integer(NumberType x) + { + static constexpr std::array, 100> digits_to_99 + { + { + {{'0', '0'}}, {{'0', '1'}}, {{'0', '2'}}, {{'0', '3'}}, {{'0', '4'}}, {{'0', '5'}}, {{'0', '6'}}, {{'0', '7'}}, {{'0', '8'}}, {{'0', '9'}}, + {{'1', '0'}}, {{'1', '1'}}, {{'1', '2'}}, {{'1', '3'}}, {{'1', '4'}}, {{'1', '5'}}, {{'1', '6'}}, {{'1', '7'}}, {{'1', '8'}}, {{'1', '9'}}, + {{'2', '0'}}, {{'2', '1'}}, {{'2', '2'}}, {{'2', '3'}}, {{'2', '4'}}, {{'2', '5'}}, {{'2', '6'}}, {{'2', '7'}}, {{'2', '8'}}, {{'2', '9'}}, + {{'3', '0'}}, {{'3', '1'}}, {{'3', '2'}}, {{'3', '3'}}, {{'3', '4'}}, {{'3', '5'}}, {{'3', '6'}}, {{'3', '7'}}, {{'3', '8'}}, {{'3', '9'}}, + {{'4', '0'}}, {{'4', '1'}}, {{'4', '2'}}, {{'4', '3'}}, {{'4', '4'}}, {{'4', '5'}}, {{'4', '6'}}, {{'4', '7'}}, {{'4', '8'}}, {{'4', '9'}}, + {{'5', '0'}}, {{'5', '1'}}, {{'5', '2'}}, {{'5', '3'}}, {{'5', '4'}}, {{'5', '5'}}, {{'5', '6'}}, {{'5', '7'}}, {{'5', '8'}}, {{'5', '9'}}, + {{'6', '0'}}, {{'6', '1'}}, {{'6', '2'}}, {{'6', '3'}}, {{'6', '4'}}, {{'6', '5'}}, {{'6', '6'}}, {{'6', '7'}}, {{'6', '8'}}, {{'6', '9'}}, + {{'7', '0'}}, {{'7', '1'}}, {{'7', '2'}}, {{'7', '3'}}, {{'7', '4'}}, {{'7', '5'}}, {{'7', '6'}}, {{'7', '7'}}, {{'7', '8'}}, {{'7', '9'}}, + {{'8', '0'}}, {{'8', '1'}}, {{'8', '2'}}, {{'8', '3'}}, {{'8', '4'}}, {{'8', '5'}}, {{'8', '6'}}, {{'8', '7'}}, {{'8', '8'}}, {{'8', '9'}}, + {{'9', '0'}}, {{'9', '1'}}, {{'9', '2'}}, {{'9', '3'}}, {{'9', '4'}}, {{'9', '5'}}, {{'9', '6'}}, {{'9', '7'}}, {{'9', '8'}}, {{'9', '9'}}, + } + }; + + // special case for "0" + if (x == 0) + { + o->write_character('0'); + return; + } + + // use a pointer to fill the buffer + auto buffer_ptr = number_buffer.begin(); // NOLINT(llvm-qualified-auto,readability-qualified-auto,cppcoreguidelines-pro-type-vararg,hicpp-vararg) + + const bool is_negative = std::is_signed::value && !(x >= 0); // see issue #755 + number_unsigned_t abs_value; + + unsigned int n_chars{}; + + if (is_negative) + { + *buffer_ptr = '-'; + abs_value = remove_sign(static_cast(x)); + + // account one more byte for the minus sign + n_chars = 1 + count_digits(abs_value); + } + else + { + abs_value = static_cast(x); + n_chars = count_digits(abs_value); + } + + // spare 1 byte for '\0' + JSON_ASSERT(n_chars < number_buffer.size() - 1); + + // jump to the end to generate the string from backward + // so we later avoid reversing the result + buffer_ptr += n_chars; + + // Fast int2ascii implementation inspired by "Fastware" talk by Andrei Alexandrescu + // See: https://www.youtube.com/watch?v=o4-CwDo2zpg + while (abs_value >= 100) + { + const auto digits_index = static_cast((abs_value % 100)); + abs_value /= 100; + *(--buffer_ptr) = digits_to_99[digits_index][1]; + *(--buffer_ptr) = digits_to_99[digits_index][0]; + } + + if (abs_value >= 10) + { + const auto digits_index = static_cast(abs_value); + *(--buffer_ptr) = digits_to_99[digits_index][1]; + *(--buffer_ptr) = digits_to_99[digits_index][0]; + } + else + { + *(--buffer_ptr) = static_cast('0' + abs_value); + } + + o->write_characters(number_buffer.data(), n_chars); + } + + /*! + @brief dump a floating-point number + + Dump a given floating-point number to output stream @a o. Works internally + with @a number_buffer. + + @param[in] x floating-point number to dump + */ + void dump_float(number_float_t x) + { + // NaN / inf + if (!std::isfinite(x)) + { + o->write_characters("null", 4); + return; + } + + // If number_float_t is an IEEE-754 single or double precision number, + // use the Grisu2 algorithm to produce short numbers which are + // guaranteed to round-trip, using strtof and strtod, resp. + // + // NB: The test below works if == . + static constexpr bool is_ieee_single_or_double + = (std::numeric_limits::is_iec559 && std::numeric_limits::digits == 24 && std::numeric_limits::max_exponent == 128) || + (std::numeric_limits::is_iec559 && std::numeric_limits::digits == 53 && std::numeric_limits::max_exponent == 1024); + + dump_float(x, std::integral_constant()); + } + + void dump_float(number_float_t x, std::true_type /*is_ieee_single_or_double*/) + { + auto* begin = number_buffer.data(); + auto* end = ::nlohmann::detail::to_chars(begin, begin + number_buffer.size(), x); + + o->write_characters(begin, static_cast(end - begin)); + } + + void dump_float(number_float_t x, std::false_type /*is_ieee_single_or_double*/) + { + // get number of digits for a float -> text -> float round-trip + static constexpr auto d = std::numeric_limits::max_digits10; + + // the actual conversion + // NOLINTNEXTLINE(cppcoreguidelines-pro-type-vararg,hicpp-vararg) + std::ptrdiff_t len = (std::snprintf)(number_buffer.data(), number_buffer.size(), "%.*g", d, x); + + // negative value indicates an error + JSON_ASSERT(len > 0); + // check if buffer was large enough + JSON_ASSERT(static_cast(len) < number_buffer.size()); + + // erase thousands separator + if (thousands_sep != '\0') + { + // NOLINTNEXTLINE(readability-qualified-auto,llvm-qualified-auto): std::remove returns an iterator, see https://github.com/nlohmann/json/issues/3081 + const auto end = std::remove(number_buffer.begin(), number_buffer.begin() + len, thousands_sep); + std::fill(end, number_buffer.end(), '\0'); + JSON_ASSERT((end - number_buffer.begin()) <= len); + len = (end - number_buffer.begin()); + } + + // convert decimal point to '.' + if (decimal_point != '\0' && decimal_point != '.') + { + // NOLINTNEXTLINE(readability-qualified-auto,llvm-qualified-auto): std::find returns an iterator, see https://github.com/nlohmann/json/issues/3081 + const auto dec_pos = std::find(number_buffer.begin(), number_buffer.end(), decimal_point); + if (dec_pos != number_buffer.end()) + { + *dec_pos = '.'; + } + } + + o->write_characters(number_buffer.data(), static_cast(len)); + + // determine if need to append ".0" + const bool value_is_int_like = + std::none_of(number_buffer.begin(), number_buffer.begin() + len + 1, + [](char c) + { + return c == '.' || c == 'e'; + }); + + if (value_is_int_like) + { + o->write_characters(".0", 2); + } + } + + /*! + @brief check whether a string is UTF-8 encoded + + The function checks each byte of a string whether it is UTF-8 encoded. The + result of the check is stored in the @a state parameter. The function must + be called initially with state 0 (accept). State 1 means the string must + be rejected, because the current byte is not allowed. If the string is + completely processed, but the state is non-zero, the string ended + prematurely; that is, the last byte indicated more bytes should have + followed. + + @param[in,out] state the state of the decoding + @param[in,out] codep codepoint (valid only if resulting state is UTF8_ACCEPT) + @param[in] byte next byte to decode + @return new state + + @note The function has been edited: a std::array is used. + + @copyright Copyright (c) 2008-2009 Bjoern Hoehrmann + @sa http://bjoern.hoehrmann.de/utf-8/decoder/dfa/ + */ + static std::uint8_t decode(std::uint8_t& state, std::uint32_t& codep, const std::uint8_t byte) noexcept + { + static const std::array utf8d = + { + { + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 00..1F + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 20..3F + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 40..5F + 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // 60..7F + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, // 80..9F + 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, // A0..BF + 8, 8, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, // C0..DF + 0xA, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x3, 0x4, 0x3, 0x3, // E0..EF + 0xB, 0x6, 0x6, 0x6, 0x5, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, 0x8, // F0..FF + 0x0, 0x1, 0x2, 0x3, 0x5, 0x8, 0x7, 0x1, 0x1, 0x1, 0x4, 0x6, 0x1, 0x1, 0x1, 0x1, // s0..s0 + 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 0, 1, 0, 1, 1, 1, 1, 1, 1, // s1..s2 + 1, 2, 1, 1, 1, 1, 1, 2, 1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 1, // s3..s4 + 1, 2, 1, 1, 1, 1, 1, 1, 1, 2, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 3, 1, 3, 1, 1, 1, 1, 1, 1, // s5..s6 + 1, 3, 1, 1, 1, 1, 1, 3, 1, 3, 1, 1, 1, 1, 1, 1, 1, 3, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // s7..s8 + } + }; + + JSON_ASSERT(byte < utf8d.size()); + const std::uint8_t type = utf8d[byte]; + + codep = (state != UTF8_ACCEPT) + ? (byte & 0x3fu) | (codep << 6u) + : (0xFFu >> type) & (byte); + + std::size_t index = 256u + static_cast(state) * 16u + static_cast(type); + JSON_ASSERT(index < 400); + state = utf8d[index]; + return state; + } + + /* + * Overload to make the compiler happy while it is instantiating + * dump_integer for number_unsigned_t. + * Must never be called. + */ + number_unsigned_t remove_sign(number_unsigned_t x) + { + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + return x; // LCOV_EXCL_LINE + } + + /* + * Helper function for dump_integer + * + * This function takes a negative signed integer and returns its absolute + * value as unsigned integer. The plus/minus shuffling is necessary as we can + * not directly remove the sign of an arbitrary signed integer as the + * absolute values of INT_MIN and INT_MAX are usually not the same. See + * #1708 for details. + */ + inline number_unsigned_t remove_sign(number_integer_t x) noexcept + { + JSON_ASSERT(x < 0 && x < (std::numeric_limits::max)()); // NOLINT(misc-redundant-expression) + return static_cast(-(x + 1)) + 1; + } + + private: + /// the output of the serializer + output_adapter_t o = nullptr; + + /// a (hopefully) large enough character buffer + std::array number_buffer{{}}; + + /// the locale + const std::lconv* loc = nullptr; + /// the locale's thousand separator character + const char thousands_sep = '\0'; + /// the locale's decimal point character + const char decimal_point = '\0'; + + /// string buffer + std::array string_buffer{{}}; + + /// the indentation character + const char indent_char; + /// the indentation string + string_t indent_string; + + /// error_handler how to react on decoding errors + const error_handler_t error_handler; +}; +} // namespace detail +} // namespace nlohmann + +// #include + +// #include + +// #include + + +#include // less +#include // initializer_list +#include // input_iterator_tag, iterator_traits +#include // allocator +#include // for out_of_range +#include // enable_if, is_convertible +#include // pair +#include // vector + +// #include + + +namespace nlohmann +{ + +/// ordered_map: a minimal map-like container that preserves insertion order +/// for use within nlohmann::basic_json +template , + class Allocator = std::allocator>> + struct ordered_map : std::vector, Allocator> +{ + using key_type = Key; + using mapped_type = T; + using Container = std::vector, Allocator>; + using typename Container::iterator; + using typename Container::const_iterator; + using typename Container::size_type; + using typename Container::value_type; + + // Explicit constructors instead of `using Container::Container` + // otherwise older compilers choke on it (GCC <= 5.5, xcode <= 9.4) + ordered_map(const Allocator& alloc = Allocator()) : Container{alloc} {} + template + ordered_map(It first, It last, const Allocator& alloc = Allocator()) + : Container{first, last, alloc} {} + ordered_map(std::initializer_list init, const Allocator& alloc = Allocator() ) + : Container{init, alloc} {} + + std::pair emplace(const key_type& key, T&& t) + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return {it, false}; + } + } + Container::emplace_back(key, t); + return {--this->end(), true}; + } + + T& operator[](const Key& key) + { + return emplace(key, T{}).first->second; + } + + const T& operator[](const Key& key) const + { + return at(key); + } + + T& at(const Key& key) + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return it->second; + } + } + + JSON_THROW(std::out_of_range("key not found")); + } + + const T& at(const Key& key) const + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return it->second; + } + } + + JSON_THROW(std::out_of_range("key not found")); + } + + size_type erase(const Key& key) + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + // Since we cannot move const Keys, re-construct them in place + for (auto next = it; ++next != this->end(); ++it) + { + it->~value_type(); // Destroy but keep allocation + new (&*it) value_type{std::move(*next)}; + } + Container::pop_back(); + return 1; + } + } + return 0; + } + + iterator erase(iterator pos) + { + auto it = pos; + + // Since we cannot move const Keys, re-construct them in place + for (auto next = it; ++next != this->end(); ++it) + { + it->~value_type(); // Destroy but keep allocation + new (&*it) value_type{std::move(*next)}; + } + Container::pop_back(); + return pos; + } + + size_type count(const Key& key) const + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return 1; + } + } + return 0; + } + + iterator find(const Key& key) + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return it; + } + } + return Container::end(); + } + + const_iterator find(const Key& key) const + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == key) + { + return it; + } + } + return Container::end(); + } + + std::pair insert( value_type&& value ) + { + return emplace(value.first, std::move(value.second)); + } + + std::pair insert( const value_type& value ) + { + for (auto it = this->begin(); it != this->end(); ++it) + { + if (it->first == value.first) + { + return {it, false}; + } + } + Container::push_back(value); + return {--this->end(), true}; + } + + template + using require_input_iter = typename std::enable_if::iterator_category, + std::input_iterator_tag>::value>::type; + + template> + void insert(InputIt first, InputIt last) + { + for (auto it = first; it != last; ++it) + { + insert(*it); + } + } +}; + +} // namespace nlohmann + + +#if defined(JSON_HAS_CPP_17) + #include +#endif + +/*! +@brief namespace for Niels Lohmann +@see https://github.com/nlohmann +@since version 1.0.0 +*/ +namespace nlohmann +{ + +/*! +@brief a class to store JSON values + +@tparam ObjectType type for JSON objects (`std::map` by default; will be used +in @ref object_t) +@tparam ArrayType type for JSON arrays (`std::vector` by default; will be used +in @ref array_t) +@tparam StringType type for JSON strings and object keys (`std::string` by +default; will be used in @ref string_t) +@tparam BooleanType type for JSON booleans (`bool` by default; will be used +in @ref boolean_t) +@tparam NumberIntegerType type for JSON integer numbers (`int64_t` by +default; will be used in @ref number_integer_t) +@tparam NumberUnsignedType type for JSON unsigned integer numbers (@c +`uint64_t` by default; will be used in @ref number_unsigned_t) +@tparam NumberFloatType type for JSON floating-point numbers (`double` by +default; will be used in @ref number_float_t) +@tparam BinaryType type for packed binary data for compatibility with binary +serialization formats (`std::vector` by default; will be used in +@ref binary_t) +@tparam AllocatorType type of the allocator to use (`std::allocator` by +default) +@tparam JSONSerializer the serializer to resolve internal calls to `to_json()` +and `from_json()` (@ref adl_serializer by default) + +@requirement The class satisfies the following concept requirements: +- Basic + - [DefaultConstructible](https://en.cppreference.com/w/cpp/named_req/DefaultConstructible): + JSON values can be default constructed. The result will be a JSON null + value. + - [MoveConstructible](https://en.cppreference.com/w/cpp/named_req/MoveConstructible): + A JSON value can be constructed from an rvalue argument. + - [CopyConstructible](https://en.cppreference.com/w/cpp/named_req/CopyConstructible): + A JSON value can be copy-constructed from an lvalue expression. + - [MoveAssignable](https://en.cppreference.com/w/cpp/named_req/MoveAssignable): + A JSON value van be assigned from an rvalue argument. + - [CopyAssignable](https://en.cppreference.com/w/cpp/named_req/CopyAssignable): + A JSON value can be copy-assigned from an lvalue expression. + - [Destructible](https://en.cppreference.com/w/cpp/named_req/Destructible): + JSON values can be destructed. +- Layout + - [StandardLayoutType](https://en.cppreference.com/w/cpp/named_req/StandardLayoutType): + JSON values have + [standard layout](https://en.cppreference.com/w/cpp/language/data_members#Standard_layout): + All non-static data members are private and standard layout types, the + class has no virtual functions or (virtual) base classes. +- Library-wide + - [EqualityComparable](https://en.cppreference.com/w/cpp/named_req/EqualityComparable): + JSON values can be compared with `==`, see @ref + operator==(const_reference,const_reference). + - [LessThanComparable](https://en.cppreference.com/w/cpp/named_req/LessThanComparable): + JSON values can be compared with `<`, see @ref + operator<(const_reference,const_reference). + - [Swappable](https://en.cppreference.com/w/cpp/named_req/Swappable): + Any JSON lvalue or rvalue of can be swapped with any lvalue or rvalue of + other compatible types, using unqualified function call @ref swap(). + - [NullablePointer](https://en.cppreference.com/w/cpp/named_req/NullablePointer): + JSON values can be compared against `std::nullptr_t` objects which are used + to model the `null` value. +- Container + - [Container](https://en.cppreference.com/w/cpp/named_req/Container): + JSON values can be used like STL containers and provide iterator access. + - [ReversibleContainer](https://en.cppreference.com/w/cpp/named_req/ReversibleContainer); + JSON values can be used like STL containers and provide reverse iterator + access. + +@invariant The member variables @a m_value and @a m_type have the following +relationship: +- If `m_type == value_t::object`, then `m_value.object != nullptr`. +- If `m_type == value_t::array`, then `m_value.array != nullptr`. +- If `m_type == value_t::string`, then `m_value.string != nullptr`. +The invariants are checked by member function assert_invariant(). + +@internal +@note ObjectType trick from https://stackoverflow.com/a/9860911 +@endinternal + +@see [RFC 8259: The JavaScript Object Notation (JSON) Data Interchange +Format](https://tools.ietf.org/html/rfc8259) + +@since version 1.0.0 + +@nosubgrouping +*/ +NLOHMANN_BASIC_JSON_TPL_DECLARATION +class basic_json // NOLINT(cppcoreguidelines-special-member-functions,hicpp-special-member-functions) +{ + private: + template friend struct detail::external_constructor; + friend ::nlohmann::json_pointer; + + template + friend class ::nlohmann::detail::parser; + friend ::nlohmann::detail::serializer; + template + friend class ::nlohmann::detail::iter_impl; + template + friend class ::nlohmann::detail::binary_writer; + template + friend class ::nlohmann::detail::binary_reader; + template + friend class ::nlohmann::detail::json_sax_dom_parser; + template + friend class ::nlohmann::detail::json_sax_dom_callback_parser; + friend class ::nlohmann::detail::exception; + + /// workaround type for MSVC + using basic_json_t = NLOHMANN_BASIC_JSON_TPL; + + JSON_PRIVATE_UNLESS_TESTED: + // convenience aliases for types residing in namespace detail; + using lexer = ::nlohmann::detail::lexer_base; + + template + static ::nlohmann::detail::parser parser( + InputAdapterType adapter, + detail::parser_callback_tcb = nullptr, + const bool allow_exceptions = true, + const bool ignore_comments = false + ) + { + return ::nlohmann::detail::parser(std::move(adapter), + std::move(cb), allow_exceptions, ignore_comments); + } + + private: + using primitive_iterator_t = ::nlohmann::detail::primitive_iterator_t; + template + using internal_iterator = ::nlohmann::detail::internal_iterator; + template + using iter_impl = ::nlohmann::detail::iter_impl; + template + using iteration_proxy = ::nlohmann::detail::iteration_proxy; + template using json_reverse_iterator = ::nlohmann::detail::json_reverse_iterator; + + template + using output_adapter_t = ::nlohmann::detail::output_adapter_t; + + template + using binary_reader = ::nlohmann::detail::binary_reader; + template using binary_writer = ::nlohmann::detail::binary_writer; + + JSON_PRIVATE_UNLESS_TESTED: + using serializer = ::nlohmann::detail::serializer; + + public: + using value_t = detail::value_t; + /// JSON Pointer, see @ref nlohmann::json_pointer + using json_pointer = ::nlohmann::json_pointer; + template + using json_serializer = JSONSerializer; + /// how to treat decoding errors + using error_handler_t = detail::error_handler_t; + /// how to treat CBOR tags + using cbor_tag_handler_t = detail::cbor_tag_handler_t; + /// helper type for initializer lists of basic_json values + using initializer_list_t = std::initializer_list>; + + using input_format_t = detail::input_format_t; + /// SAX interface type, see @ref nlohmann::json_sax + using json_sax_t = json_sax; + + //////////////// + // exceptions // + //////////////// + + /// @name exceptions + /// Classes to implement user-defined exceptions. + /// @{ + + /// @copydoc detail::exception + using exception = detail::exception; + /// @copydoc detail::parse_error + using parse_error = detail::parse_error; + /// @copydoc detail::invalid_iterator + using invalid_iterator = detail::invalid_iterator; + /// @copydoc detail::type_error + using type_error = detail::type_error; + /// @copydoc detail::out_of_range + using out_of_range = detail::out_of_range; + /// @copydoc detail::other_error + using other_error = detail::other_error; + + /// @} + + + ///////////////////// + // container types // + ///////////////////// + + /// @name container types + /// The canonic container types to use @ref basic_json like any other STL + /// container. + /// @{ + + /// the type of elements in a basic_json container + using value_type = basic_json; + + /// the type of an element reference + using reference = value_type&; + /// the type of an element const reference + using const_reference = const value_type&; + + /// a type to represent differences between iterators + using difference_type = std::ptrdiff_t; + /// a type to represent container sizes + using size_type = std::size_t; + + /// the allocator type + using allocator_type = AllocatorType; + + /// the type of an element pointer + using pointer = typename std::allocator_traits::pointer; + /// the type of an element const pointer + using const_pointer = typename std::allocator_traits::const_pointer; + + /// an iterator for a basic_json container + using iterator = iter_impl; + /// a const iterator for a basic_json container + using const_iterator = iter_impl; + /// a reverse iterator for a basic_json container + using reverse_iterator = json_reverse_iterator; + /// a const reverse iterator for a basic_json container + using const_reverse_iterator = json_reverse_iterator; + + /// @} + + + /*! + @brief returns the allocator associated with the container + */ + static allocator_type get_allocator() + { + return allocator_type(); + } + + /*! + @brief returns version information on the library + + This function returns a JSON object with information about the library, + including the version number and information on the platform and compiler. + + @return JSON object holding version information + key | description + ----------- | --------------- + `compiler` | Information on the used compiler. It is an object with the following keys: `c++` (the used C++ standard), `family` (the compiler family; possible values are `clang`, `icc`, `gcc`, `ilecpp`, `msvc`, `pgcpp`, `sunpro`, and `unknown`), and `version` (the compiler version). + `copyright` | The copyright line for the library as string. + `name` | The name of the library as string. + `platform` | The used platform as string. Possible values are `win32`, `linux`, `apple`, `unix`, and `unknown`. + `url` | The URL of the project as string. + `version` | The version of the library. It is an object with the following keys: `major`, `minor`, and `patch` as defined by [Semantic Versioning](http://semver.org), and `string` (the version string). + + @liveexample{The following code shows an example output of the `meta()` + function.,meta} + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @complexity Constant. + + @since 2.1.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json meta() + { + basic_json result; + + result["copyright"] = "(C) 2013-2021 Niels Lohmann"; + result["name"] = "JSON for Modern C++"; + result["url"] = "https://github.com/nlohmann/json"; + result["version"]["string"] = + std::to_string(NLOHMANN_JSON_VERSION_MAJOR) + "." + + std::to_string(NLOHMANN_JSON_VERSION_MINOR) + "." + + std::to_string(NLOHMANN_JSON_VERSION_PATCH); + result["version"]["major"] = NLOHMANN_JSON_VERSION_MAJOR; + result["version"]["minor"] = NLOHMANN_JSON_VERSION_MINOR; + result["version"]["patch"] = NLOHMANN_JSON_VERSION_PATCH; + +#ifdef _WIN32 + result["platform"] = "win32"; +#elif defined __linux__ + result["platform"] = "linux"; +#elif defined __APPLE__ + result["platform"] = "apple"; +#elif defined __unix__ + result["platform"] = "unix"; +#else + result["platform"] = "unknown"; +#endif + +#if defined(__ICC) || defined(__INTEL_COMPILER) + result["compiler"] = {{"family", "icc"}, {"version", __INTEL_COMPILER}}; +#elif defined(__clang__) + result["compiler"] = {{"family", "clang"}, {"version", __clang_version__}}; +#elif defined(__GNUC__) || defined(__GNUG__) + result["compiler"] = {{"family", "gcc"}, {"version", std::to_string(__GNUC__) + "." + std::to_string(__GNUC_MINOR__) + "." + std::to_string(__GNUC_PATCHLEVEL__)}}; +#elif defined(__HP_cc) || defined(__HP_aCC) + result["compiler"] = "hp" +#elif defined(__IBMCPP__) + result["compiler"] = {{"family", "ilecpp"}, {"version", __IBMCPP__}}; +#elif defined(_MSC_VER) + result["compiler"] = {{"family", "msvc"}, {"version", _MSC_VER}}; +#elif defined(__PGI) + result["compiler"] = {{"family", "pgcpp"}, {"version", __PGI}}; +#elif defined(__SUNPRO_CC) + result["compiler"] = {{"family", "sunpro"}, {"version", __SUNPRO_CC}}; +#else + result["compiler"] = {{"family", "unknown"}, {"version", "unknown"}}; +#endif + +#ifdef __cplusplus + result["compiler"]["c++"] = std::to_string(__cplusplus); +#else + result["compiler"]["c++"] = "unknown"; +#endif + return result; + } + + + /////////////////////////// + // JSON value data types // + /////////////////////////// + + /// @name JSON value data types + /// The data types to store a JSON value. These types are derived from + /// the template arguments passed to class @ref basic_json. + /// @{ + +#if defined(JSON_HAS_CPP_14) + // Use transparent comparator if possible, combined with perfect forwarding + // on find() and count() calls prevents unnecessary string construction. + using object_comparator_t = std::less<>; +#else + using object_comparator_t = std::less; +#endif + + /*! + @brief a type for an object + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes JSON objects as follows: + > An object is an unordered collection of zero or more name/value pairs, + > where a name is a string and a value is a string, number, boolean, null, + > object, or array. + + To store objects in C++, a type is defined by the template parameters + described below. + + @tparam ObjectType the container to store objects (e.g., `std::map` or + `std::unordered_map`) + @tparam StringType the type of the keys or names (e.g., `std::string`). + The comparison function `std::less` is used to order elements + inside the container. + @tparam AllocatorType the allocator to use for objects (e.g., + `std::allocator`) + + #### Default type + + With the default values for @a ObjectType (`std::map`), @a StringType + (`std::string`), and @a AllocatorType (`std::allocator`), the default + value for @a object_t is: + + @code {.cpp} + std::map< + std::string, // key_type + basic_json, // value_type + std::less, // key_compare + std::allocator> // allocator_type + > + @endcode + + #### Behavior + + The choice of @a object_t influences the behavior of the JSON class. With + the default type, objects have the following behavior: + + - When all names are unique, objects will be interoperable in the sense + that all software implementations receiving that object will agree on + the name-value mappings. + - When the names within an object are not unique, it is unspecified which + one of the values for a given key will be chosen. For instance, + `{"key": 2, "key": 1}` could be equal to either `{"key": 1}` or + `{"key": 2}`. + - Internally, name/value pairs are stored in lexicographical order of the + names. Objects will also be serialized (see @ref dump) in this order. + For instance, `{"b": 1, "a": 2}` and `{"a": 2, "b": 1}` will be stored + and serialized as `{"a": 2, "b": 1}`. + - When comparing objects, the order of the name/value pairs is irrelevant. + This makes objects interoperable in the sense that they will not be + affected by these differences. For instance, `{"b": 1, "a": 2}` and + `{"a": 2, "b": 1}` will be treated as equal. + + #### Limits + + [RFC 8259](https://tools.ietf.org/html/rfc8259) specifies: + > An implementation may set limits on the maximum depth of nesting. + + In this class, the object's limit of nesting is not explicitly constrained. + However, a maximum depth of nesting may be introduced by the compiler or + runtime environment. A theoretical limit can be queried by calling the + @ref max_size function of a JSON object. + + #### Storage + + Objects are stored as pointers in a @ref basic_json type. That is, for any + access to object values, a pointer of type `object_t*` must be + dereferenced. + + @sa see @ref array_t -- type for an array value + + @since version 1.0.0 + + @note The order name/value pairs are added to the object is *not* + preserved by the library. Therefore, iterating an object may return + name/value pairs in a different order than they were originally stored. In + fact, keys will be traversed in alphabetical order as `std::map` with + `std::less` is used by default. Please note this behavior conforms to [RFC + 8259](https://tools.ietf.org/html/rfc8259), because any order implements the + specified "unordered" nature of JSON objects. + */ + using object_t = ObjectType>>; + + /*! + @brief a type for an array + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes JSON arrays as follows: + > An array is an ordered sequence of zero or more values. + + To store objects in C++, a type is defined by the template parameters + explained below. + + @tparam ArrayType container type to store arrays (e.g., `std::vector` or + `std::list`) + @tparam AllocatorType allocator to use for arrays (e.g., `std::allocator`) + + #### Default type + + With the default values for @a ArrayType (`std::vector`) and @a + AllocatorType (`std::allocator`), the default value for @a array_t is: + + @code {.cpp} + std::vector< + basic_json, // value_type + std::allocator // allocator_type + > + @endcode + + #### Limits + + [RFC 8259](https://tools.ietf.org/html/rfc8259) specifies: + > An implementation may set limits on the maximum depth of nesting. + + In this class, the array's limit of nesting is not explicitly constrained. + However, a maximum depth of nesting may be introduced by the compiler or + runtime environment. A theoretical limit can be queried by calling the + @ref max_size function of a JSON array. + + #### Storage + + Arrays are stored as pointers in a @ref basic_json type. That is, for any + access to array values, a pointer of type `array_t*` must be dereferenced. + + @sa see @ref object_t -- type for an object value + + @since version 1.0.0 + */ + using array_t = ArrayType>; + + /*! + @brief a type for a string + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes JSON strings as follows: + > A string is a sequence of zero or more Unicode characters. + + To store objects in C++, a type is defined by the template parameter + described below. Unicode values are split by the JSON class into + byte-sized characters during deserialization. + + @tparam StringType the container to store strings (e.g., `std::string`). + Note this container is used for keys/names in objects, see @ref object_t. + + #### Default type + + With the default values for @a StringType (`std::string`), the default + value for @a string_t is: + + @code {.cpp} + std::string + @endcode + + #### Encoding + + Strings are stored in UTF-8 encoding. Therefore, functions like + `std::string::size()` or `std::string::length()` return the number of + bytes in the string rather than the number of characters or glyphs. + + #### String comparison + + [RFC 8259](https://tools.ietf.org/html/rfc8259) states: + > Software implementations are typically required to test names of object + > members for equality. Implementations that transform the textual + > representation into sequences of Unicode code units and then perform the + > comparison numerically, code unit by code unit, are interoperable in the + > sense that implementations will agree in all cases on equality or + > inequality of two strings. For example, implementations that compare + > strings with escaped characters unconverted may incorrectly find that + > `"a\\b"` and `"a\u005Cb"` are not equal. + + This implementation is interoperable as it does compare strings code unit + by code unit. + + #### Storage + + String values are stored as pointers in a @ref basic_json type. That is, + for any access to string values, a pointer of type `string_t*` must be + dereferenced. + + @since version 1.0.0 + */ + using string_t = StringType; + + /*! + @brief a type for a boolean + + [RFC 8259](https://tools.ietf.org/html/rfc8259) implicitly describes a boolean as a + type which differentiates the two literals `true` and `false`. + + To store objects in C++, a type is defined by the template parameter @a + BooleanType which chooses the type to use. + + #### Default type + + With the default values for @a BooleanType (`bool`), the default value for + @a boolean_t is: + + @code {.cpp} + bool + @endcode + + #### Storage + + Boolean values are stored directly inside a @ref basic_json type. + + @since version 1.0.0 + */ + using boolean_t = BooleanType; + + /*! + @brief a type for a number (integer) + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes numbers as follows: + > The representation of numbers is similar to that used in most + > programming languages. A number is represented in base 10 using decimal + > digits. It contains an integer component that may be prefixed with an + > optional minus sign, which may be followed by a fraction part and/or an + > exponent part. Leading zeros are not allowed. (...) Numeric values that + > cannot be represented in the grammar below (such as Infinity and NaN) + > are not permitted. + + This description includes both integer and floating-point numbers. + However, C++ allows more precise storage if it is known whether the number + is a signed integer, an unsigned integer or a floating-point number. + Therefore, three different types, @ref number_integer_t, @ref + number_unsigned_t and @ref number_float_t are used. + + To store integer numbers in C++, a type is defined by the template + parameter @a NumberIntegerType which chooses the type to use. + + #### Default type + + With the default values for @a NumberIntegerType (`int64_t`), the default + value for @a number_integer_t is: + + @code {.cpp} + int64_t + @endcode + + #### Default behavior + + - The restrictions about leading zeros is not enforced in C++. Instead, + leading zeros in integer literals lead to an interpretation as octal + number. Internally, the value will be stored as decimal number. For + instance, the C++ integer literal `010` will be serialized to `8`. + During deserialization, leading zeros yield an error. + - Not-a-number (NaN) values will be serialized to `null`. + + #### Limits + + [RFC 8259](https://tools.ietf.org/html/rfc8259) specifies: + > An implementation may set limits on the range and precision of numbers. + + When the default type is used, the maximal integer number that can be + stored is `9223372036854775807` (INT64_MAX) and the minimal integer number + that can be stored is `-9223372036854775808` (INT64_MIN). Integer numbers + that are out of range will yield over/underflow when used in a + constructor. During deserialization, too large or small integer numbers + will be automatically be stored as @ref number_unsigned_t or @ref + number_float_t. + + [RFC 8259](https://tools.ietf.org/html/rfc8259) further states: + > Note that when such software is used, numbers that are integers and are + > in the range \f$[-2^{53}+1, 2^{53}-1]\f$ are interoperable in the sense + > that implementations will agree exactly on their numeric values. + + As this range is a subrange of the exactly supported range [INT64_MIN, + INT64_MAX], this class's integer type is interoperable. + + #### Storage + + Integer number values are stored directly inside a @ref basic_json type. + + @sa see @ref number_float_t -- type for number values (floating-point) + + @sa see @ref number_unsigned_t -- type for number values (unsigned integer) + + @since version 1.0.0 + */ + using number_integer_t = NumberIntegerType; + + /*! + @brief a type for a number (unsigned) + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes numbers as follows: + > The representation of numbers is similar to that used in most + > programming languages. A number is represented in base 10 using decimal + > digits. It contains an integer component that may be prefixed with an + > optional minus sign, which may be followed by a fraction part and/or an + > exponent part. Leading zeros are not allowed. (...) Numeric values that + > cannot be represented in the grammar below (such as Infinity and NaN) + > are not permitted. + + This description includes both integer and floating-point numbers. + However, C++ allows more precise storage if it is known whether the number + is a signed integer, an unsigned integer or a floating-point number. + Therefore, three different types, @ref number_integer_t, @ref + number_unsigned_t and @ref number_float_t are used. + + To store unsigned integer numbers in C++, a type is defined by the + template parameter @a NumberUnsignedType which chooses the type to use. + + #### Default type + + With the default values for @a NumberUnsignedType (`uint64_t`), the + default value for @a number_unsigned_t is: + + @code {.cpp} + uint64_t + @endcode + + #### Default behavior + + - The restrictions about leading zeros is not enforced in C++. Instead, + leading zeros in integer literals lead to an interpretation as octal + number. Internally, the value will be stored as decimal number. For + instance, the C++ integer literal `010` will be serialized to `8`. + During deserialization, leading zeros yield an error. + - Not-a-number (NaN) values will be serialized to `null`. + + #### Limits + + [RFC 8259](https://tools.ietf.org/html/rfc8259) specifies: + > An implementation may set limits on the range and precision of numbers. + + When the default type is used, the maximal integer number that can be + stored is `18446744073709551615` (UINT64_MAX) and the minimal integer + number that can be stored is `0`. Integer numbers that are out of range + will yield over/underflow when used in a constructor. During + deserialization, too large or small integer numbers will be automatically + be stored as @ref number_integer_t or @ref number_float_t. + + [RFC 8259](https://tools.ietf.org/html/rfc8259) further states: + > Note that when such software is used, numbers that are integers and are + > in the range \f$[-2^{53}+1, 2^{53}-1]\f$ are interoperable in the sense + > that implementations will agree exactly on their numeric values. + + As this range is a subrange (when considered in conjunction with the + number_integer_t type) of the exactly supported range [0, UINT64_MAX], + this class's integer type is interoperable. + + #### Storage + + Integer number values are stored directly inside a @ref basic_json type. + + @sa see @ref number_float_t -- type for number values (floating-point) + @sa see @ref number_integer_t -- type for number values (integer) + + @since version 2.0.0 + */ + using number_unsigned_t = NumberUnsignedType; + + /*! + @brief a type for a number (floating-point) + + [RFC 8259](https://tools.ietf.org/html/rfc8259) describes numbers as follows: + > The representation of numbers is similar to that used in most + > programming languages. A number is represented in base 10 using decimal + > digits. It contains an integer component that may be prefixed with an + > optional minus sign, which may be followed by a fraction part and/or an + > exponent part. Leading zeros are not allowed. (...) Numeric values that + > cannot be represented in the grammar below (such as Infinity and NaN) + > are not permitted. + + This description includes both integer and floating-point numbers. + However, C++ allows more precise storage if it is known whether the number + is a signed integer, an unsigned integer or a floating-point number. + Therefore, three different types, @ref number_integer_t, @ref + number_unsigned_t and @ref number_float_t are used. + + To store floating-point numbers in C++, a type is defined by the template + parameter @a NumberFloatType which chooses the type to use. + + #### Default type + + With the default values for @a NumberFloatType (`double`), the default + value for @a number_float_t is: + + @code {.cpp} + double + @endcode + + #### Default behavior + + - The restrictions about leading zeros is not enforced in C++. Instead, + leading zeros in floating-point literals will be ignored. Internally, + the value will be stored as decimal number. For instance, the C++ + floating-point literal `01.2` will be serialized to `1.2`. During + deserialization, leading zeros yield an error. + - Not-a-number (NaN) values will be serialized to `null`. + + #### Limits + + [RFC 8259](https://tools.ietf.org/html/rfc8259) states: + > This specification allows implementations to set limits on the range and + > precision of numbers accepted. Since software that implements IEEE + > 754-2008 binary64 (double precision) numbers is generally available and + > widely used, good interoperability can be achieved by implementations + > that expect no more precision or range than these provide, in the sense + > that implementations will approximate JSON numbers within the expected + > precision. + + This implementation does exactly follow this approach, as it uses double + precision floating-point numbers. Note values smaller than + `-1.79769313486232e+308` and values greater than `1.79769313486232e+308` + will be stored as NaN internally and be serialized to `null`. + + #### Storage + + Floating-point number values are stored directly inside a @ref basic_json + type. + + @sa see @ref number_integer_t -- type for number values (integer) + + @sa see @ref number_unsigned_t -- type for number values (unsigned integer) + + @since version 1.0.0 + */ + using number_float_t = NumberFloatType; + + /*! + @brief a type for a packed binary type + + This type is a type designed to carry binary data that appears in various + serialized formats, such as CBOR's Major Type 2, MessagePack's bin, and + BSON's generic binary subtype. This type is NOT a part of standard JSON and + exists solely for compatibility with these binary types. As such, it is + simply defined as an ordered sequence of zero or more byte values. + + Additionally, as an implementation detail, the subtype of the binary data is + carried around as a `std::uint8_t`, which is compatible with both of the + binary data formats that use binary subtyping, (though the specific + numbering is incompatible with each other, and it is up to the user to + translate between them). + + [CBOR's RFC 7049](https://tools.ietf.org/html/rfc7049) describes this type + as: + > Major type 2: a byte string. The string's length in bytes is represented + > following the rules for positive integers (major type 0). + + [MessagePack's documentation on the bin type + family](https://github.com/msgpack/msgpack/blob/master/spec.md#bin-format-family) + describes this type as: + > Bin format family stores an byte array in 2, 3, or 5 bytes of extra bytes + > in addition to the size of the byte array. + + [BSON's specifications](http://bsonspec.org/spec.html) describe several + binary types; however, this type is intended to represent the generic binary + type which has the description: + > Generic binary subtype - This is the most commonly used binary subtype and + > should be the 'default' for drivers and tools. + + None of these impose any limitations on the internal representation other + than the basic unit of storage be some type of array whose parts are + decomposable into bytes. + + The default representation of this binary format is a + `std::vector`, which is a very common way to represent a byte + array in modern C++. + + #### Default type + + The default values for @a BinaryType is `std::vector` + + #### Storage + + Binary Arrays are stored as pointers in a @ref basic_json type. That is, + for any access to array values, a pointer of the type `binary_t*` must be + dereferenced. + + #### Notes on subtypes + + - CBOR + - Binary values are represented as byte strings. Subtypes are serialized + as tagged values. + - MessagePack + - If a subtype is given and the binary array contains exactly 1, 2, 4, 8, + or 16 elements, the fixext family (fixext1, fixext2, fixext4, fixext8) + is used. For other sizes, the ext family (ext8, ext16, ext32) is used. + The subtype is then added as singed 8-bit integer. + - If no subtype is given, the bin family (bin8, bin16, bin32) is used. + - BSON + - If a subtype is given, it is used and added as unsigned 8-bit integer. + - If no subtype is given, the generic binary subtype 0x00 is used. + + @sa see @ref binary -- create a binary array + + @since version 3.8.0 + */ + using binary_t = nlohmann::byte_container_with_subtype; + /// @} + + private: + + /// helper for exception-safe object creation + template + JSON_HEDLEY_RETURNS_NON_NULL + static T* create(Args&& ... args) + { + AllocatorType alloc; + using AllocatorTraits = std::allocator_traits>; + + auto deleter = [&](T * obj) + { + AllocatorTraits::deallocate(alloc, obj, 1); + }; + std::unique_ptr obj(AllocatorTraits::allocate(alloc, 1), deleter); + AllocatorTraits::construct(alloc, obj.get(), std::forward(args)...); + JSON_ASSERT(obj != nullptr); + return obj.release(); + } + + //////////////////////// + // JSON value storage // + //////////////////////// + + JSON_PRIVATE_UNLESS_TESTED: + /*! + @brief a JSON value + + The actual storage for a JSON value of the @ref basic_json class. This + union combines the different storage types for the JSON value types + defined in @ref value_t. + + JSON type | value_t type | used type + --------- | --------------- | ------------------------ + object | object | pointer to @ref object_t + array | array | pointer to @ref array_t + string | string | pointer to @ref string_t + boolean | boolean | @ref boolean_t + number | number_integer | @ref number_integer_t + number | number_unsigned | @ref number_unsigned_t + number | number_float | @ref number_float_t + binary | binary | pointer to @ref binary_t + null | null | *no value is stored* + + @note Variable-length types (objects, arrays, and strings) are stored as + pointers. The size of the union should not exceed 64 bits if the default + value types are used. + + @since version 1.0.0 + */ + union json_value + { + /// object (stored with pointer to save storage) + object_t* object; + /// array (stored with pointer to save storage) + array_t* array; + /// string (stored with pointer to save storage) + string_t* string; + /// binary (stored with pointer to save storage) + binary_t* binary; + /// boolean + boolean_t boolean; + /// number (integer) + number_integer_t number_integer; + /// number (unsigned integer) + number_unsigned_t number_unsigned; + /// number (floating-point) + number_float_t number_float; + + /// default constructor (for null values) + json_value() = default; + /// constructor for booleans + json_value(boolean_t v) noexcept : boolean(v) {} + /// constructor for numbers (integer) + json_value(number_integer_t v) noexcept : number_integer(v) {} + /// constructor for numbers (unsigned) + json_value(number_unsigned_t v) noexcept : number_unsigned(v) {} + /// constructor for numbers (floating-point) + json_value(number_float_t v) noexcept : number_float(v) {} + /// constructor for empty values of a given type + json_value(value_t t) + { + switch (t) + { + case value_t::object: + { + object = create(); + break; + } + + case value_t::array: + { + array = create(); + break; + } + + case value_t::string: + { + string = create(""); + break; + } + + case value_t::binary: + { + binary = create(); + break; + } + + case value_t::boolean: + { + boolean = boolean_t(false); + break; + } + + case value_t::number_integer: + { + number_integer = number_integer_t(0); + break; + } + + case value_t::number_unsigned: + { + number_unsigned = number_unsigned_t(0); + break; + } + + case value_t::number_float: + { + number_float = number_float_t(0.0); + break; + } + + case value_t::null: + { + object = nullptr; // silence warning, see #821 + break; + } + + case value_t::discarded: + default: + { + object = nullptr; // silence warning, see #821 + if (JSON_HEDLEY_UNLIKELY(t == value_t::null)) + { + JSON_THROW(other_error::create(500, "961c151d2e87f2686a955a9be24d316f1362bf21 3.10.4", basic_json())); // LCOV_EXCL_LINE + } + break; + } + } + } + + /// constructor for strings + json_value(const string_t& value) + { + string = create(value); + } + + /// constructor for rvalue strings + json_value(string_t&& value) + { + string = create(std::move(value)); + } + + /// constructor for objects + json_value(const object_t& value) + { + object = create(value); + } + + /// constructor for rvalue objects + json_value(object_t&& value) + { + object = create(std::move(value)); + } + + /// constructor for arrays + json_value(const array_t& value) + { + array = create(value); + } + + /// constructor for rvalue arrays + json_value(array_t&& value) + { + array = create(std::move(value)); + } + + /// constructor for binary arrays + json_value(const typename binary_t::container_type& value) + { + binary = create(value); + } + + /// constructor for rvalue binary arrays + json_value(typename binary_t::container_type&& value) + { + binary = create(std::move(value)); + } + + /// constructor for binary arrays (internal type) + json_value(const binary_t& value) + { + binary = create(value); + } + + /// constructor for rvalue binary arrays (internal type) + json_value(binary_t&& value) + { + binary = create(std::move(value)); + } + + void destroy(value_t t) + { + if (t == value_t::array || t == value_t::object) + { + // flatten the current json_value to a heap-allocated stack + std::vector stack; + + // move the top-level items to stack + if (t == value_t::array) + { + stack.reserve(array->size()); + std::move(array->begin(), array->end(), std::back_inserter(stack)); + } + else + { + stack.reserve(object->size()); + for (auto&& it : *object) + { + stack.push_back(std::move(it.second)); + } + } + + while (!stack.empty()) + { + // move the last item to local variable to be processed + basic_json current_item(std::move(stack.back())); + stack.pop_back(); + + // if current_item is array/object, move + // its children to the stack to be processed later + if (current_item.is_array()) + { + std::move(current_item.m_value.array->begin(), current_item.m_value.array->end(), std::back_inserter(stack)); + + current_item.m_value.array->clear(); + } + else if (current_item.is_object()) + { + for (auto&& it : *current_item.m_value.object) + { + stack.push_back(std::move(it.second)); + } + + current_item.m_value.object->clear(); + } + + // it's now safe that current_item get destructed + // since it doesn't have any children + } + } + + switch (t) + { + case value_t::object: + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, object); + std::allocator_traits::deallocate(alloc, object, 1); + break; + } + + case value_t::array: + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, array); + std::allocator_traits::deallocate(alloc, array, 1); + break; + } + + case value_t::string: + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, string); + std::allocator_traits::deallocate(alloc, string, 1); + break; + } + + case value_t::binary: + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, binary); + std::allocator_traits::deallocate(alloc, binary, 1); + break; + } + + case value_t::null: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::discarded: + default: + { + break; + } + } + } + }; + + private: + /*! + @brief checks the class invariants + + This function asserts the class invariants. It needs to be called at the + end of every constructor to make sure that created objects respect the + invariant. Furthermore, it has to be called each time the type of a JSON + value is changed, because the invariant expresses a relationship between + @a m_type and @a m_value. + + Furthermore, the parent relation is checked for arrays and objects: If + @a check_parents true and the value is an array or object, then the + container's elements must have the current value as parent. + + @param[in] check_parents whether the parent relation should be checked. + The value is true by default and should only be set to false + during destruction of objects when the invariant does not + need to hold. + */ + void assert_invariant(bool check_parents = true) const noexcept + { + JSON_ASSERT(m_type != value_t::object || m_value.object != nullptr); + JSON_ASSERT(m_type != value_t::array || m_value.array != nullptr); + JSON_ASSERT(m_type != value_t::string || m_value.string != nullptr); + JSON_ASSERT(m_type != value_t::binary || m_value.binary != nullptr); + +#if JSON_DIAGNOSTICS + JSON_TRY + { + // cppcheck-suppress assertWithSideEffect + JSON_ASSERT(!check_parents || !is_structured() || std::all_of(begin(), end(), [this](const basic_json & j) + { + return j.m_parent == this; + })); + } + JSON_CATCH(...) {} // LCOV_EXCL_LINE +#endif + static_cast(check_parents); + } + + void set_parents() + { +#if JSON_DIAGNOSTICS + switch (m_type) + { + case value_t::array: + { + for (auto& element : *m_value.array) + { + element.m_parent = this; + } + break; + } + + case value_t::object: + { + for (auto& element : *m_value.object) + { + element.second.m_parent = this; + } + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + break; + } +#endif + } + + iterator set_parents(iterator it, typename iterator::difference_type count) + { +#if JSON_DIAGNOSTICS + for (typename iterator::difference_type i = 0; i < count; ++i) + { + (it + i)->m_parent = this; + } +#else + static_cast(count); +#endif + return it; + } + + reference set_parent(reference j, std::size_t old_capacity = std::size_t(-1)) + { +#if JSON_DIAGNOSTICS + if (old_capacity != std::size_t(-1)) + { + // see https://github.com/nlohmann/json/issues/2838 + JSON_ASSERT(type() == value_t::array); + if (JSON_HEDLEY_UNLIKELY(m_value.array->capacity() != old_capacity)) + { + // capacity has changed: update all parents + set_parents(); + return j; + } + } + + // ordered_json uses a vector internally, so pointers could have + // been invalidated; see https://github.com/nlohmann/json/issues/2962 +#ifdef JSON_HEDLEY_MSVC_VERSION +#pragma warning(push ) +#pragma warning(disable : 4127) // ignore warning to replace if with if constexpr +#endif + if (detail::is_ordered_map::value) + { + set_parents(); + return j; + } +#ifdef JSON_HEDLEY_MSVC_VERSION +#pragma warning( pop ) +#endif + + j.m_parent = this; +#else + static_cast(j); + static_cast(old_capacity); +#endif + return j; + } + + public: + ////////////////////////// + // JSON parser callback // + ////////////////////////// + + /*! + @brief parser event types + + The parser callback distinguishes the following events: + - `object_start`: the parser read `{` and started to process a JSON object + - `key`: the parser read a key of a value in an object + - `object_end`: the parser read `}` and finished processing a JSON object + - `array_start`: the parser read `[` and started to process a JSON array + - `array_end`: the parser read `]` and finished processing a JSON array + - `value`: the parser finished reading a JSON value + + @image html callback_events.png "Example when certain parse events are triggered" + + @sa see @ref parser_callback_t for more information and examples + */ + using parse_event_t = detail::parse_event_t; + + /*! + @brief per-element parser callback type + + With a parser callback function, the result of parsing a JSON text can be + influenced. When passed to @ref parse, it is called on certain events + (passed as @ref parse_event_t via parameter @a event) with a set recursion + depth @a depth and context JSON value @a parsed. The return value of the + callback function is a boolean indicating whether the element that emitted + the callback shall be kept or not. + + We distinguish six scenarios (determined by the event type) in which the + callback function can be called. The following table describes the values + of the parameters @a depth, @a event, and @a parsed. + + parameter @a event | description | parameter @a depth | parameter @a parsed + ------------------ | ----------- | ------------------ | ------------------- + parse_event_t::object_start | the parser read `{` and started to process a JSON object | depth of the parent of the JSON object | a JSON value with type discarded + parse_event_t::key | the parser read a key of a value in an object | depth of the currently parsed JSON object | a JSON string containing the key + parse_event_t::object_end | the parser read `}` and finished processing a JSON object | depth of the parent of the JSON object | the parsed JSON object + parse_event_t::array_start | the parser read `[` and started to process a JSON array | depth of the parent of the JSON array | a JSON value with type discarded + parse_event_t::array_end | the parser read `]` and finished processing a JSON array | depth of the parent of the JSON array | the parsed JSON array + parse_event_t::value | the parser finished reading a JSON value | depth of the value | the parsed JSON value + + @image html callback_events.png "Example when certain parse events are triggered" + + Discarding a value (i.e., returning `false`) has different effects + depending on the context in which function was called: + + - Discarded values in structured types are skipped. That is, the parser + will behave as if the discarded value was never read. + - In case a value outside a structured type is skipped, it is replaced + with `null`. This case happens if the top-level element is skipped. + + @param[in] depth the depth of the recursion during parsing + + @param[in] event an event of type parse_event_t indicating the context in + the callback function has been called + + @param[in,out] parsed the current intermediate parse result; note that + writing to this value has no effect for parse_event_t::key events + + @return Whether the JSON value which called the function during parsing + should be kept (`true`) or not (`false`). In the latter case, it is either + skipped completely or replaced by an empty discarded object. + + @sa see @ref parse for examples + + @since version 1.0.0 + */ + using parser_callback_t = detail::parser_callback_t; + + ////////////////// + // constructors // + ////////////////// + + /// @name constructors and destructors + /// Constructors of class @ref basic_json, copy/move constructor, copy + /// assignment, static functions creating objects, and the destructor. + /// @{ + + /*! + @brief create an empty value with a given type + + Create an empty JSON value with a given type. The value will be default + initialized with an empty value which depends on the type: + + Value type | initial value + ----------- | ------------- + null | `null` + boolean | `false` + string | `""` + number | `0` + object | `{}` + array | `[]` + binary | empty array + + @param[in] v the type of the value to create + + @complexity Constant. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The following code shows the constructor for different @ref + value_t values,basic_json__value_t} + + @sa see @ref clear() -- restores the postcondition of this constructor + + @since version 1.0.0 + */ + basic_json(const value_t v) + : m_type(v), m_value(v) + { + assert_invariant(); + } + + /*! + @brief create a null object + + Create a `null` JSON value. It either takes a null pointer as parameter + (explicitly creating `null`) or no parameter (implicitly creating `null`). + The passed null pointer itself is not read -- it is only used to choose + the right constructor. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this constructor never throws + exceptions. + + @liveexample{The following code shows the constructor with and without a + null pointer parameter.,basic_json__nullptr_t} + + @since version 1.0.0 + */ + basic_json(std::nullptr_t = nullptr) noexcept + : basic_json(value_t::null) + { + assert_invariant(); + } + + /*! + @brief create a JSON value + + This is a "catch all" constructor for all compatible JSON types; that is, + types for which a `to_json()` method exists. The constructor forwards the + parameter @a val to that method (to `json_serializer::to_json` method + with `U = uncvref_t`, to be exact). + + Template type @a CompatibleType includes, but is not limited to, the + following types: + - **arrays**: @ref array_t and all kinds of compatible containers such as + `std::vector`, `std::deque`, `std::list`, `std::forward_list`, + `std::array`, `std::valarray`, `std::set`, `std::unordered_set`, + `std::multiset`, and `std::unordered_multiset` with a `value_type` from + which a @ref basic_json value can be constructed. + - **objects**: @ref object_t and all kinds of compatible associative + containers such as `std::map`, `std::unordered_map`, `std::multimap`, + and `std::unordered_multimap` with a `key_type` compatible to + @ref string_t and a `value_type` from which a @ref basic_json value can + be constructed. + - **strings**: @ref string_t, string literals, and all compatible string + containers can be used. + - **numbers**: @ref number_integer_t, @ref number_unsigned_t, + @ref number_float_t, and all convertible number types such as `int`, + `size_t`, `int64_t`, `float` or `double` can be used. + - **boolean**: @ref boolean_t / `bool` can be used. + - **binary**: @ref binary_t / `std::vector` may be used, + unfortunately because string literals cannot be distinguished from binary + character arrays by the C++ type system, all types compatible with `const + char*` will be directed to the string constructor instead. This is both + for backwards compatibility, and due to the fact that a binary type is not + a standard JSON type. + + See the examples below. + + @tparam CompatibleType a type such that: + - @a CompatibleType is not derived from `std::istream`, + - @a CompatibleType is not @ref basic_json (to avoid hijacking copy/move + constructors), + - @a CompatibleType is not a different @ref basic_json type (i.e. with different template arguments) + - @a CompatibleType is not a @ref basic_json nested type (e.g., + @ref json_pointer, @ref iterator, etc ...) + - `json_serializer` has a `to_json(basic_json_t&, CompatibleType&&)` method + + @tparam U = `uncvref_t` + + @param[in] val the value to be forwarded to the respective constructor + + @complexity Usually linear in the size of the passed @a val, also + depending on the implementation of the called `to_json()` + method. + + @exceptionsafety Depends on the called constructor. For types directly + supported by the library (i.e., all types for which no `to_json()` function + was provided), strong guarantee holds: if an exception is thrown, there are + no changes to any JSON value. + + @liveexample{The following code shows the constructor with several + compatible types.,basic_json__CompatibleType} + + @since version 2.1.0 + */ + template < typename CompatibleType, + typename U = detail::uncvref_t, + detail::enable_if_t < + !detail::is_basic_json::value && detail::is_compatible_type::value, int > = 0 > + basic_json(CompatibleType && val) noexcept(noexcept( // NOLINT(bugprone-forwarding-reference-overload,bugprone-exception-escape) + JSONSerializer::to_json(std::declval(), + std::forward(val)))) + { + JSONSerializer::to_json(*this, std::forward(val)); + set_parents(); + assert_invariant(); + } + + /*! + @brief create a JSON value from an existing one + + This is a constructor for existing @ref basic_json types. + It does not hijack copy/move constructors, since the parameter has different + template arguments than the current ones. + + The constructor tries to convert the internal @ref m_value of the parameter. + + @tparam BasicJsonType a type such that: + - @a BasicJsonType is a @ref basic_json type. + - @a BasicJsonType has different template arguments than @ref basic_json_t. + + @param[in] val the @ref basic_json value to be converted. + + @complexity Usually linear in the size of the passed @a val, also + depending on the implementation of the called `to_json()` + method. + + @exceptionsafety Depends on the called constructor. For types directly + supported by the library (i.e., all types for which no `to_json()` function + was provided), strong guarantee holds: if an exception is thrown, there are + no changes to any JSON value. + + @since version 3.2.0 + */ + template < typename BasicJsonType, + detail::enable_if_t < + detail::is_basic_json::value&& !std::is_same::value, int > = 0 > + basic_json(const BasicJsonType& val) + { + using other_boolean_t = typename BasicJsonType::boolean_t; + using other_number_float_t = typename BasicJsonType::number_float_t; + using other_number_integer_t = typename BasicJsonType::number_integer_t; + using other_number_unsigned_t = typename BasicJsonType::number_unsigned_t; + using other_string_t = typename BasicJsonType::string_t; + using other_object_t = typename BasicJsonType::object_t; + using other_array_t = typename BasicJsonType::array_t; + using other_binary_t = typename BasicJsonType::binary_t; + + switch (val.type()) + { + case value_t::boolean: + JSONSerializer::to_json(*this, val.template get()); + break; + case value_t::number_float: + JSONSerializer::to_json(*this, val.template get()); + break; + case value_t::number_integer: + JSONSerializer::to_json(*this, val.template get()); + break; + case value_t::number_unsigned: + JSONSerializer::to_json(*this, val.template get()); + break; + case value_t::string: + JSONSerializer::to_json(*this, val.template get_ref()); + break; + case value_t::object: + JSONSerializer::to_json(*this, val.template get_ref()); + break; + case value_t::array: + JSONSerializer::to_json(*this, val.template get_ref()); + break; + case value_t::binary: + JSONSerializer::to_json(*this, val.template get_ref()); + break; + case value_t::null: + *this = nullptr; + break; + case value_t::discarded: + m_type = value_t::discarded; + break; + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + set_parents(); + assert_invariant(); + } + + /*! + @brief create a container (array or object) from an initializer list + + Creates a JSON value of type array or object from the passed initializer + list @a init. In case @a type_deduction is `true` (default), the type of + the JSON value to be created is deducted from the initializer list @a init + according to the following rules: + + 1. If the list is empty, an empty JSON object value `{}` is created. + 2. If the list consists of pairs whose first element is a string, a JSON + object value is created where the first elements of the pairs are + treated as keys and the second elements are as values. + 3. In all other cases, an array is created. + + The rules aim to create the best fit between a C++ initializer list and + JSON values. The rationale is as follows: + + 1. The empty initializer list is written as `{}` which is exactly an empty + JSON object. + 2. C++ has no way of describing mapped types other than to list a list of + pairs. As JSON requires that keys must be of type string, rule 2 is the + weakest constraint one can pose on initializer lists to interpret them + as an object. + 3. In all other cases, the initializer list could not be interpreted as + JSON object type, so interpreting it as JSON array type is safe. + + With the rules described above, the following JSON values cannot be + expressed by an initializer list: + + - the empty array (`[]`): use @ref array(initializer_list_t) + with an empty initializer list in this case + - arrays whose elements satisfy rule 2: use @ref + array(initializer_list_t) with the same initializer list + in this case + + @note When used without parentheses around an empty initializer list, @ref + basic_json() is called instead of this function, yielding the JSON null + value. + + @param[in] init initializer list with JSON values + + @param[in] type_deduction internal parameter; when set to `true`, the type + of the JSON value is deducted from the initializer list @a init; when set + to `false`, the type provided via @a manual_type is forced. This mode is + used by the functions @ref array(initializer_list_t) and + @ref object(initializer_list_t). + + @param[in] manual_type internal parameter; when @a type_deduction is set + to `false`, the created JSON value will use the provided type (only @ref + value_t::array and @ref value_t::object are valid); when @a type_deduction + is set to `true`, this parameter has no effect + + @throw type_error.301 if @a type_deduction is `false`, @a manual_type is + `value_t::object`, but @a init contains an element which is not a pair + whose first element is a string. In this case, the constructor could not + create an object. If @a type_deduction would have be `true`, an array + would have been created. See @ref object(initializer_list_t) + for an example. + + @complexity Linear in the size of the initializer list @a init. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The example below shows how JSON values are created from + initializer lists.,basic_json__list_init_t} + + @sa see @ref array(initializer_list_t) -- create a JSON array + value from an initializer list + @sa see @ref object(initializer_list_t) -- create a JSON object + value from an initializer list + + @since version 1.0.0 + */ + basic_json(initializer_list_t init, + bool type_deduction = true, + value_t manual_type = value_t::array) + { + // check if each element is an array with two elements whose first + // element is a string + bool is_an_object = std::all_of(init.begin(), init.end(), + [](const detail::json_ref& element_ref) + { + return element_ref->is_array() && element_ref->size() == 2 && (*element_ref)[0].is_string(); + }); + + // adjust type if type deduction is not wanted + if (!type_deduction) + { + // if array is wanted, do not create an object though possible + if (manual_type == value_t::array) + { + is_an_object = false; + } + + // if object is wanted but impossible, throw an exception + if (JSON_HEDLEY_UNLIKELY(manual_type == value_t::object && !is_an_object)) + { + JSON_THROW(type_error::create(301, "cannot create object from initializer list", basic_json())); + } + } + + if (is_an_object) + { + // the initializer list is a list of pairs -> create object + m_type = value_t::object; + m_value = value_t::object; + + for (auto& element_ref : init) + { + auto element = element_ref.moved_or_copied(); + m_value.object->emplace( + std::move(*((*element.m_value.array)[0].m_value.string)), + std::move((*element.m_value.array)[1])); + } + } + else + { + // the initializer list describes an array -> create array + m_type = value_t::array; + m_value.array = create(init.begin(), init.end()); + } + + set_parents(); + assert_invariant(); + } + + /*! + @brief explicitly create a binary array (without subtype) + + Creates a JSON binary array value from a given binary container. Binary + values are part of various binary formats, such as CBOR, MessagePack, and + BSON. This constructor is used to create a value for serialization to those + formats. + + @note Note, this function exists because of the difficulty in correctly + specifying the correct template overload in the standard value ctor, as both + JSON arrays and JSON binary arrays are backed with some form of a + `std::vector`. Because JSON binary arrays are a non-standard extension it + was decided that it would be best to prevent automatic initialization of a + binary array type, for backwards compatibility and so it does not happen on + accident. + + @param[in] init container containing bytes to use as binary type + + @return JSON binary array value + + @complexity Linear in the size of @a init. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @since version 3.8.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json binary(const typename binary_t::container_type& init) + { + auto res = basic_json(); + res.m_type = value_t::binary; + res.m_value = init; + return res; + } + + /*! + @brief explicitly create a binary array (with subtype) + + Creates a JSON binary array value from a given binary container. Binary + values are part of various binary formats, such as CBOR, MessagePack, and + BSON. This constructor is used to create a value for serialization to those + formats. + + @note Note, this function exists because of the difficulty in correctly + specifying the correct template overload in the standard value ctor, as both + JSON arrays and JSON binary arrays are backed with some form of a + `std::vector`. Because JSON binary arrays are a non-standard extension it + was decided that it would be best to prevent automatic initialization of a + binary array type, for backwards compatibility and so it does not happen on + accident. + + @param[in] init container containing bytes to use as binary type + @param[in] subtype subtype to use in MessagePack and BSON + + @return JSON binary array value + + @complexity Linear in the size of @a init. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @since version 3.8.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json binary(const typename binary_t::container_type& init, typename binary_t::subtype_type subtype) + { + auto res = basic_json(); + res.m_type = value_t::binary; + res.m_value = binary_t(init, subtype); + return res; + } + + /// @copydoc binary(const typename binary_t::container_type&) + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json binary(typename binary_t::container_type&& init) + { + auto res = basic_json(); + res.m_type = value_t::binary; + res.m_value = std::move(init); + return res; + } + + /// @copydoc binary(const typename binary_t::container_type&, typename binary_t::subtype_type) + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json binary(typename binary_t::container_type&& init, typename binary_t::subtype_type subtype) + { + auto res = basic_json(); + res.m_type = value_t::binary; + res.m_value = binary_t(std::move(init), subtype); + return res; + } + + /*! + @brief explicitly create an array from an initializer list + + Creates a JSON array value from a given initializer list. That is, given a + list of values `a, b, c`, creates the JSON value `[a, b, c]`. If the + initializer list is empty, the empty array `[]` is created. + + @note This function is only needed to express two edge cases that cannot + be realized with the initializer list constructor (@ref + basic_json(initializer_list_t, bool, value_t)). These cases + are: + 1. creating an array whose elements are all pairs whose first element is a + string -- in this case, the initializer list constructor would create an + object, taking the first elements as keys + 2. creating an empty array -- passing the empty initializer list to the + initializer list constructor yields an empty object + + @param[in] init initializer list with JSON values to create an array from + (optional) + + @return JSON array value + + @complexity Linear in the size of @a init. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The following code shows an example for the `array` + function.,array} + + @sa see @ref basic_json(initializer_list_t, bool, value_t) -- + create a JSON value from an initializer list + @sa see @ref object(initializer_list_t) -- create a JSON object + value from an initializer list + + @since version 1.0.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json array(initializer_list_t init = {}) + { + return basic_json(init, false, value_t::array); + } + + /*! + @brief explicitly create an object from an initializer list + + Creates a JSON object value from a given initializer list. The initializer + lists elements must be pairs, and their first elements must be strings. If + the initializer list is empty, the empty object `{}` is created. + + @note This function is only added for symmetry reasons. In contrast to the + related function @ref array(initializer_list_t), there are + no cases which can only be expressed by this function. That is, any + initializer list @a init can also be passed to the initializer list + constructor @ref basic_json(initializer_list_t, bool, value_t). + + @param[in] init initializer list to create an object from (optional) + + @return JSON object value + + @throw type_error.301 if @a init is not a list of pairs whose first + elements are strings. In this case, no object can be created. When such a + value is passed to @ref basic_json(initializer_list_t, bool, value_t), + an array would have been created from the passed initializer list @a init. + See example below. + + @complexity Linear in the size of @a init. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The following code shows an example for the `object` + function.,object} + + @sa see @ref basic_json(initializer_list_t, bool, value_t) -- + create a JSON value from an initializer list + @sa see @ref array(initializer_list_t) -- create a JSON array + value from an initializer list + + @since version 1.0.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json object(initializer_list_t init = {}) + { + return basic_json(init, false, value_t::object); + } + + /*! + @brief construct an array with count copies of given value + + Constructs a JSON array value by creating @a cnt copies of a passed value. + In case @a cnt is `0`, an empty array is created. + + @param[in] cnt the number of JSON copies of @a val to create + @param[in] val the JSON value to copy + + @post `std::distance(begin(),end()) == cnt` holds. + + @complexity Linear in @a cnt. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The following code shows examples for the @ref + basic_json(size_type\, const basic_json&) + constructor.,basic_json__size_type_basic_json} + + @since version 1.0.0 + */ + basic_json(size_type cnt, const basic_json& val) + : m_type(value_t::array) + { + m_value.array = create(cnt, val); + set_parents(); + assert_invariant(); + } + + /*! + @brief construct a JSON container given an iterator range + + Constructs the JSON value with the contents of the range `[first, last)`. + The semantics depends on the different types a JSON value can have: + - In case of a null type, invalid_iterator.206 is thrown. + - In case of other primitive types (number, boolean, or string), @a first + must be `begin()` and @a last must be `end()`. In this case, the value is + copied. Otherwise, invalid_iterator.204 is thrown. + - In case of structured types (array, object), the constructor behaves as + similar versions for `std::vector` or `std::map`; that is, a JSON array + or object is constructed from the values in the range. + + @tparam InputIT an input iterator type (@ref iterator or @ref + const_iterator) + + @param[in] first begin of the range to copy from (included) + @param[in] last end of the range to copy from (excluded) + + @pre Iterators @a first and @a last must be initialized. **This + precondition is enforced with an assertion (see warning).** If + assertions are switched off, a violation of this precondition yields + undefined behavior. + + @pre Range `[first, last)` is valid. Usually, this precondition cannot be + checked efficiently. Only certain edge cases are detected; see the + description of the exceptions below. A violation of this precondition + yields undefined behavior. + + @warning A precondition is enforced with a runtime assertion that will + result in calling `std::abort` if this precondition is not met. + Assertions can be disabled by defining `NDEBUG` at compile time. + See https://en.cppreference.com/w/cpp/error/assert for more + information. + + @throw invalid_iterator.201 if iterators @a first and @a last are not + compatible (i.e., do not belong to the same JSON value). In this case, + the range `[first, last)` is undefined. + @throw invalid_iterator.204 if iterators @a first and @a last belong to a + primitive type (number, boolean, or string), but @a first does not point + to the first element any more. In this case, the range `[first, last)` is + undefined. See example code below. + @throw invalid_iterator.206 if iterators @a first and @a last belong to a + null value. In this case, the range `[first, last)` is undefined. + + @complexity Linear in distance between @a first and @a last. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @liveexample{The example below shows several ways to create JSON values by + specifying a subrange with iterators.,basic_json__InputIt_InputIt} + + @since version 1.0.0 + */ + template < class InputIT, typename std::enable_if < + std::is_same::value || + std::is_same::value, int >::type = 0 > + basic_json(InputIT first, InputIT last) + { + JSON_ASSERT(first.m_object != nullptr); + JSON_ASSERT(last.m_object != nullptr); + + // make sure iterator fits the current value + if (JSON_HEDLEY_UNLIKELY(first.m_object != last.m_object)) + { + JSON_THROW(invalid_iterator::create(201, "iterators are not compatible", basic_json())); + } + + // copy type from first iterator + m_type = first.m_object->m_type; + + // check if iterator range is complete for primitive values + switch (m_type) + { + case value_t::boolean: + case value_t::number_float: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::string: + { + if (JSON_HEDLEY_UNLIKELY(!first.m_it.primitive_iterator.is_begin() + || !last.m_it.primitive_iterator.is_end())) + { + JSON_THROW(invalid_iterator::create(204, "iterators out of range", *first.m_object)); + } + break; + } + + case value_t::null: + case value_t::object: + case value_t::array: + case value_t::binary: + case value_t::discarded: + default: + break; + } + + switch (m_type) + { + case value_t::number_integer: + { + m_value.number_integer = first.m_object->m_value.number_integer; + break; + } + + case value_t::number_unsigned: + { + m_value.number_unsigned = first.m_object->m_value.number_unsigned; + break; + } + + case value_t::number_float: + { + m_value.number_float = first.m_object->m_value.number_float; + break; + } + + case value_t::boolean: + { + m_value.boolean = first.m_object->m_value.boolean; + break; + } + + case value_t::string: + { + m_value = *first.m_object->m_value.string; + break; + } + + case value_t::object: + { + m_value.object = create(first.m_it.object_iterator, + last.m_it.object_iterator); + break; + } + + case value_t::array: + { + m_value.array = create(first.m_it.array_iterator, + last.m_it.array_iterator); + break; + } + + case value_t::binary: + { + m_value = *first.m_object->m_value.binary; + break; + } + + case value_t::null: + case value_t::discarded: + default: + JSON_THROW(invalid_iterator::create(206, "cannot construct with iterators from " + std::string(first.m_object->type_name()), *first.m_object)); + } + + set_parents(); + assert_invariant(); + } + + + /////////////////////////////////////// + // other constructors and destructor // + /////////////////////////////////////// + + template, + std::is_same>::value, int> = 0 > + basic_json(const JsonRef& ref) : basic_json(ref.moved_or_copied()) {} + + /*! + @brief copy constructor + + Creates a copy of a given JSON value. + + @param[in] other the JSON value to copy + + @post `*this == other` + + @complexity Linear in the size of @a other. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes to any JSON value. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is linear. + - As postcondition, it holds: `other == basic_json(other)`. + + @liveexample{The following code shows an example for the copy + constructor.,basic_json__basic_json} + + @since version 1.0.0 + */ + basic_json(const basic_json& other) + : m_type(other.m_type) + { + // check of passed value is valid + other.assert_invariant(); + + switch (m_type) + { + case value_t::object: + { + m_value = *other.m_value.object; + break; + } + + case value_t::array: + { + m_value = *other.m_value.array; + break; + } + + case value_t::string: + { + m_value = *other.m_value.string; + break; + } + + case value_t::boolean: + { + m_value = other.m_value.boolean; + break; + } + + case value_t::number_integer: + { + m_value = other.m_value.number_integer; + break; + } + + case value_t::number_unsigned: + { + m_value = other.m_value.number_unsigned; + break; + } + + case value_t::number_float: + { + m_value = other.m_value.number_float; + break; + } + + case value_t::binary: + { + m_value = *other.m_value.binary; + break; + } + + case value_t::null: + case value_t::discarded: + default: + break; + } + + set_parents(); + assert_invariant(); + } + + /*! + @brief move constructor + + Move constructor. Constructs a JSON value with the contents of the given + value @a other using move semantics. It "steals" the resources from @a + other and leaves it as JSON null value. + + @param[in,out] other value to move to this object + + @post `*this` has the same value as @a other before the call. + @post @a other is a JSON null value. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this constructor never throws + exceptions. + + @requirement This function helps `basic_json` satisfying the + [MoveConstructible](https://en.cppreference.com/w/cpp/named_req/MoveConstructible) + requirements. + + @liveexample{The code below shows the move constructor explicitly called + via std::move.,basic_json__moveconstructor} + + @since version 1.0.0 + */ + basic_json(basic_json&& other) noexcept + : m_type(std::move(other.m_type)), + m_value(std::move(other.m_value)) + { + // check that passed value is valid + other.assert_invariant(false); + + // invalidate payload + other.m_type = value_t::null; + other.m_value = {}; + + set_parents(); + assert_invariant(); + } + + /*! + @brief copy assignment + + Copy assignment operator. Copies a JSON value via the "copy and swap" + strategy: It is expressed in terms of the copy constructor, destructor, + and the `swap()` member function. + + @param[in] other value to copy from + + @complexity Linear. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is linear. + + @liveexample{The code below shows and example for the copy assignment. It + creates a copy of value `a` which is then swapped with `b`. Finally\, the + copy of `a` (which is the null value after the swap) is + destroyed.,basic_json__copyassignment} + + @since version 1.0.0 + */ + basic_json& operator=(basic_json other) noexcept ( + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value&& + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value + ) + { + // check that passed value is valid + other.assert_invariant(); + + using std::swap; + swap(m_type, other.m_type); + swap(m_value, other.m_value); + + set_parents(); + assert_invariant(); + return *this; + } + + /*! + @brief destructor + + Destroys the JSON value and frees all allocated memory. + + @complexity Linear. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is linear. + - All stored elements are destroyed and all memory is freed. + + @since version 1.0.0 + */ + ~basic_json() noexcept + { + assert_invariant(false); + m_value.destroy(m_type); + } + + /// @} + + public: + /////////////////////// + // object inspection // + /////////////////////// + + /// @name object inspection + /// Functions to inspect the type of a JSON value. + /// @{ + + /*! + @brief serialization + + Serialization function for JSON values. The function tries to mimic + Python's `json.dumps()` function, and currently supports its @a indent + and @a ensure_ascii parameters. + + @param[in] indent If indent is nonnegative, then array elements and object + members will be pretty-printed with that indent level. An indent level of + `0` will only insert newlines. `-1` (the default) selects the most compact + representation. + @param[in] indent_char The character to use for indentation if @a indent is + greater than `0`. The default is ` ` (space). + @param[in] ensure_ascii If @a ensure_ascii is true, all non-ASCII characters + in the output are escaped with `\uXXXX` sequences, and the result consists + of ASCII characters only. + @param[in] error_handler how to react on decoding errors; there are three + possible values: `strict` (throws and exception in case a decoding error + occurs; default), `replace` (replace invalid UTF-8 sequences with U+FFFD), + and `ignore` (ignore invalid UTF-8 sequences during serialization; all + bytes are copied to the output unchanged). + + @return string containing the serialization of the JSON value + + @throw type_error.316 if a string stored inside the JSON value is not + UTF-8 encoded and @a error_handler is set to strict + + @note Binary values are serialized as object containing two keys: + - "bytes": an array of bytes as integers + - "subtype": the subtype as integer or "null" if the binary has no subtype + + @complexity Linear. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @liveexample{The following example shows the effect of different @a indent\, + @a indent_char\, and @a ensure_ascii parameters to the result of the + serialization.,dump} + + @see https://docs.python.org/2/library/json.html#json.dump + + @since version 1.0.0; indentation character @a indent_char, option + @a ensure_ascii and exceptions added in version 3.0.0; error + handlers added in version 3.4.0; serialization of binary values added + in version 3.8.0. + */ + string_t dump(const int indent = -1, + const char indent_char = ' ', + const bool ensure_ascii = false, + const error_handler_t error_handler = error_handler_t::strict) const + { + string_t result; + serializer s(detail::output_adapter(result), indent_char, error_handler); + + if (indent >= 0) + { + s.dump(*this, true, ensure_ascii, static_cast(indent)); + } + else + { + s.dump(*this, false, ensure_ascii, 0); + } + + return result; + } + + /*! + @brief return the type of the JSON value (explicit) + + Return the type of the JSON value as a value from the @ref value_t + enumeration. + + @return the type of the JSON value + Value type | return value + ------------------------- | ------------------------- + null | value_t::null + boolean | value_t::boolean + string | value_t::string + number (integer) | value_t::number_integer + number (unsigned integer) | value_t::number_unsigned + number (floating-point) | value_t::number_float + object | value_t::object + array | value_t::array + binary | value_t::binary + discarded | value_t::discarded + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `type()` for all JSON + types.,type} + + @sa see @ref operator value_t() -- return the type of the JSON value (implicit) + @sa see @ref type_name() -- return the type as string + + @since version 1.0.0 + */ + constexpr value_t type() const noexcept + { + return m_type; + } + + /*! + @brief return whether type is primitive + + This function returns true if and only if the JSON type is primitive + (string, number, boolean, or null). + + @return `true` if type is primitive (string, number, boolean, or null), + `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_primitive()` for all JSON + types.,is_primitive} + + @sa see @ref is_structured() -- returns whether JSON value is structured + @sa see @ref is_null() -- returns whether JSON value is `null` + @sa see @ref is_string() -- returns whether JSON value is a string + @sa see @ref is_boolean() -- returns whether JSON value is a boolean + @sa see @ref is_number() -- returns whether JSON value is a number + @sa see @ref is_binary() -- returns whether JSON value is a binary array + + @since version 1.0.0 + */ + constexpr bool is_primitive() const noexcept + { + return is_null() || is_string() || is_boolean() || is_number() || is_binary(); + } + + /*! + @brief return whether type is structured + + This function returns true if and only if the JSON type is structured + (array or object). + + @return `true` if type is structured (array or object), `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_structured()` for all JSON + types.,is_structured} + + @sa see @ref is_primitive() -- returns whether value is primitive + @sa see @ref is_array() -- returns whether value is an array + @sa see @ref is_object() -- returns whether value is an object + + @since version 1.0.0 + */ + constexpr bool is_structured() const noexcept + { + return is_array() || is_object(); + } + + /*! + @brief return whether value is null + + This function returns true if and only if the JSON value is null. + + @return `true` if type is null, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_null()` for all JSON + types.,is_null} + + @since version 1.0.0 + */ + constexpr bool is_null() const noexcept + { + return m_type == value_t::null; + } + + /*! + @brief return whether value is a boolean + + This function returns true if and only if the JSON value is a boolean. + + @return `true` if type is boolean, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_boolean()` for all JSON + types.,is_boolean} + + @since version 1.0.0 + */ + constexpr bool is_boolean() const noexcept + { + return m_type == value_t::boolean; + } + + /*! + @brief return whether value is a number + + This function returns true if and only if the JSON value is a number. This + includes both integer (signed and unsigned) and floating-point values. + + @return `true` if type is number (regardless whether integer, unsigned + integer or floating-type), `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_number()` for all JSON + types.,is_number} + + @sa see @ref is_number_integer() -- check if value is an integer or unsigned + integer number + @sa see @ref is_number_unsigned() -- check if value is an unsigned integer + number + @sa see @ref is_number_float() -- check if value is a floating-point number + + @since version 1.0.0 + */ + constexpr bool is_number() const noexcept + { + return is_number_integer() || is_number_float(); + } + + /*! + @brief return whether value is an integer number + + This function returns true if and only if the JSON value is a signed or + unsigned integer number. This excludes floating-point values. + + @return `true` if type is an integer or unsigned integer number, `false` + otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_number_integer()` for all + JSON types.,is_number_integer} + + @sa see @ref is_number() -- check if value is a number + @sa see @ref is_number_unsigned() -- check if value is an unsigned integer + number + @sa see @ref is_number_float() -- check if value is a floating-point number + + @since version 1.0.0 + */ + constexpr bool is_number_integer() const noexcept + { + return m_type == value_t::number_integer || m_type == value_t::number_unsigned; + } + + /*! + @brief return whether value is an unsigned integer number + + This function returns true if and only if the JSON value is an unsigned + integer number. This excludes floating-point and signed integer values. + + @return `true` if type is an unsigned integer number, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_number_unsigned()` for all + JSON types.,is_number_unsigned} + + @sa see @ref is_number() -- check if value is a number + @sa see @ref is_number_integer() -- check if value is an integer or unsigned + integer number + @sa see @ref is_number_float() -- check if value is a floating-point number + + @since version 2.0.0 + */ + constexpr bool is_number_unsigned() const noexcept + { + return m_type == value_t::number_unsigned; + } + + /*! + @brief return whether value is a floating-point number + + This function returns true if and only if the JSON value is a + floating-point number. This excludes signed and unsigned integer values. + + @return `true` if type is a floating-point number, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_number_float()` for all + JSON types.,is_number_float} + + @sa see @ref is_number() -- check if value is number + @sa see @ref is_number_integer() -- check if value is an integer number + @sa see @ref is_number_unsigned() -- check if value is an unsigned integer + number + + @since version 1.0.0 + */ + constexpr bool is_number_float() const noexcept + { + return m_type == value_t::number_float; + } + + /*! + @brief return whether value is an object + + This function returns true if and only if the JSON value is an object. + + @return `true` if type is object, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_object()` for all JSON + types.,is_object} + + @since version 1.0.0 + */ + constexpr bool is_object() const noexcept + { + return m_type == value_t::object; + } + + /*! + @brief return whether value is an array + + This function returns true if and only if the JSON value is an array. + + @return `true` if type is array, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_array()` for all JSON + types.,is_array} + + @since version 1.0.0 + */ + constexpr bool is_array() const noexcept + { + return m_type == value_t::array; + } + + /*! + @brief return whether value is a string + + This function returns true if and only if the JSON value is a string. + + @return `true` if type is string, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_string()` for all JSON + types.,is_string} + + @since version 1.0.0 + */ + constexpr bool is_string() const noexcept + { + return m_type == value_t::string; + } + + /*! + @brief return whether value is a binary array + + This function returns true if and only if the JSON value is a binary array. + + @return `true` if type is binary array, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_binary()` for all JSON + types.,is_binary} + + @since version 3.8.0 + */ + constexpr bool is_binary() const noexcept + { + return m_type == value_t::binary; + } + + /*! + @brief return whether value is discarded + + This function returns true if and only if the JSON value was discarded + during parsing with a callback function (see @ref parser_callback_t). + + @note This function will always be `false` for JSON values after parsing. + That is, discarded values can only occur during parsing, but will be + removed when inside a structured value or replaced by null in other cases. + + @return `true` if type is discarded, `false` otherwise. + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies `is_discarded()` for all JSON + types.,is_discarded} + + @since version 1.0.0 + */ + constexpr bool is_discarded() const noexcept + { + return m_type == value_t::discarded; + } + + /*! + @brief return the type of the JSON value (implicit) + + Implicitly return the type of the JSON value as a value from the @ref + value_t enumeration. + + @return the type of the JSON value + + @complexity Constant. + + @exceptionsafety No-throw guarantee: this member function never throws + exceptions. + + @liveexample{The following code exemplifies the @ref value_t operator for + all JSON types.,operator__value_t} + + @sa see @ref type() -- return the type of the JSON value (explicit) + @sa see @ref type_name() -- return the type as string + + @since version 1.0.0 + */ + constexpr operator value_t() const noexcept + { + return m_type; + } + + /// @} + + private: + ////////////////// + // value access // + ////////////////// + + /// get a boolean (explicit) + boolean_t get_impl(boolean_t* /*unused*/) const + { + if (JSON_HEDLEY_LIKELY(is_boolean())) + { + return m_value.boolean; + } + + JSON_THROW(type_error::create(302, "type must be boolean, but is " + std::string(type_name()), *this)); + } + + /// get a pointer to the value (object) + object_t* get_impl_ptr(object_t* /*unused*/) noexcept + { + return is_object() ? m_value.object : nullptr; + } + + /// get a pointer to the value (object) + constexpr const object_t* get_impl_ptr(const object_t* /*unused*/) const noexcept + { + return is_object() ? m_value.object : nullptr; + } + + /// get a pointer to the value (array) + array_t* get_impl_ptr(array_t* /*unused*/) noexcept + { + return is_array() ? m_value.array : nullptr; + } + + /// get a pointer to the value (array) + constexpr const array_t* get_impl_ptr(const array_t* /*unused*/) const noexcept + { + return is_array() ? m_value.array : nullptr; + } + + /// get a pointer to the value (string) + string_t* get_impl_ptr(string_t* /*unused*/) noexcept + { + return is_string() ? m_value.string : nullptr; + } + + /// get a pointer to the value (string) + constexpr const string_t* get_impl_ptr(const string_t* /*unused*/) const noexcept + { + return is_string() ? m_value.string : nullptr; + } + + /// get a pointer to the value (boolean) + boolean_t* get_impl_ptr(boolean_t* /*unused*/) noexcept + { + return is_boolean() ? &m_value.boolean : nullptr; + } + + /// get a pointer to the value (boolean) + constexpr const boolean_t* get_impl_ptr(const boolean_t* /*unused*/) const noexcept + { + return is_boolean() ? &m_value.boolean : nullptr; + } + + /// get a pointer to the value (integer number) + number_integer_t* get_impl_ptr(number_integer_t* /*unused*/) noexcept + { + return is_number_integer() ? &m_value.number_integer : nullptr; + } + + /// get a pointer to the value (integer number) + constexpr const number_integer_t* get_impl_ptr(const number_integer_t* /*unused*/) const noexcept + { + return is_number_integer() ? &m_value.number_integer : nullptr; + } + + /// get a pointer to the value (unsigned number) + number_unsigned_t* get_impl_ptr(number_unsigned_t* /*unused*/) noexcept + { + return is_number_unsigned() ? &m_value.number_unsigned : nullptr; + } + + /// get a pointer to the value (unsigned number) + constexpr const number_unsigned_t* get_impl_ptr(const number_unsigned_t* /*unused*/) const noexcept + { + return is_number_unsigned() ? &m_value.number_unsigned : nullptr; + } + + /// get a pointer to the value (floating-point number) + number_float_t* get_impl_ptr(number_float_t* /*unused*/) noexcept + { + return is_number_float() ? &m_value.number_float : nullptr; + } + + /// get a pointer to the value (floating-point number) + constexpr const number_float_t* get_impl_ptr(const number_float_t* /*unused*/) const noexcept + { + return is_number_float() ? &m_value.number_float : nullptr; + } + + /// get a pointer to the value (binary) + binary_t* get_impl_ptr(binary_t* /*unused*/) noexcept + { + return is_binary() ? m_value.binary : nullptr; + } + + /// get a pointer to the value (binary) + constexpr const binary_t* get_impl_ptr(const binary_t* /*unused*/) const noexcept + { + return is_binary() ? m_value.binary : nullptr; + } + + /*! + @brief helper function to implement get_ref() + + This function helps to implement get_ref() without code duplication for + const and non-const overloads + + @tparam ThisType will be deduced as `basic_json` or `const basic_json` + + @throw type_error.303 if ReferenceType does not match underlying value + type of the current JSON + */ + template + static ReferenceType get_ref_impl(ThisType& obj) + { + // delegate the call to get_ptr<>() + auto* ptr = obj.template get_ptr::type>(); + + if (JSON_HEDLEY_LIKELY(ptr != nullptr)) + { + return *ptr; + } + + JSON_THROW(type_error::create(303, "incompatible ReferenceType for get_ref, actual type is " + std::string(obj.type_name()), obj)); + } + + public: + /// @name value access + /// Direct access to the stored value of a JSON value. + /// @{ + + /*! + @brief get a pointer value (implicit) + + Implicit pointer access to the internally stored JSON value. No copies are + made. + + @warning Writing data to the pointee of the result yields an undefined + state. + + @tparam PointerType pointer type; must be a pointer to @ref array_t, @ref + object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, + @ref number_unsigned_t, or @ref number_float_t. Enforced by a static + assertion. + + @return pointer to the internally stored JSON value if the requested + pointer type @a PointerType fits to the JSON value; `nullptr` otherwise + + @complexity Constant. + + @liveexample{The example below shows how pointers to internal values of a + JSON value can be requested. Note that no type conversions are made and a + `nullptr` is returned if the value and the requested pointer type does not + match.,get_ptr} + + @since version 1.0.0 + */ + template::value, int>::type = 0> + auto get_ptr() noexcept -> decltype(std::declval().get_impl_ptr(std::declval())) + { + // delegate the call to get_impl_ptr<>() + return get_impl_ptr(static_cast(nullptr)); + } + + /*! + @brief get a pointer value (implicit) + @copydoc get_ptr() + */ + template < typename PointerType, typename std::enable_if < + std::is_pointer::value&& + std::is_const::type>::value, int >::type = 0 > + constexpr auto get_ptr() const noexcept -> decltype(std::declval().get_impl_ptr(std::declval())) + { + // delegate the call to get_impl_ptr<>() const + return get_impl_ptr(static_cast(nullptr)); + } + + private: + /*! + @brief get a value (explicit) + + Explicit type conversion between the JSON value and a compatible value + which is [CopyConstructible](https://en.cppreference.com/w/cpp/named_req/CopyConstructible) + and [DefaultConstructible](https://en.cppreference.com/w/cpp/named_req/DefaultConstructible). + The value is converted by calling the @ref json_serializer + `from_json()` method. + + The function is equivalent to executing + @code {.cpp} + ValueType ret; + JSONSerializer::from_json(*this, ret); + return ret; + @endcode + + This overloads is chosen if: + - @a ValueType is not @ref basic_json, + - @ref json_serializer has a `from_json()` method of the form + `void from_json(const basic_json&, ValueType&)`, and + - @ref json_serializer does not have a `from_json()` method of + the form `ValueType from_json(const basic_json&)` + + @tparam ValueType the returned value type + + @return copy of the JSON value, converted to @a ValueType + + @throw what @ref json_serializer `from_json()` method throws + + @liveexample{The example below shows several conversions from JSON values + to other types. There a few things to note: (1) Floating-point numbers can + be converted to integers\, (2) A JSON array can be converted to a standard + `std::vector`\, (3) A JSON object can be converted to C++ + associative containers such as `std::unordered_map`.,get__ValueType_const} + + @since version 2.1.0 + */ + template < typename ValueType, + detail::enable_if_t < + detail::is_default_constructible::value&& + detail::has_from_json::value, + int > = 0 > + ValueType get_impl(detail::priority_tag<0> /*unused*/) const noexcept(noexcept( + JSONSerializer::from_json(std::declval(), std::declval()))) + { + auto ret = ValueType(); + JSONSerializer::from_json(*this, ret); + return ret; + } + + /*! + @brief get a value (explicit); special case + + Explicit type conversion between the JSON value and a compatible value + which is **not** [CopyConstructible](https://en.cppreference.com/w/cpp/named_req/CopyConstructible) + and **not** [DefaultConstructible](https://en.cppreference.com/w/cpp/named_req/DefaultConstructible). + The value is converted by calling the @ref json_serializer + `from_json()` method. + + The function is equivalent to executing + @code {.cpp} + return JSONSerializer::from_json(*this); + @endcode + + This overloads is chosen if: + - @a ValueType is not @ref basic_json and + - @ref json_serializer has a `from_json()` method of the form + `ValueType from_json(const basic_json&)` + + @note If @ref json_serializer has both overloads of + `from_json()`, this one is chosen. + + @tparam ValueType the returned value type + + @return copy of the JSON value, converted to @a ValueType + + @throw what @ref json_serializer `from_json()` method throws + + @since version 2.1.0 + */ + template < typename ValueType, + detail::enable_if_t < + detail::has_non_default_from_json::value, + int > = 0 > + ValueType get_impl(detail::priority_tag<1> /*unused*/) const noexcept(noexcept( + JSONSerializer::from_json(std::declval()))) + { + return JSONSerializer::from_json(*this); + } + + /*! + @brief get special-case overload + + This overloads converts the current @ref basic_json in a different + @ref basic_json type + + @tparam BasicJsonType == @ref basic_json + + @return a copy of *this, converted into @a BasicJsonType + + @complexity Depending on the implementation of the called `from_json()` + method. + + @since version 3.2.0 + */ + template < typename BasicJsonType, + detail::enable_if_t < + detail::is_basic_json::value, + int > = 0 > + BasicJsonType get_impl(detail::priority_tag<2> /*unused*/) const + { + return *this; + } + + /*! + @brief get special-case overload + + This overloads avoids a lot of template boilerplate, it can be seen as the + identity method + + @tparam BasicJsonType == @ref basic_json + + @return a copy of *this + + @complexity Constant. + + @since version 2.1.0 + */ + template::value, + int> = 0> + basic_json get_impl(detail::priority_tag<3> /*unused*/) const + { + return *this; + } + + /*! + @brief get a pointer value (explicit) + @copydoc get() + */ + template::value, + int> = 0> + constexpr auto get_impl(detail::priority_tag<4> /*unused*/) const noexcept + -> decltype(std::declval().template get_ptr()) + { + // delegate the call to get_ptr + return get_ptr(); + } + + public: + /*! + @brief get a (pointer) value (explicit) + + Performs explicit type conversion between the JSON value and a compatible value if required. + + - If the requested type is a pointer to the internally stored JSON value that pointer is returned. + No copies are made. + + - If the requested type is the current @ref basic_json, or a different @ref basic_json convertible + from the current @ref basic_json. + + - Otherwise the value is converted by calling the @ref json_serializer `from_json()` + method. + + @tparam ValueTypeCV the provided value type + @tparam ValueType the returned value type + + @return copy of the JSON value, converted to @tparam ValueType if necessary + + @throw what @ref json_serializer `from_json()` method throws if conversion is required + + @since version 2.1.0 + */ + template < typename ValueTypeCV, typename ValueType = detail::uncvref_t> +#if defined(JSON_HAS_CPP_14) + constexpr +#endif + auto get() const noexcept( + noexcept(std::declval().template get_impl(detail::priority_tag<4> {}))) + -> decltype(std::declval().template get_impl(detail::priority_tag<4> {})) + { + // we cannot static_assert on ValueTypeCV being non-const, because + // there is support for get(), which is why we + // still need the uncvref + static_assert(!std::is_reference::value, + "get() cannot be used with reference types, you might want to use get_ref()"); + return get_impl(detail::priority_tag<4> {}); + } + + /*! + @brief get a pointer value (explicit) + + Explicit pointer access to the internally stored JSON value. No copies are + made. + + @warning The pointer becomes invalid if the underlying JSON object + changes. + + @tparam PointerType pointer type; must be a pointer to @ref array_t, @ref + object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, + @ref number_unsigned_t, or @ref number_float_t. + + @return pointer to the internally stored JSON value if the requested + pointer type @a PointerType fits to the JSON value; `nullptr` otherwise + + @complexity Constant. + + @liveexample{The example below shows how pointers to internal values of a + JSON value can be requested. Note that no type conversions are made and a + `nullptr` is returned if the value and the requested pointer type does not + match.,get__PointerType} + + @sa see @ref get_ptr() for explicit pointer-member access + + @since version 1.0.0 + */ + template::value, int>::type = 0> + auto get() noexcept -> decltype(std::declval().template get_ptr()) + { + // delegate the call to get_ptr + return get_ptr(); + } + + /*! + @brief get a value (explicit) + + Explicit type conversion between the JSON value and a compatible value. + The value is filled into the input parameter by calling the @ref json_serializer + `from_json()` method. + + The function is equivalent to executing + @code {.cpp} + ValueType v; + JSONSerializer::from_json(*this, v); + @endcode + + This overloads is chosen if: + - @a ValueType is not @ref basic_json, + - @ref json_serializer has a `from_json()` method of the form + `void from_json(const basic_json&, ValueType&)`, and + + @tparam ValueType the input parameter type. + + @return the input parameter, allowing chaining calls. + + @throw what @ref json_serializer `from_json()` method throws + + @liveexample{The example below shows several conversions from JSON values + to other types. There a few things to note: (1) Floating-point numbers can + be converted to integers\, (2) A JSON array can be converted to a standard + `std::vector`\, (3) A JSON object can be converted to C++ + associative containers such as `std::unordered_map`.,get_to} + + @since version 3.3.0 + */ + template < typename ValueType, + detail::enable_if_t < + !detail::is_basic_json::value&& + detail::has_from_json::value, + int > = 0 > + ValueType & get_to(ValueType& v) const noexcept(noexcept( + JSONSerializer::from_json(std::declval(), v))) + { + JSONSerializer::from_json(*this, v); + return v; + } + + // specialization to allow to call get_to with a basic_json value + // see https://github.com/nlohmann/json/issues/2175 + template::value, + int> = 0> + ValueType & get_to(ValueType& v) const + { + v = *this; + return v; + } + + template < + typename T, std::size_t N, + typename Array = T (&)[N], // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) + detail::enable_if_t < + detail::has_from_json::value, int > = 0 > + Array get_to(T (&v)[N]) const // NOLINT(cppcoreguidelines-avoid-c-arrays,hicpp-avoid-c-arrays,modernize-avoid-c-arrays) + noexcept(noexcept(JSONSerializer::from_json( + std::declval(), v))) + { + JSONSerializer::from_json(*this, v); + return v; + } + + /*! + @brief get a reference value (implicit) + + Implicit reference access to the internally stored JSON value. No copies + are made. + + @warning Writing data to the referee of the result yields an undefined + state. + + @tparam ReferenceType reference type; must be a reference to @ref array_t, + @ref object_t, @ref string_t, @ref boolean_t, @ref number_integer_t, or + @ref number_float_t. Enforced by static assertion. + + @return reference to the internally stored JSON value if the requested + reference type @a ReferenceType fits to the JSON value; throws + type_error.303 otherwise + + @throw type_error.303 in case passed type @a ReferenceType is incompatible + with the stored JSON value; see example below + + @complexity Constant. + + @liveexample{The example shows several calls to `get_ref()`.,get_ref} + + @since version 1.1.0 + */ + template::value, int>::type = 0> + ReferenceType get_ref() + { + // delegate call to get_ref_impl + return get_ref_impl(*this); + } + + /*! + @brief get a reference value (implicit) + @copydoc get_ref() + */ + template < typename ReferenceType, typename std::enable_if < + std::is_reference::value&& + std::is_const::type>::value, int >::type = 0 > + ReferenceType get_ref() const + { + // delegate call to get_ref_impl + return get_ref_impl(*this); + } + + /*! + @brief get a value (implicit) + + Implicit type conversion between the JSON value and a compatible value. + The call is realized by calling @ref get() const. + + @tparam ValueType non-pointer type compatible to the JSON value, for + instance `int` for JSON integer numbers, `bool` for JSON booleans, or + `std::vector` types for JSON arrays. The character type of @ref string_t + as well as an initializer list of this type is excluded to avoid + ambiguities as these types implicitly convert to `std::string`. + + @return copy of the JSON value, converted to type @a ValueType + + @throw type_error.302 in case passed type @a ValueType is incompatible + to the JSON value type (e.g., the JSON value is of type boolean, but a + string is requested); see example below + + @complexity Linear in the size of the JSON value. + + @liveexample{The example below shows several conversions from JSON values + to other types. There a few things to note: (1) Floating-point numbers can + be converted to integers\, (2) A JSON array can be converted to a standard + `std::vector`\, (3) A JSON object can be converted to C++ + associative containers such as `std::unordered_map`.,operator__ValueType} + + @since version 1.0.0 + */ + template < typename ValueType, typename std::enable_if < + detail::conjunction < + detail::negation>, + detail::negation>>, + detail::negation>, + detail::negation>, + detail::negation>>, + +#if defined(JSON_HAS_CPP_17) && (defined(__GNUC__) || (defined(_MSC_VER) && _MSC_VER >= 1910 && _MSC_VER <= 1914)) + detail::negation>, +#endif + detail::is_detected_lazy + >::value, int >::type = 0 > + JSON_EXPLICIT operator ValueType() const + { + // delegate the call to get<>() const + return get(); + } + + /*! + @return reference to the binary value + + @throw type_error.302 if the value is not binary + + @sa see @ref is_binary() to check if the value is binary + + @since version 3.8.0 + */ + binary_t& get_binary() + { + if (!is_binary()) + { + JSON_THROW(type_error::create(302, "type must be binary, but is " + std::string(type_name()), *this)); + } + + return *get_ptr(); + } + + /// @copydoc get_binary() + const binary_t& get_binary() const + { + if (!is_binary()) + { + JSON_THROW(type_error::create(302, "type must be binary, but is " + std::string(type_name()), *this)); + } + + return *get_ptr(); + } + + /// @} + + + //////////////////// + // element access // + //////////////////// + + /// @name element access + /// Access to the JSON value. + /// @{ + + /*! + @brief access specified array element with bounds checking + + Returns a reference to the element at specified location @a idx, with + bounds checking. + + @param[in] idx index of the element to access + + @return reference to the element at index @a idx + + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. See example below. + @throw out_of_range.401 if the index @a idx is out of range of the array; + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__size_type} + */ + reference at(size_type idx) + { + // at only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + JSON_TRY + { + return set_parent(m_value.array->at(idx)); + } + JSON_CATCH (std::out_of_range&) + { + // create better exception explanation + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range", *this)); + } + } + else + { + JSON_THROW(type_error::create(304, "cannot use at() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief access specified array element with bounds checking + + Returns a const reference to the element at specified location @a idx, + with bounds checking. + + @param[in] idx index of the element to access + + @return const reference to the element at index @a idx + + @throw type_error.304 if the JSON value is not an array; in this case, + calling `at` with an index makes no sense. See example below. + @throw out_of_range.401 if the index @a idx is out of range of the array; + that is, `idx >= size()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @since version 1.0.0 + + @liveexample{The example below shows how array elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__size_type_const} + */ + const_reference at(size_type idx) const + { + // at only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + JSON_TRY + { + return m_value.array->at(idx); + } + JSON_CATCH (std::out_of_range&) + { + // create better exception explanation + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range", *this)); + } + } + else + { + JSON_THROW(type_error::create(304, "cannot use at() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief access specified object element with bounds checking + + Returns a reference to the element at with specified key @a key, with + bounds checking. + + @param[in] key key of the element to access + + @return reference to the element at key @a key + + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. See example below. + @throw out_of_range.403 if the key @a key is is not stored in the object; + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Logarithmic in the size of the container. + + @sa see @ref operator[](const typename object_t::key_type&) for unchecked + access by reference + @sa see @ref value() for access by value with a default value + + @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read and + written using `at()`. It also demonstrates the different exceptions that + can be thrown.,at__object_t_key_type} + */ + reference at(const typename object_t::key_type& key) + { + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + JSON_TRY + { + return set_parent(m_value.object->at(key)); + } + JSON_CATCH (std::out_of_range&) + { + // create better exception explanation + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found", *this)); + } + } + else + { + JSON_THROW(type_error::create(304, "cannot use at() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief access specified object element with bounds checking + + Returns a const reference to the element at with specified key @a key, + with bounds checking. + + @param[in] key key of the element to access + + @return const reference to the element at key @a key + + @throw type_error.304 if the JSON value is not an object; in this case, + calling `at` with a key makes no sense. See example below. + @throw out_of_range.403 if the key @a key is is not stored in the object; + that is, `find(key) == end()`. See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Logarithmic in the size of the container. + + @sa see @ref operator[](const typename object_t::key_type&) for unchecked + access by reference + @sa see @ref value() for access by value with a default value + + @since version 1.0.0 + + @liveexample{The example below shows how object elements can be read using + `at()`. It also demonstrates the different exceptions that can be thrown., + at__object_t_key_type_const} + */ + const_reference at(const typename object_t::key_type& key) const + { + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + JSON_TRY + { + return m_value.object->at(key); + } + JSON_CATCH (std::out_of_range&) + { + // create better exception explanation + JSON_THROW(out_of_range::create(403, "key '" + key + "' not found", *this)); + } + } + else + { + JSON_THROW(type_error::create(304, "cannot use at() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief access specified array element + + Returns a reference to the element at specified location @a idx. + + @note If @a idx is beyond the range of the array (i.e., `idx >= size()`), + then the array is silently filled up with `null` values to make `idx` a + valid reference to the last stored element. + + @param[in] idx index of the element to access + + @return reference to the element at index @a idx + + @throw type_error.305 if the JSON value is not an array or null; in that + cases, using the [] operator with an index makes no sense. + + @complexity Constant if @a idx is in the range of the array. Otherwise + linear in `idx - size()`. + + @liveexample{The example below shows how array elements can be read and + written using `[]` operator. Note the addition of `null` + values.,operatorarray__size_type} + + @since version 1.0.0 + */ + reference operator[](size_type idx) + { + // implicitly convert null value to an empty array + if (is_null()) + { + m_type = value_t::array; + m_value.array = create(); + assert_invariant(); + } + + // operator[] only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + // fill up array with null values if given idx is outside range + if (idx >= m_value.array->size()) + { +#if JSON_DIAGNOSTICS + // remember array size & capacity before resizing + const auto old_size = m_value.array->size(); + const auto old_capacity = m_value.array->capacity(); +#endif + m_value.array->resize(idx + 1); + +#if JSON_DIAGNOSTICS + if (JSON_HEDLEY_UNLIKELY(m_value.array->capacity() != old_capacity)) + { + // capacity has changed: update all parents + set_parents(); + } + else + { + // set parent for values added above + set_parents(begin() + static_cast(old_size), static_cast(idx + 1 - old_size)); + } +#endif + assert_invariant(); + } + + return m_value.array->operator[](idx); + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a numeric argument with " + std::string(type_name()), *this)); + } + + /*! + @brief access specified array element + + Returns a const reference to the element at specified location @a idx. + + @param[in] idx index of the element to access + + @return const reference to the element at index @a idx + + @throw type_error.305 if the JSON value is not an array; in that case, + using the [] operator with an index makes no sense. + + @complexity Constant. + + @liveexample{The example below shows how array elements can be read using + the `[]` operator.,operatorarray__size_type_const} + + @since version 1.0.0 + */ + const_reference operator[](size_type idx) const + { + // const operator[] only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + return m_value.array->operator[](idx); + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a numeric argument with " + std::string(type_name()), *this)); + } + + /*! + @brief access specified object element + + Returns a reference to the element at with specified key @a key. + + @note If @a key is not found in the object, then it is silently added to + the object and filled with a `null` value to make `key` a valid reference. + In case the value was `null` before, it is converted to an object. + + @param[in] key key of the element to access + + @return reference to the element at key @a key + + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be read and + written using the `[]` operator.,operatorarray__key_type} + + @sa see @ref at(const typename object_t::key_type&) for access by reference + with range checking + @sa see @ref value() for access by value with a default value + + @since version 1.0.0 + */ + reference operator[](const typename object_t::key_type& key) + { + // implicitly convert null value to an empty object + if (is_null()) + { + m_type = value_t::object; + m_value.object = create(); + assert_invariant(); + } + + // operator[] only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + return set_parent(m_value.object->operator[](key)); + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a string argument with " + std::string(type_name()), *this)); + } + + /*! + @brief read-only access specified object element + + Returns a const reference to the element at with specified key @a key. No + bounds checking is performed. + + @warning If the element with key @a key does not exist, the behavior is + undefined. + + @param[in] key key of the element to access + + @return const reference to the element at key @a key + + @pre The element with key @a key must exist. **This precondition is + enforced with an assertion.** + + @throw type_error.305 if the JSON value is not an object; in that case, + using the [] operator with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be read using + the `[]` operator.,operatorarray__key_type_const} + + @sa see @ref at(const typename object_t::key_type&) for access by reference + with range checking + @sa see @ref value() for access by value with a default value + + @since version 1.0.0 + */ + const_reference operator[](const typename object_t::key_type& key) const + { + // const operator[] only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + JSON_ASSERT(m_value.object->find(key) != m_value.object->end()); + return m_value.object->find(key)->second; + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a string argument with " + std::string(type_name()), *this)); + } + + /*! + @brief access specified object element + + Returns a reference to the element at with specified key @a key. + + @note If @a key is not found in the object, then it is silently added to + the object and filled with a `null` value to make `key` a valid reference. + In case the value was `null` before, it is converted to an object. + + @param[in] key key of the element to access + + @return reference to the element at key @a key + + @throw type_error.305 if the JSON value is not an object or null; in that + cases, using the [] operator with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be read and + written using the `[]` operator.,operatorarray__key_type} + + @sa see @ref at(const typename object_t::key_type&) for access by reference + with range checking + @sa see @ref value() for access by value with a default value + + @since version 1.1.0 + */ + template + JSON_HEDLEY_NON_NULL(2) + reference operator[](T* key) + { + // implicitly convert null to object + if (is_null()) + { + m_type = value_t::object; + m_value = value_t::object; + assert_invariant(); + } + + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + return set_parent(m_value.object->operator[](key)); + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a string argument with " + std::string(type_name()), *this)); + } + + /*! + @brief read-only access specified object element + + Returns a const reference to the element at with specified key @a key. No + bounds checking is performed. + + @warning If the element with key @a key does not exist, the behavior is + undefined. + + @param[in] key key of the element to access + + @return const reference to the element at key @a key + + @pre The element with key @a key must exist. **This precondition is + enforced with an assertion.** + + @throw type_error.305 if the JSON value is not an object; in that case, + using the [] operator with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be read using + the `[]` operator.,operatorarray__key_type_const} + + @sa see @ref at(const typename object_t::key_type&) for access by reference + with range checking + @sa see @ref value() for access by value with a default value + + @since version 1.1.0 + */ + template + JSON_HEDLEY_NON_NULL(2) + const_reference operator[](T* key) const + { + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + JSON_ASSERT(m_value.object->find(key) != m_value.object->end()); + return m_value.object->find(key)->second; + } + + JSON_THROW(type_error::create(305, "cannot use operator[] with a string argument with " + std::string(type_name()), *this)); + } + + /*! + @brief access specified object element with default value + + Returns either a copy of an object's element at the specified key @a key + or a given default value if no element with key @a key exists. + + The function is basically equivalent to executing + @code {.cpp} + try { + return at(key); + } catch(out_of_range) { + return default_value; + } + @endcode + + @note Unlike @ref at(const typename object_t::key_type&), this function + does not throw if the given key @a key was not found. + + @note Unlike @ref operator[](const typename object_t::key_type& key), this + function does not implicitly add an element to the position defined by @a + key. This function is furthermore also applicable to const objects. + + @param[in] key key of the element to access + @param[in] default_value the value to return if @a key is not found + + @tparam ValueType type compatible to JSON values, for instance `int` for + JSON integer numbers, `bool` for JSON booleans, or `std::vector` types for + JSON arrays. Note the type of the expected value at @a key and the default + value @a default_value must be compatible. + + @return copy of the element at key @a key or @a default_value if @a key + is not found + + @throw type_error.302 if @a default_value does not match the type of the + value at @a key + @throw type_error.306 if the JSON value is not an object; in that case, + using `value()` with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be queried + with a default value.,basic_json__value} + + @sa see @ref at(const typename object_t::key_type&) for access by reference + with range checking + @sa see @ref operator[](const typename object_t::key_type&) for unchecked + access by reference + + @since version 1.0.0 + */ + // using std::is_convertible in a std::enable_if will fail when using explicit conversions + template < class ValueType, typename std::enable_if < + detail::is_getable::value + && !std::is_same::value, int >::type = 0 > + ValueType value(const typename object_t::key_type& key, const ValueType& default_value) const + { + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + // if key is found, return value and given default value otherwise + const auto it = find(key); + if (it != end()) + { + return it->template get(); + } + + return default_value; + } + + JSON_THROW(type_error::create(306, "cannot use value() with " + std::string(type_name()), *this)); + } + + /*! + @brief overload for a default value of type const char* + @copydoc basic_json::value(const typename object_t::key_type&, const ValueType&) const + */ + string_t value(const typename object_t::key_type& key, const char* default_value) const + { + return value(key, string_t(default_value)); + } + + /*! + @brief access specified object element via JSON Pointer with default value + + Returns either a copy of an object's element at the specified key @a key + or a given default value if no element with key @a key exists. + + The function is basically equivalent to executing + @code {.cpp} + try { + return at(ptr); + } catch(out_of_range) { + return default_value; + } + @endcode + + @note Unlike @ref at(const json_pointer&), this function does not throw + if the given key @a key was not found. + + @param[in] ptr a JSON pointer to the element to access + @param[in] default_value the value to return if @a ptr found no value + + @tparam ValueType type compatible to JSON values, for instance `int` for + JSON integer numbers, `bool` for JSON booleans, or `std::vector` types for + JSON arrays. Note the type of the expected value at @a key and the default + value @a default_value must be compatible. + + @return copy of the element at key @a key or @a default_value if @a key + is not found + + @throw type_error.302 if @a default_value does not match the type of the + value at @a ptr + @throw type_error.306 if the JSON value is not an object; in that case, + using `value()` with a key makes no sense. + + @complexity Logarithmic in the size of the container. + + @liveexample{The example below shows how object elements can be queried + with a default value.,basic_json__value_ptr} + + @sa see @ref operator[](const json_pointer&) for unchecked access by reference + + @since version 2.0.2 + */ + template::value, int>::type = 0> + ValueType value(const json_pointer& ptr, const ValueType& default_value) const + { + // at only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + // if pointer resolves a value, return it or use default value + JSON_TRY + { + return ptr.get_checked(this).template get(); + } + JSON_INTERNAL_CATCH (out_of_range&) + { + return default_value; + } + } + + JSON_THROW(type_error::create(306, "cannot use value() with " + std::string(type_name()), *this)); + } + + /*! + @brief overload for a default value of type const char* + @copydoc basic_json::value(const json_pointer&, ValueType) const + */ + JSON_HEDLEY_NON_NULL(3) + string_t value(const json_pointer& ptr, const char* default_value) const + { + return value(ptr, string_t(default_value)); + } + + /*! + @brief access the first element + + Returns a reference to the first element in the container. For a JSON + container `c`, the expression `c.front()` is equivalent to `*c.begin()`. + + @return In case of a structured type (array or object), a reference to the + first element is returned. In case of number, string, boolean, or binary + values, a reference to the value is returned. + + @complexity Constant. + + @pre The JSON value must not be `null` (would throw `std::out_of_range`) + or an empty array or object (undefined behavior, **guarded by + assertions**). + @post The JSON value remains unchanged. + + @throw invalid_iterator.214 when called on `null` value + + @liveexample{The following code shows an example for `front()`.,front} + + @sa see @ref back() -- access the last element + + @since version 1.0.0 + */ + reference front() + { + return *begin(); + } + + /*! + @copydoc basic_json::front() + */ + const_reference front() const + { + return *cbegin(); + } + + /*! + @brief access the last element + + Returns a reference to the last element in the container. For a JSON + container `c`, the expression `c.back()` is equivalent to + @code {.cpp} + auto tmp = c.end(); + --tmp; + return *tmp; + @endcode + + @return In case of a structured type (array or object), a reference to the + last element is returned. In case of number, string, boolean, or binary + values, a reference to the value is returned. + + @complexity Constant. + + @pre The JSON value must not be `null` (would throw `std::out_of_range`) + or an empty array or object (undefined behavior, **guarded by + assertions**). + @post The JSON value remains unchanged. + + @throw invalid_iterator.214 when called on a `null` value. See example + below. + + @liveexample{The following code shows an example for `back()`.,back} + + @sa see @ref front() -- access the first element + + @since version 1.0.0 + */ + reference back() + { + auto tmp = end(); + --tmp; + return *tmp; + } + + /*! + @copydoc basic_json::back() + */ + const_reference back() const + { + auto tmp = cend(); + --tmp; + return *tmp; + } + + /*! + @brief remove element given an iterator + + Removes the element specified by iterator @a pos. The iterator @a pos must + be valid and dereferenceable. Thus the `end()` iterator (which is valid, + but is not dereferenceable) cannot be used as a value for @a pos. + + If called on a primitive type other than `null`, the resulting JSON value + will be `null`. + + @param[in] pos iterator to the element to remove + @return Iterator following the last removed element. If the iterator @a + pos refers to the last element, the `end()` iterator is returned. + + @tparam IteratorType an @ref iterator or @ref const_iterator + + @post Invalidates iterators and references at or after the point of the + erase, including the `end()` iterator. + + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` + @throw invalid_iterator.202 if called on an iterator which does not belong + to the current JSON value; example: `"iterator does not fit current + value"` + @throw invalid_iterator.205 if called on a primitive type with invalid + iterator (i.e., any iterator which is not `begin()`); example: `"iterator + out of range"` + + @complexity The complexity depends on the type: + - objects: amortized constant + - arrays: linear in distance between @a pos and the end of the container + - strings and binary: linear in the length of the member + - other types: constant + + @liveexample{The example shows the result of `erase()` for different JSON + types.,erase__IteratorType} + + @sa see @ref erase(IteratorType, IteratorType) -- removes the elements in + the given range + @sa see @ref erase(const typename object_t::key_type&) -- removes the element + from an object at the given key + @sa see @ref erase(const size_type) -- removes the element from an array at + the given index + + @since version 1.0.0 + */ + template < class IteratorType, typename std::enable_if < + std::is_same::value || + std::is_same::value, int >::type + = 0 > + IteratorType erase(IteratorType pos) + { + // make sure iterator fits the current value + if (JSON_HEDLEY_UNLIKELY(this != pos.m_object)) + { + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value", *this)); + } + + IteratorType result = end(); + + switch (m_type) + { + case value_t::boolean: + case value_t::number_float: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::string: + case value_t::binary: + { + if (JSON_HEDLEY_UNLIKELY(!pos.m_it.primitive_iterator.is_begin())) + { + JSON_THROW(invalid_iterator::create(205, "iterator out of range", *this)); + } + + if (is_string()) + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, m_value.string); + std::allocator_traits::deallocate(alloc, m_value.string, 1); + m_value.string = nullptr; + } + else if (is_binary()) + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, m_value.binary); + std::allocator_traits::deallocate(alloc, m_value.binary, 1); + m_value.binary = nullptr; + } + + m_type = value_t::null; + assert_invariant(); + break; + } + + case value_t::object: + { + result.m_it.object_iterator = m_value.object->erase(pos.m_it.object_iterator); + break; + } + + case value_t::array: + { + result.m_it.array_iterator = m_value.array->erase(pos.m_it.array_iterator); + break; + } + + case value_t::null: + case value_t::discarded: + default: + JSON_THROW(type_error::create(307, "cannot use erase() with " + std::string(type_name()), *this)); + } + + return result; + } + + /*! + @brief remove elements given an iterator range + + Removes the element specified by the range `[first; last)`. The iterator + @a first does not need to be dereferenceable if `first == last`: erasing + an empty range is a no-op. + + If called on a primitive type other than `null`, the resulting JSON value + will be `null`. + + @param[in] first iterator to the beginning of the range to remove + @param[in] last iterator past the end of the range to remove + @return Iterator following the last removed element. If the iterator @a + second refers to the last element, the `end()` iterator is returned. + + @tparam IteratorType an @ref iterator or @ref const_iterator + + @post Invalidates iterators and references at or after the point of the + erase, including the `end()` iterator. + + @throw type_error.307 if called on a `null` value; example: `"cannot use + erase() with null"` + @throw invalid_iterator.203 if called on iterators which does not belong + to the current JSON value; example: `"iterators do not fit current value"` + @throw invalid_iterator.204 if called on a primitive type with invalid + iterators (i.e., if `first != begin()` and `last != end()`); example: + `"iterators out of range"` + + @complexity The complexity depends on the type: + - objects: `log(size()) + std::distance(first, last)` + - arrays: linear in the distance between @a first and @a last, plus linear + in the distance between @a last and end of the container + - strings and binary: linear in the length of the member + - other types: constant + + @liveexample{The example shows the result of `erase()` for different JSON + types.,erase__IteratorType_IteratorType} + + @sa see @ref erase(IteratorType) -- removes the element at a given position + @sa see @ref erase(const typename object_t::key_type&) -- removes the element + from an object at the given key + @sa see @ref erase(const size_type) -- removes the element from an array at + the given index + + @since version 1.0.0 + */ + template < class IteratorType, typename std::enable_if < + std::is_same::value || + std::is_same::value, int >::type + = 0 > + IteratorType erase(IteratorType first, IteratorType last) + { + // make sure iterator fits the current value + if (JSON_HEDLEY_UNLIKELY(this != first.m_object || this != last.m_object)) + { + JSON_THROW(invalid_iterator::create(203, "iterators do not fit current value", *this)); + } + + IteratorType result = end(); + + switch (m_type) + { + case value_t::boolean: + case value_t::number_float: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::string: + case value_t::binary: + { + if (JSON_HEDLEY_LIKELY(!first.m_it.primitive_iterator.is_begin() + || !last.m_it.primitive_iterator.is_end())) + { + JSON_THROW(invalid_iterator::create(204, "iterators out of range", *this)); + } + + if (is_string()) + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, m_value.string); + std::allocator_traits::deallocate(alloc, m_value.string, 1); + m_value.string = nullptr; + } + else if (is_binary()) + { + AllocatorType alloc; + std::allocator_traits::destroy(alloc, m_value.binary); + std::allocator_traits::deallocate(alloc, m_value.binary, 1); + m_value.binary = nullptr; + } + + m_type = value_t::null; + assert_invariant(); + break; + } + + case value_t::object: + { + result.m_it.object_iterator = m_value.object->erase(first.m_it.object_iterator, + last.m_it.object_iterator); + break; + } + + case value_t::array: + { + result.m_it.array_iterator = m_value.array->erase(first.m_it.array_iterator, + last.m_it.array_iterator); + break; + } + + case value_t::null: + case value_t::discarded: + default: + JSON_THROW(type_error::create(307, "cannot use erase() with " + std::string(type_name()), *this)); + } + + return result; + } + + /*! + @brief remove element from a JSON object given a key + + Removes elements from a JSON object with the key value @a key. + + @param[in] key value of the elements to remove + + @return Number of elements removed. If @a ObjectType is the default + `std::map` type, the return value will always be `0` (@a key was not + found) or `1` (@a key was found). + + @post References and iterators to the erased elements are invalidated. + Other references and iterators are not affected. + + @throw type_error.307 when called on a type other than JSON object; + example: `"cannot use erase() with null"` + + @complexity `log(size()) + count(key)` + + @liveexample{The example shows the effect of `erase()`.,erase__key_type} + + @sa see @ref erase(IteratorType) -- removes the element at a given position + @sa see @ref erase(IteratorType, IteratorType) -- removes the elements in + the given range + @sa see @ref erase(const size_type) -- removes the element from an array at + the given index + + @since version 1.0.0 + */ + size_type erase(const typename object_t::key_type& key) + { + // this erase only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + return m_value.object->erase(key); + } + + JSON_THROW(type_error::create(307, "cannot use erase() with " + std::string(type_name()), *this)); + } + + /*! + @brief remove element from a JSON array given an index + + Removes element from a JSON array at the index @a idx. + + @param[in] idx index of the element to remove + + @throw type_error.307 when called on a type other than JSON object; + example: `"cannot use erase() with null"` + @throw out_of_range.401 when `idx >= size()`; example: `"array index 17 + is out of range"` + + @complexity Linear in distance between @a idx and the end of the container. + + @liveexample{The example shows the effect of `erase()`.,erase__size_type} + + @sa see @ref erase(IteratorType) -- removes the element at a given position + @sa see @ref erase(IteratorType, IteratorType) -- removes the elements in + the given range + @sa see @ref erase(const typename object_t::key_type&) -- removes the element + from an object at the given key + + @since version 1.0.0 + */ + void erase(const size_type idx) + { + // this erase only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + if (JSON_HEDLEY_UNLIKELY(idx >= size())) + { + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range", *this)); + } + + m_value.array->erase(m_value.array->begin() + static_cast(idx)); + } + else + { + JSON_THROW(type_error::create(307, "cannot use erase() with " + std::string(type_name()), *this)); + } + } + + /// @} + + + //////////// + // lookup // + //////////// + + /// @name lookup + /// @{ + + /*! + @brief find an element in a JSON object + + Finds an element in a JSON object with key equivalent to @a key. If the + element is not found or the JSON value is not an object, end() is + returned. + + @note This method always returns @ref end() when executed on a JSON type + that is not an object. + + @param[in] key key value of the element to search for. + + @return Iterator to an element with key equivalent to @a key. If no such + element is found or the JSON value is not an object, past-the-end (see + @ref end()) iterator is returned. + + @complexity Logarithmic in the size of the JSON object. + + @liveexample{The example shows how `find()` is used.,find__key_type} + + @sa see @ref contains(KeyT&&) const -- checks whether a key exists + + @since version 1.0.0 + */ + template + iterator find(KeyT&& key) + { + auto result = end(); + + if (is_object()) + { + result.m_it.object_iterator = m_value.object->find(std::forward(key)); + } + + return result; + } + + /*! + @brief find an element in a JSON object + @copydoc find(KeyT&&) + */ + template + const_iterator find(KeyT&& key) const + { + auto result = cend(); + + if (is_object()) + { + result.m_it.object_iterator = m_value.object->find(std::forward(key)); + } + + return result; + } + + /*! + @brief returns the number of occurrences of a key in a JSON object + + Returns the number of elements with key @a key. If ObjectType is the + default `std::map` type, the return value will always be `0` (@a key was + not found) or `1` (@a key was found). + + @note This method always returns `0` when executed on a JSON type that is + not an object. + + @param[in] key key value of the element to count + + @return Number of elements with key @a key. If the JSON value is not an + object, the return value will be `0`. + + @complexity Logarithmic in the size of the JSON object. + + @liveexample{The example shows how `count()` is used.,count} + + @since version 1.0.0 + */ + template + size_type count(KeyT&& key) const + { + // return 0 for all nonobject types + return is_object() ? m_value.object->count(std::forward(key)) : 0; + } + + /*! + @brief check the existence of an element in a JSON object + + Check whether an element exists in a JSON object with key equivalent to + @a key. If the element is not found or the JSON value is not an object, + false is returned. + + @note This method always returns false when executed on a JSON type + that is not an object. + + @param[in] key key value to check its existence. + + @return true if an element with specified @a key exists. If no such + element with such key is found or the JSON value is not an object, + false is returned. + + @complexity Logarithmic in the size of the JSON object. + + @liveexample{The following code shows an example for `contains()`.,contains} + + @sa see @ref find(KeyT&&) -- returns an iterator to an object element + @sa see @ref contains(const json_pointer&) const -- checks the existence for a JSON pointer + + @since version 3.6.0 + */ + template < typename KeyT, typename std::enable_if < + !std::is_same::type, json_pointer>::value, int >::type = 0 > + bool contains(KeyT && key) const + { + return is_object() && m_value.object->find(std::forward(key)) != m_value.object->end(); + } + + /*! + @brief check the existence of an element in a JSON object given a JSON pointer + + Check whether the given JSON pointer @a ptr can be resolved in the current + JSON value. + + @note This method can be executed on any JSON value type. + + @param[in] ptr JSON pointer to check its existence. + + @return true if the JSON pointer can be resolved to a stored value, false + otherwise. + + @post If `j.contains(ptr)` returns true, it is safe to call `j[ptr]`. + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + + @complexity Logarithmic in the size of the JSON object. + + @liveexample{The following code shows an example for `contains()`.,contains_json_pointer} + + @sa see @ref contains(KeyT &&) const -- checks the existence of a key + + @since version 3.7.0 + */ + bool contains(const json_pointer& ptr) const + { + return ptr.contains(this); + } + + /// @} + + + /////////////// + // iterators // + /////////////// + + /// @name iterators + /// @{ + + /*! + @brief returns an iterator to the first element + + Returns an iterator to the first element. + + @image html range-begin-end.svg "Illustration from cppreference.com" + + @return iterator to the first element + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + + @liveexample{The following code shows an example for `begin()`.,begin} + + @sa see @ref cbegin() -- returns a const iterator to the beginning + @sa see @ref end() -- returns an iterator to the end + @sa see @ref cend() -- returns a const iterator to the end + + @since version 1.0.0 + */ + iterator begin() noexcept + { + iterator result(this); + result.set_begin(); + return result; + } + + /*! + @copydoc basic_json::cbegin() + */ + const_iterator begin() const noexcept + { + return cbegin(); + } + + /*! + @brief returns a const iterator to the first element + + Returns a const iterator to the first element. + + @image html range-begin-end.svg "Illustration from cppreference.com" + + @return const iterator to the first element + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + - Has the semantics of `const_cast(*this).begin()`. + + @liveexample{The following code shows an example for `cbegin()`.,cbegin} + + @sa see @ref begin() -- returns an iterator to the beginning + @sa see @ref end() -- returns an iterator to the end + @sa see @ref cend() -- returns a const iterator to the end + + @since version 1.0.0 + */ + const_iterator cbegin() const noexcept + { + const_iterator result(this); + result.set_begin(); + return result; + } + + /*! + @brief returns an iterator to one past the last element + + Returns an iterator to one past the last element. + + @image html range-begin-end.svg "Illustration from cppreference.com" + + @return iterator one past the last element + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + + @liveexample{The following code shows an example for `end()`.,end} + + @sa see @ref cend() -- returns a const iterator to the end + @sa see @ref begin() -- returns an iterator to the beginning + @sa see @ref cbegin() -- returns a const iterator to the beginning + + @since version 1.0.0 + */ + iterator end() noexcept + { + iterator result(this); + result.set_end(); + return result; + } + + /*! + @copydoc basic_json::cend() + */ + const_iterator end() const noexcept + { + return cend(); + } + + /*! + @brief returns a const iterator to one past the last element + + Returns a const iterator to one past the last element. + + @image html range-begin-end.svg "Illustration from cppreference.com" + + @return const iterator one past the last element + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + - Has the semantics of `const_cast(*this).end()`. + + @liveexample{The following code shows an example for `cend()`.,cend} + + @sa see @ref end() -- returns an iterator to the end + @sa see @ref begin() -- returns an iterator to the beginning + @sa see @ref cbegin() -- returns a const iterator to the beginning + + @since version 1.0.0 + */ + const_iterator cend() const noexcept + { + const_iterator result(this); + result.set_end(); + return result; + } + + /*! + @brief returns an iterator to the reverse-beginning + + Returns an iterator to the reverse-beginning; that is, the last element. + + @image html range-rbegin-rend.svg "Illustration from cppreference.com" + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [ReversibleContainer](https://en.cppreference.com/w/cpp/named_req/ReversibleContainer) + requirements: + - The complexity is constant. + - Has the semantics of `reverse_iterator(end())`. + + @liveexample{The following code shows an example for `rbegin()`.,rbegin} + + @sa see @ref crbegin() -- returns a const reverse iterator to the beginning + @sa see @ref rend() -- returns a reverse iterator to the end + @sa see @ref crend() -- returns a const reverse iterator to the end + + @since version 1.0.0 + */ + reverse_iterator rbegin() noexcept + { + return reverse_iterator(end()); + } + + /*! + @copydoc basic_json::crbegin() + */ + const_reverse_iterator rbegin() const noexcept + { + return crbegin(); + } + + /*! + @brief returns an iterator to the reverse-end + + Returns an iterator to the reverse-end; that is, one before the first + element. + + @image html range-rbegin-rend.svg "Illustration from cppreference.com" + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [ReversibleContainer](https://en.cppreference.com/w/cpp/named_req/ReversibleContainer) + requirements: + - The complexity is constant. + - Has the semantics of `reverse_iterator(begin())`. + + @liveexample{The following code shows an example for `rend()`.,rend} + + @sa see @ref crend() -- returns a const reverse iterator to the end + @sa see @ref rbegin() -- returns a reverse iterator to the beginning + @sa see @ref crbegin() -- returns a const reverse iterator to the beginning + + @since version 1.0.0 + */ + reverse_iterator rend() noexcept + { + return reverse_iterator(begin()); + } + + /*! + @copydoc basic_json::crend() + */ + const_reverse_iterator rend() const noexcept + { + return crend(); + } + + /*! + @brief returns a const reverse iterator to the last element + + Returns a const iterator to the reverse-beginning; that is, the last + element. + + @image html range-rbegin-rend.svg "Illustration from cppreference.com" + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [ReversibleContainer](https://en.cppreference.com/w/cpp/named_req/ReversibleContainer) + requirements: + - The complexity is constant. + - Has the semantics of `const_cast(*this).rbegin()`. + + @liveexample{The following code shows an example for `crbegin()`.,crbegin} + + @sa see @ref rbegin() -- returns a reverse iterator to the beginning + @sa see @ref rend() -- returns a reverse iterator to the end + @sa see @ref crend() -- returns a const reverse iterator to the end + + @since version 1.0.0 + */ + const_reverse_iterator crbegin() const noexcept + { + return const_reverse_iterator(cend()); + } + + /*! + @brief returns a const reverse iterator to one before the first + + Returns a const reverse iterator to the reverse-end; that is, one before + the first element. + + @image html range-rbegin-rend.svg "Illustration from cppreference.com" + + @complexity Constant. + + @requirement This function helps `basic_json` satisfying the + [ReversibleContainer](https://en.cppreference.com/w/cpp/named_req/ReversibleContainer) + requirements: + - The complexity is constant. + - Has the semantics of `const_cast(*this).rend()`. + + @liveexample{The following code shows an example for `crend()`.,crend} + + @sa see @ref rend() -- returns a reverse iterator to the end + @sa see @ref rbegin() -- returns a reverse iterator to the beginning + @sa see @ref crbegin() -- returns a const reverse iterator to the beginning + + @since version 1.0.0 + */ + const_reverse_iterator crend() const noexcept + { + return const_reverse_iterator(cbegin()); + } + + public: + /*! + @brief wrapper to access iterator member functions in range-based for + + This function allows to access @ref iterator::key() and @ref + iterator::value() during range-based for loops. In these loops, a + reference to the JSON values is returned, so there is no access to the + underlying iterator. + + For loop without iterator_wrapper: + + @code{cpp} + for (auto it = j_object.begin(); it != j_object.end(); ++it) + { + std::cout << "key: " << it.key() << ", value:" << it.value() << '\n'; + } + @endcode + + Range-based for loop without iterator proxy: + + @code{cpp} + for (auto it : j_object) + { + // "it" is of type json::reference and has no key() member + std::cout << "value: " << it << '\n'; + } + @endcode + + Range-based for loop with iterator proxy: + + @code{cpp} + for (auto it : json::iterator_wrapper(j_object)) + { + std::cout << "key: " << it.key() << ", value:" << it.value() << '\n'; + } + @endcode + + @note When iterating over an array, `key()` will return the index of the + element as string (see example). + + @param[in] ref reference to a JSON value + @return iteration proxy object wrapping @a ref with an interface to use in + range-based for loops + + @liveexample{The following code shows how the wrapper is used,iterator_wrapper} + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @note The name of this function is not yet final and may change in the + future. + + @deprecated This stream operator is deprecated and will be removed in + future 4.0.0 of the library. Please use @ref items() instead; + that is, replace `json::iterator_wrapper(j)` with `j.items()`. + */ + JSON_HEDLEY_DEPRECATED_FOR(3.1.0, items()) + static iteration_proxy iterator_wrapper(reference ref) noexcept + { + return ref.items(); + } + + /*! + @copydoc iterator_wrapper(reference) + */ + JSON_HEDLEY_DEPRECATED_FOR(3.1.0, items()) + static iteration_proxy iterator_wrapper(const_reference ref) noexcept + { + return ref.items(); + } + + /*! + @brief helper to access iterator member functions in range-based for + + This function allows to access @ref iterator::key() and @ref + iterator::value() during range-based for loops. In these loops, a + reference to the JSON values is returned, so there is no access to the + underlying iterator. + + For loop without `items()` function: + + @code{cpp} + for (auto it = j_object.begin(); it != j_object.end(); ++it) + { + std::cout << "key: " << it.key() << ", value:" << it.value() << '\n'; + } + @endcode + + Range-based for loop without `items()` function: + + @code{cpp} + for (auto it : j_object) + { + // "it" is of type json::reference and has no key() member + std::cout << "value: " << it << '\n'; + } + @endcode + + Range-based for loop with `items()` function: + + @code{cpp} + for (auto& el : j_object.items()) + { + std::cout << "key: " << el.key() << ", value:" << el.value() << '\n'; + } + @endcode + + The `items()` function also allows to use + [structured bindings](https://en.cppreference.com/w/cpp/language/structured_binding) + (C++17): + + @code{cpp} + for (auto& [key, val] : j_object.items()) + { + std::cout << "key: " << key << ", value:" << val << '\n'; + } + @endcode + + @note When iterating over an array, `key()` will return the index of the + element as string (see example). For primitive types (e.g., numbers), + `key()` returns an empty string. + + @warning Using `items()` on temporary objects is dangerous. Make sure the + object's lifetime exeeds the iteration. See + for more + information. + + @return iteration proxy object wrapping @a ref with an interface to use in + range-based for loops + + @liveexample{The following code shows how the function is used.,items} + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @since version 3.1.0, structured bindings support since 3.5.0. + */ + iteration_proxy items() noexcept + { + return iteration_proxy(*this); + } + + /*! + @copydoc items() + */ + iteration_proxy items() const noexcept + { + return iteration_proxy(*this); + } + + /// @} + + + ////////////// + // capacity // + ////////////// + + /// @name capacity + /// @{ + + /*! + @brief checks whether the container is empty. + + Checks if a JSON value has no elements (i.e. whether its @ref size is `0`). + + @return The return value depends on the different types and is + defined as follows: + Value type | return value + ----------- | ------------- + null | `true` + boolean | `false` + string | `false` + number | `false` + binary | `false` + object | result of function `object_t::empty()` + array | result of function `array_t::empty()` + + @liveexample{The following code uses `empty()` to check if a JSON + object contains any elements.,empty} + + @complexity Constant, as long as @ref array_t and @ref object_t satisfy + the Container concept; that is, their `empty()` functions have constant + complexity. + + @iterators No changes. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @note This function does not return whether a string stored as JSON value + is empty - it returns whether the JSON container itself is empty which is + false in the case of a string. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + - Has the semantics of `begin() == end()`. + + @sa see @ref size() -- returns the number of elements + + @since version 1.0.0 + */ + bool empty() const noexcept + { + switch (m_type) + { + case value_t::null: + { + // null values are empty + return true; + } + + case value_t::array: + { + // delegate call to array_t::empty() + return m_value.array->empty(); + } + + case value_t::object: + { + // delegate call to object_t::empty() + return m_value.object->empty(); + } + + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + // all other types are nonempty + return false; + } + } + } + + /*! + @brief returns the number of elements + + Returns the number of elements in a JSON value. + + @return The return value depends on the different types and is + defined as follows: + Value type | return value + ----------- | ------------- + null | `0` + boolean | `1` + string | `1` + number | `1` + binary | `1` + object | result of function object_t::size() + array | result of function array_t::size() + + @liveexample{The following code calls `size()` on the different value + types.,size} + + @complexity Constant, as long as @ref array_t and @ref object_t satisfy + the Container concept; that is, their size() functions have constant + complexity. + + @iterators No changes. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @note This function does not return the length of a string stored as JSON + value - it returns the number of elements in the JSON value which is 1 in + the case of a string. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + - Has the semantics of `std::distance(begin(), end())`. + + @sa see @ref empty() -- checks whether the container is empty + @sa see @ref max_size() -- returns the maximal number of elements + + @since version 1.0.0 + */ + size_type size() const noexcept + { + switch (m_type) + { + case value_t::null: + { + // null values are empty + return 0; + } + + case value_t::array: + { + // delegate call to array_t::size() + return m_value.array->size(); + } + + case value_t::object: + { + // delegate call to object_t::size() + return m_value.object->size(); + } + + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + // all other types have size 1 + return 1; + } + } + } + + /*! + @brief returns the maximum possible number of elements + + Returns the maximum number of elements a JSON value is able to hold due to + system or library implementation limitations, i.e. `std::distance(begin(), + end())` for the JSON value. + + @return The return value depends on the different types and is + defined as follows: + Value type | return value + ----------- | ------------- + null | `0` (same as `size()`) + boolean | `1` (same as `size()`) + string | `1` (same as `size()`) + number | `1` (same as `size()`) + binary | `1` (same as `size()`) + object | result of function `object_t::max_size()` + array | result of function `array_t::max_size()` + + @liveexample{The following code calls `max_size()` on the different value + types. Note the output is implementation specific.,max_size} + + @complexity Constant, as long as @ref array_t and @ref object_t satisfy + the Container concept; that is, their `max_size()` functions have constant + complexity. + + @iterators No changes. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @requirement This function helps `basic_json` satisfying the + [Container](https://en.cppreference.com/w/cpp/named_req/Container) + requirements: + - The complexity is constant. + - Has the semantics of returning `b.size()` where `b` is the largest + possible JSON value. + + @sa see @ref size() -- returns the number of elements + + @since version 1.0.0 + */ + size_type max_size() const noexcept + { + switch (m_type) + { + case value_t::array: + { + // delegate call to array_t::max_size() + return m_value.array->max_size(); + } + + case value_t::object: + { + // delegate call to object_t::max_size() + return m_value.object->max_size(); + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + // all other types have max_size() == size() + return size(); + } + } + } + + /// @} + + + /////////////// + // modifiers // + /////////////// + + /// @name modifiers + /// @{ + + /*! + @brief clears the contents + + Clears the content of a JSON value and resets it to the default value as + if @ref basic_json(value_t) would have been called with the current value + type from @ref type(): + + Value type | initial value + ----------- | ------------- + null | `null` + boolean | `false` + string | `""` + number | `0` + binary | An empty byte vector + object | `{}` + array | `[]` + + @post Has the same effect as calling + @code {.cpp} + *this = basic_json(type()); + @endcode + + @liveexample{The example below shows the effect of `clear()` to different + JSON types.,clear} + + @complexity Linear in the size of the JSON value. + + @iterators All iterators, pointers and references related to this container + are invalidated. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @sa see @ref basic_json(value_t) -- constructor that creates an object with the + same value than calling `clear()` + + @since version 1.0.0 + */ + void clear() noexcept + { + switch (m_type) + { + case value_t::number_integer: + { + m_value.number_integer = 0; + break; + } + + case value_t::number_unsigned: + { + m_value.number_unsigned = 0; + break; + } + + case value_t::number_float: + { + m_value.number_float = 0.0; + break; + } + + case value_t::boolean: + { + m_value.boolean = false; + break; + } + + case value_t::string: + { + m_value.string->clear(); + break; + } + + case value_t::binary: + { + m_value.binary->clear(); + break; + } + + case value_t::array: + { + m_value.array->clear(); + break; + } + + case value_t::object: + { + m_value.object->clear(); + break; + } + + case value_t::null: + case value_t::discarded: + default: + break; + } + } + + /*! + @brief add an object to an array + + Appends the given element @a val to the end of the JSON value. If the + function is called on a JSON null value, an empty array is created before + appending @a val. + + @param[in] val the value to add to the JSON array + + @throw type_error.308 when called on a type other than JSON array or + null; example: `"cannot use push_back() with number"` + + @complexity Amortized constant. + + @liveexample{The example shows how `push_back()` and `+=` can be used to + add elements to a JSON array. Note how the `null` value was silently + converted to a JSON array.,push_back} + + @since version 1.0.0 + */ + void push_back(basic_json&& val) + { + // push_back only works for null objects or arrays + if (JSON_HEDLEY_UNLIKELY(!(is_null() || is_array()))) + { + JSON_THROW(type_error::create(308, "cannot use push_back() with " + std::string(type_name()), *this)); + } + + // transform null object into an array + if (is_null()) + { + m_type = value_t::array; + m_value = value_t::array; + assert_invariant(); + } + + // add element to array (move semantics) + const auto old_capacity = m_value.array->capacity(); + m_value.array->push_back(std::move(val)); + set_parent(m_value.array->back(), old_capacity); + // if val is moved from, basic_json move constructor marks it null so we do not call the destructor + } + + /*! + @brief add an object to an array + @copydoc push_back(basic_json&&) + */ + reference operator+=(basic_json&& val) + { + push_back(std::move(val)); + return *this; + } + + /*! + @brief add an object to an array + @copydoc push_back(basic_json&&) + */ + void push_back(const basic_json& val) + { + // push_back only works for null objects or arrays + if (JSON_HEDLEY_UNLIKELY(!(is_null() || is_array()))) + { + JSON_THROW(type_error::create(308, "cannot use push_back() with " + std::string(type_name()), *this)); + } + + // transform null object into an array + if (is_null()) + { + m_type = value_t::array; + m_value = value_t::array; + assert_invariant(); + } + + // add element to array + const auto old_capacity = m_value.array->capacity(); + m_value.array->push_back(val); + set_parent(m_value.array->back(), old_capacity); + } + + /*! + @brief add an object to an array + @copydoc push_back(basic_json&&) + */ + reference operator+=(const basic_json& val) + { + push_back(val); + return *this; + } + + /*! + @brief add an object to an object + + Inserts the given element @a val to the JSON object. If the function is + called on a JSON null value, an empty object is created before inserting + @a val. + + @param[in] val the value to add to the JSON object + + @throw type_error.308 when called on a type other than JSON object or + null; example: `"cannot use push_back() with number"` + + @complexity Logarithmic in the size of the container, O(log(`size()`)). + + @liveexample{The example shows how `push_back()` and `+=` can be used to + add elements to a JSON object. Note how the `null` value was silently + converted to a JSON object.,push_back__object_t__value} + + @since version 1.0.0 + */ + void push_back(const typename object_t::value_type& val) + { + // push_back only works for null objects or objects + if (JSON_HEDLEY_UNLIKELY(!(is_null() || is_object()))) + { + JSON_THROW(type_error::create(308, "cannot use push_back() with " + std::string(type_name()), *this)); + } + + // transform null object into an object + if (is_null()) + { + m_type = value_t::object; + m_value = value_t::object; + assert_invariant(); + } + + // add element to object + auto res = m_value.object->insert(val); + set_parent(res.first->second); + } + + /*! + @brief add an object to an object + @copydoc push_back(const typename object_t::value_type&) + */ + reference operator+=(const typename object_t::value_type& val) + { + push_back(val); + return *this; + } + + /*! + @brief add an object to an object + + This function allows to use `push_back` with an initializer list. In case + + 1. the current value is an object, + 2. the initializer list @a init contains only two elements, and + 3. the first element of @a init is a string, + + @a init is converted into an object element and added using + @ref push_back(const typename object_t::value_type&). Otherwise, @a init + is converted to a JSON value and added using @ref push_back(basic_json&&). + + @param[in] init an initializer list + + @complexity Linear in the size of the initializer list @a init. + + @note This function is required to resolve an ambiguous overload error, + because pairs like `{"key", "value"}` can be both interpreted as + `object_t::value_type` or `std::initializer_list`, see + https://github.com/nlohmann/json/issues/235 for more information. + + @liveexample{The example shows how initializer lists are treated as + objects when possible.,push_back__initializer_list} + */ + void push_back(initializer_list_t init) + { + if (is_object() && init.size() == 2 && (*init.begin())->is_string()) + { + basic_json&& key = init.begin()->moved_or_copied(); + push_back(typename object_t::value_type( + std::move(key.get_ref()), (init.begin() + 1)->moved_or_copied())); + } + else + { + push_back(basic_json(init)); + } + } + + /*! + @brief add an object to an object + @copydoc push_back(initializer_list_t) + */ + reference operator+=(initializer_list_t init) + { + push_back(init); + return *this; + } + + /*! + @brief add an object to an array + + Creates a JSON value from the passed parameters @a args to the end of the + JSON value. If the function is called on a JSON null value, an empty array + is created before appending the value created from @a args. + + @param[in] args arguments to forward to a constructor of @ref basic_json + @tparam Args compatible types to create a @ref basic_json object + + @return reference to the inserted element + + @throw type_error.311 when called on a type other than JSON array or + null; example: `"cannot use emplace_back() with number"` + + @complexity Amortized constant. + + @liveexample{The example shows how `push_back()` can be used to add + elements to a JSON array. Note how the `null` value was silently converted + to a JSON array.,emplace_back} + + @since version 2.0.8, returns reference since 3.7.0 + */ + template + reference emplace_back(Args&& ... args) + { + // emplace_back only works for null objects or arrays + if (JSON_HEDLEY_UNLIKELY(!(is_null() || is_array()))) + { + JSON_THROW(type_error::create(311, "cannot use emplace_back() with " + std::string(type_name()), *this)); + } + + // transform null object into an array + if (is_null()) + { + m_type = value_t::array; + m_value = value_t::array; + assert_invariant(); + } + + // add element to array (perfect forwarding) + const auto old_capacity = m_value.array->capacity(); + m_value.array->emplace_back(std::forward(args)...); + return set_parent(m_value.array->back(), old_capacity); + } + + /*! + @brief add an object to an object if key does not exist + + Inserts a new element into a JSON object constructed in-place with the + given @a args if there is no element with the key in the container. If the + function is called on a JSON null value, an empty object is created before + appending the value created from @a args. + + @param[in] args arguments to forward to a constructor of @ref basic_json + @tparam Args compatible types to create a @ref basic_json object + + @return a pair consisting of an iterator to the inserted element, or the + already-existing element if no insertion happened, and a bool + denoting whether the insertion took place. + + @throw type_error.311 when called on a type other than JSON object or + null; example: `"cannot use emplace() with number"` + + @complexity Logarithmic in the size of the container, O(log(`size()`)). + + @liveexample{The example shows how `emplace()` can be used to add elements + to a JSON object. Note how the `null` value was silently converted to a + JSON object. Further note how no value is added if there was already one + value stored with the same key.,emplace} + + @since version 2.0.8 + */ + template + std::pair emplace(Args&& ... args) + { + // emplace only works for null objects or arrays + if (JSON_HEDLEY_UNLIKELY(!(is_null() || is_object()))) + { + JSON_THROW(type_error::create(311, "cannot use emplace() with " + std::string(type_name()), *this)); + } + + // transform null object into an object + if (is_null()) + { + m_type = value_t::object; + m_value = value_t::object; + assert_invariant(); + } + + // add element to array (perfect forwarding) + auto res = m_value.object->emplace(std::forward(args)...); + set_parent(res.first->second); + + // create result iterator and set iterator to the result of emplace + auto it = begin(); + it.m_it.object_iterator = res.first; + + // return pair of iterator and boolean + return {it, res.second}; + } + + /// Helper for insertion of an iterator + /// @note: This uses std::distance to support GCC 4.8, + /// see https://github.com/nlohmann/json/pull/1257 + template + iterator insert_iterator(const_iterator pos, Args&& ... args) + { + iterator result(this); + JSON_ASSERT(m_value.array != nullptr); + + auto insert_pos = std::distance(m_value.array->begin(), pos.m_it.array_iterator); + m_value.array->insert(pos.m_it.array_iterator, std::forward(args)...); + result.m_it.array_iterator = m_value.array->begin() + insert_pos; + + // This could have been written as: + // result.m_it.array_iterator = m_value.array->insert(pos.m_it.array_iterator, cnt, val); + // but the return value of insert is missing in GCC 4.8, so it is written this way instead. + + set_parents(); + return result; + } + + /*! + @brief inserts element + + Inserts element @a val before iterator @a pos. + + @param[in] pos iterator before which the content will be inserted; may be + the end() iterator + @param[in] val element to insert + @return iterator pointing to the inserted @a val. + + @throw type_error.309 if called on JSON values other than arrays; + example: `"cannot use insert() with string"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` + + @complexity Constant plus linear in the distance between @a pos and end of + the container. + + @liveexample{The example shows how `insert()` is used.,insert} + + @since version 1.0.0 + */ + iterator insert(const_iterator pos, const basic_json& val) + { + // insert only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + // check if iterator pos fits to this JSON value + if (JSON_HEDLEY_UNLIKELY(pos.m_object != this)) + { + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value", *this)); + } + + // insert to array and return iterator + return insert_iterator(pos, val); + } + + JSON_THROW(type_error::create(309, "cannot use insert() with " + std::string(type_name()), *this)); + } + + /*! + @brief inserts element + @copydoc insert(const_iterator, const basic_json&) + */ + iterator insert(const_iterator pos, basic_json&& val) + { + return insert(pos, val); + } + + /*! + @brief inserts elements + + Inserts @a cnt copies of @a val before iterator @a pos. + + @param[in] pos iterator before which the content will be inserted; may be + the end() iterator + @param[in] cnt number of copies of @a val to insert + @param[in] val element to insert + @return iterator pointing to the first element inserted, or @a pos if + `cnt==0` + + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` + + @complexity Linear in @a cnt plus linear in the distance between @a pos + and end of the container. + + @liveexample{The example shows how `insert()` is used.,insert__count} + + @since version 1.0.0 + */ + iterator insert(const_iterator pos, size_type cnt, const basic_json& val) + { + // insert only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + // check if iterator pos fits to this JSON value + if (JSON_HEDLEY_UNLIKELY(pos.m_object != this)) + { + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value", *this)); + } + + // insert to array and return iterator + return insert_iterator(pos, cnt, val); + } + + JSON_THROW(type_error::create(309, "cannot use insert() with " + std::string(type_name()), *this)); + } + + /*! + @brief inserts elements + + Inserts elements from range `[first, last)` before iterator @a pos. + + @param[in] pos iterator before which the content will be inserted; may be + the end() iterator + @param[in] first begin of the range of elements to insert + @param[in] last end of the range of elements to insert + + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + @throw invalid_iterator.211 if @a first or @a last are iterators into + container for which insert is called; example: `"passed iterators may not + belong to container"` + + @return iterator pointing to the first element inserted, or @a pos if + `first==last` + + @complexity Linear in `std::distance(first, last)` plus linear in the + distance between @a pos and end of the container. + + @liveexample{The example shows how `insert()` is used.,insert__range} + + @since version 1.0.0 + */ + iterator insert(const_iterator pos, const_iterator first, const_iterator last) + { + // insert only works for arrays + if (JSON_HEDLEY_UNLIKELY(!is_array())) + { + JSON_THROW(type_error::create(309, "cannot use insert() with " + std::string(type_name()), *this)); + } + + // check if iterator pos fits to this JSON value + if (JSON_HEDLEY_UNLIKELY(pos.m_object != this)) + { + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value", *this)); + } + + // check if range iterators belong to the same JSON object + if (JSON_HEDLEY_UNLIKELY(first.m_object != last.m_object)) + { + JSON_THROW(invalid_iterator::create(210, "iterators do not fit", *this)); + } + + if (JSON_HEDLEY_UNLIKELY(first.m_object == this)) + { + JSON_THROW(invalid_iterator::create(211, "passed iterators may not belong to container", *this)); + } + + // insert to array and return iterator + return insert_iterator(pos, first.m_it.array_iterator, last.m_it.array_iterator); + } + + /*! + @brief inserts elements + + Inserts elements from initializer list @a ilist before iterator @a pos. + + @param[in] pos iterator before which the content will be inserted; may be + the end() iterator + @param[in] ilist initializer list to insert the values from + + @throw type_error.309 if called on JSON values other than arrays; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if @a pos is not an iterator of *this; + example: `"iterator does not fit current value"` + + @return iterator pointing to the first element inserted, or @a pos if + `ilist` is empty + + @complexity Linear in `ilist.size()` plus linear in the distance between + @a pos and end of the container. + + @liveexample{The example shows how `insert()` is used.,insert__ilist} + + @since version 1.0.0 + */ + iterator insert(const_iterator pos, initializer_list_t ilist) + { + // insert only works for arrays + if (JSON_HEDLEY_UNLIKELY(!is_array())) + { + JSON_THROW(type_error::create(309, "cannot use insert() with " + std::string(type_name()), *this)); + } + + // check if iterator pos fits to this JSON value + if (JSON_HEDLEY_UNLIKELY(pos.m_object != this)) + { + JSON_THROW(invalid_iterator::create(202, "iterator does not fit current value", *this)); + } + + // insert to array and return iterator + return insert_iterator(pos, ilist.begin(), ilist.end()); + } + + /*! + @brief inserts elements + + Inserts elements from range `[first, last)`. + + @param[in] first begin of the range of elements to insert + @param[in] last end of the range of elements to insert + + @throw type_error.309 if called on JSON values other than objects; example: + `"cannot use insert() with string"` + @throw invalid_iterator.202 if iterator @a first or @a last does does not + point to an object; example: `"iterators first and last must point to + objects"` + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + + @complexity Logarithmic: `O(N*log(size() + N))`, where `N` is the number + of elements to insert. + + @liveexample{The example shows how `insert()` is used.,insert__range_object} + + @since version 3.0.0 + */ + void insert(const_iterator first, const_iterator last) + { + // insert only works for objects + if (JSON_HEDLEY_UNLIKELY(!is_object())) + { + JSON_THROW(type_error::create(309, "cannot use insert() with " + std::string(type_name()), *this)); + } + + // check if range iterators belong to the same JSON object + if (JSON_HEDLEY_UNLIKELY(first.m_object != last.m_object)) + { + JSON_THROW(invalid_iterator::create(210, "iterators do not fit", *this)); + } + + // passed iterators must belong to objects + if (JSON_HEDLEY_UNLIKELY(!first.m_object->is_object())) + { + JSON_THROW(invalid_iterator::create(202, "iterators first and last must point to objects", *this)); + } + + m_value.object->insert(first.m_it.object_iterator, last.m_it.object_iterator); + } + + /*! + @brief updates a JSON object from another object, overwriting existing keys + + Inserts all values from JSON object @a j and overwrites existing keys. + + @param[in] j JSON object to read values from + + @throw type_error.312 if called on JSON values other than objects; example: + `"cannot use update() with string"` + + @complexity O(N*log(size() + N)), where N is the number of elements to + insert. + + @liveexample{The example shows how `update()` is used.,update} + + @sa https://docs.python.org/3.6/library/stdtypes.html#dict.update + + @since version 3.0.0 + */ + void update(const_reference j) + { + // implicitly convert null value to an empty object + if (is_null()) + { + m_type = value_t::object; + m_value.object = create(); + assert_invariant(); + } + + if (JSON_HEDLEY_UNLIKELY(!is_object())) + { + JSON_THROW(type_error::create(312, "cannot use update() with " + std::string(type_name()), *this)); + } + if (JSON_HEDLEY_UNLIKELY(!j.is_object())) + { + JSON_THROW(type_error::create(312, "cannot use update() with " + std::string(j.type_name()), *this)); + } + + for (auto it = j.cbegin(); it != j.cend(); ++it) + { + m_value.object->operator[](it.key()) = it.value(); +#if JSON_DIAGNOSTICS + m_value.object->operator[](it.key()).m_parent = this; +#endif + } + } + + /*! + @brief updates a JSON object from another object, overwriting existing keys + + Inserts all values from from range `[first, last)` and overwrites existing + keys. + + @param[in] first begin of the range of elements to insert + @param[in] last end of the range of elements to insert + + @throw type_error.312 if called on JSON values other than objects; example: + `"cannot use update() with string"` + @throw invalid_iterator.202 if iterator @a first or @a last does does not + point to an object; example: `"iterators first and last must point to + objects"` + @throw invalid_iterator.210 if @a first and @a last do not belong to the + same JSON value; example: `"iterators do not fit"` + + @complexity O(N*log(size() + N)), where N is the number of elements to + insert. + + @liveexample{The example shows how `update()` is used__range.,update} + + @sa https://docs.python.org/3.6/library/stdtypes.html#dict.update + + @since version 3.0.0 + */ + void update(const_iterator first, const_iterator last) + { + // implicitly convert null value to an empty object + if (is_null()) + { + m_type = value_t::object; + m_value.object = create(); + assert_invariant(); + } + + if (JSON_HEDLEY_UNLIKELY(!is_object())) + { + JSON_THROW(type_error::create(312, "cannot use update() with " + std::string(type_name()), *this)); + } + + // check if range iterators belong to the same JSON object + if (JSON_HEDLEY_UNLIKELY(first.m_object != last.m_object)) + { + JSON_THROW(invalid_iterator::create(210, "iterators do not fit", *this)); + } + + // passed iterators must belong to objects + if (JSON_HEDLEY_UNLIKELY(!first.m_object->is_object() + || !last.m_object->is_object())) + { + JSON_THROW(invalid_iterator::create(202, "iterators first and last must point to objects", *this)); + } + + for (auto it = first; it != last; ++it) + { + m_value.object->operator[](it.key()) = it.value(); +#if JSON_DIAGNOSTICS + m_value.object->operator[](it.key()).m_parent = this; +#endif + } + } + + /*! + @brief exchanges the values + + Exchanges the contents of the JSON value with those of @a other. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. + + @param[in,out] other JSON value to exchange the contents with + + @complexity Constant. + + @liveexample{The example below shows how JSON values can be swapped with + `swap()`.,swap__reference} + + @since version 1.0.0 + */ + void swap(reference other) noexcept ( + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value&& + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value + ) + { + std::swap(m_type, other.m_type); + std::swap(m_value, other.m_value); + + set_parents(); + other.set_parents(); + assert_invariant(); + } + + /*! + @brief exchanges the values + + Exchanges the contents of the JSON value from @a left with those of @a right. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. implemented as a friend function callable via ADL. + + @param[in,out] left JSON value to exchange the contents with + @param[in,out] right JSON value to exchange the contents with + + @complexity Constant. + + @liveexample{The example below shows how JSON values can be swapped with + `swap()`.,swap__reference} + + @since version 1.0.0 + */ + friend void swap(reference left, reference right) noexcept ( + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value&& + std::is_nothrow_move_constructible::value&& + std::is_nothrow_move_assignable::value + ) + { + left.swap(right); + } + + /*! + @brief exchanges the values + + Exchanges the contents of a JSON array with those of @a other. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. + + @param[in,out] other array to exchange the contents with + + @throw type_error.310 when JSON value is not an array; example: `"cannot + use swap() with string"` + + @complexity Constant. + + @liveexample{The example below shows how arrays can be swapped with + `swap()`.,swap__array_t} + + @since version 1.0.0 + */ + void swap(array_t& other) // NOLINT(bugprone-exception-escape) + { + // swap only works for arrays + if (JSON_HEDLEY_LIKELY(is_array())) + { + std::swap(*(m_value.array), other); + } + else + { + JSON_THROW(type_error::create(310, "cannot use swap() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief exchanges the values + + Exchanges the contents of a JSON object with those of @a other. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. + + @param[in,out] other object to exchange the contents with + + @throw type_error.310 when JSON value is not an object; example: + `"cannot use swap() with string"` + + @complexity Constant. + + @liveexample{The example below shows how objects can be swapped with + `swap()`.,swap__object_t} + + @since version 1.0.0 + */ + void swap(object_t& other) // NOLINT(bugprone-exception-escape) + { + // swap only works for objects + if (JSON_HEDLEY_LIKELY(is_object())) + { + std::swap(*(m_value.object), other); + } + else + { + JSON_THROW(type_error::create(310, "cannot use swap() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief exchanges the values + + Exchanges the contents of a JSON string with those of @a other. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. + + @param[in,out] other string to exchange the contents with + + @throw type_error.310 when JSON value is not a string; example: `"cannot + use swap() with boolean"` + + @complexity Constant. + + @liveexample{The example below shows how strings can be swapped with + `swap()`.,swap__string_t} + + @since version 1.0.0 + */ + void swap(string_t& other) // NOLINT(bugprone-exception-escape) + { + // swap only works for strings + if (JSON_HEDLEY_LIKELY(is_string())) + { + std::swap(*(m_value.string), other); + } + else + { + JSON_THROW(type_error::create(310, "cannot use swap() with " + std::string(type_name()), *this)); + } + } + + /*! + @brief exchanges the values + + Exchanges the contents of a JSON string with those of @a other. Does not + invoke any move, copy, or swap operations on individual elements. All + iterators and references remain valid. The past-the-end iterator is + invalidated. + + @param[in,out] other binary to exchange the contents with + + @throw type_error.310 when JSON value is not a string; example: `"cannot + use swap() with boolean"` + + @complexity Constant. + + @liveexample{The example below shows how strings can be swapped with + `swap()`.,swap__binary_t} + + @since version 3.8.0 + */ + void swap(binary_t& other) // NOLINT(bugprone-exception-escape) + { + // swap only works for strings + if (JSON_HEDLEY_LIKELY(is_binary())) + { + std::swap(*(m_value.binary), other); + } + else + { + JSON_THROW(type_error::create(310, "cannot use swap() with " + std::string(type_name()), *this)); + } + } + + /// @copydoc swap(binary_t&) + void swap(typename binary_t::container_type& other) // NOLINT(bugprone-exception-escape) + { + // swap only works for strings + if (JSON_HEDLEY_LIKELY(is_binary())) + { + std::swap(*(m_value.binary), other); + } + else + { + JSON_THROW(type_error::create(310, "cannot use swap() with " + std::string(type_name()), *this)); + } + } + + /// @} + + public: + ////////////////////////////////////////// + // lexicographical comparison operators // + ////////////////////////////////////////// + + /// @name lexicographical comparison operators + /// @{ + + /*! + @brief comparison: equal + + Compares two JSON values for equality according to the following rules: + - Two JSON values are equal if (1) they are from the same type and (2) + their stored values are the same according to their respective + `operator==`. + - Integer and floating-point numbers are automatically converted before + comparison. Note that two NaN values are always treated as unequal. + - Two JSON null values are equal. + + @note Floating-point inside JSON values numbers are compared with + `json::number_float_t::operator==` which is `double::operator==` by + default. To compare floating-point while respecting an epsilon, an alternative + [comparison function](https://github.com/mariokonrad/marnav/blob/master/include/marnav/math/floatingpoint.hpp#L34-#L39) + could be used, for instance + @code {.cpp} + template::value, T>::type> + inline bool is_same(T a, T b, T epsilon = std::numeric_limits::epsilon()) noexcept + { + return std::abs(a - b) <= epsilon; + } + @endcode + Or you can self-defined operator equal function like this: + @code {.cpp} + bool my_equal(const_reference lhs, const_reference rhs) { + const auto lhs_type lhs.type(); + const auto rhs_type rhs.type(); + if (lhs_type == rhs_type) { + switch(lhs_type) + // self_defined case + case value_t::number_float: + return std::abs(lhs - rhs) <= std::numeric_limits::epsilon(); + // other cases remain the same with the original + ... + } + ... + } + @endcode + + @note NaN values never compare equal to themselves or to other NaN values. + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether the values @a lhs and @a rhs are equal + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @complexity Linear. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__equal} + + @since version 1.0.0 + */ + friend bool operator==(const_reference lhs, const_reference rhs) noexcept + { +#ifdef __GNUC__ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wfloat-equal" +#endif + const auto lhs_type = lhs.type(); + const auto rhs_type = rhs.type(); + + if (lhs_type == rhs_type) + { + switch (lhs_type) + { + case value_t::array: + return *lhs.m_value.array == *rhs.m_value.array; + + case value_t::object: + return *lhs.m_value.object == *rhs.m_value.object; + + case value_t::null: + return true; + + case value_t::string: + return *lhs.m_value.string == *rhs.m_value.string; + + case value_t::boolean: + return lhs.m_value.boolean == rhs.m_value.boolean; + + case value_t::number_integer: + return lhs.m_value.number_integer == rhs.m_value.number_integer; + + case value_t::number_unsigned: + return lhs.m_value.number_unsigned == rhs.m_value.number_unsigned; + + case value_t::number_float: + return lhs.m_value.number_float == rhs.m_value.number_float; + + case value_t::binary: + return *lhs.m_value.binary == *rhs.m_value.binary; + + case value_t::discarded: + default: + return false; + } + } + else if (lhs_type == value_t::number_integer && rhs_type == value_t::number_float) + { + return static_cast(lhs.m_value.number_integer) == rhs.m_value.number_float; + } + else if (lhs_type == value_t::number_float && rhs_type == value_t::number_integer) + { + return lhs.m_value.number_float == static_cast(rhs.m_value.number_integer); + } + else if (lhs_type == value_t::number_unsigned && rhs_type == value_t::number_float) + { + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_float; + } + else if (lhs_type == value_t::number_float && rhs_type == value_t::number_unsigned) + { + return lhs.m_value.number_float == static_cast(rhs.m_value.number_unsigned); + } + else if (lhs_type == value_t::number_unsigned && rhs_type == value_t::number_integer) + { + return static_cast(lhs.m_value.number_unsigned) == rhs.m_value.number_integer; + } + else if (lhs_type == value_t::number_integer && rhs_type == value_t::number_unsigned) + { + return lhs.m_value.number_integer == static_cast(rhs.m_value.number_unsigned); + } + + return false; +#ifdef __GNUC__ +#pragma GCC diagnostic pop +#endif + } + + /*! + @brief comparison: equal + @copydoc operator==(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator==(const_reference lhs, ScalarType rhs) noexcept + { + return lhs == basic_json(rhs); + } + + /*! + @brief comparison: equal + @copydoc operator==(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator==(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) == rhs; + } + + /*! + @brief comparison: not equal + + Compares two JSON values for inequality by calculating `not (lhs == rhs)`. + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether the values @a lhs and @a rhs are not equal + + @complexity Linear. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__notequal} + + @since version 1.0.0 + */ + friend bool operator!=(const_reference lhs, const_reference rhs) noexcept + { + return !(lhs == rhs); + } + + /*! + @brief comparison: not equal + @copydoc operator!=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator!=(const_reference lhs, ScalarType rhs) noexcept + { + return lhs != basic_json(rhs); + } + + /*! + @brief comparison: not equal + @copydoc operator!=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator!=(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) != rhs; + } + + /*! + @brief comparison: less than + + Compares whether one JSON value @a lhs is less than another JSON value @a + rhs according to the following rules: + - If @a lhs and @a rhs have the same type, the values are compared using + the default `<` operator. + - Integer and floating-point numbers are automatically converted before + comparison + - In case @a lhs and @a rhs have different types, the values are ignored + and the order of the types is considered, see + @ref operator<(const value_t, const value_t). + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether @a lhs is less than @a rhs + + @complexity Linear. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__less} + + @since version 1.0.0 + */ + friend bool operator<(const_reference lhs, const_reference rhs) noexcept + { + const auto lhs_type = lhs.type(); + const auto rhs_type = rhs.type(); + + if (lhs_type == rhs_type) + { + switch (lhs_type) + { + case value_t::array: + // note parentheses are necessary, see + // https://github.com/nlohmann/json/issues/1530 + return (*lhs.m_value.array) < (*rhs.m_value.array); + + case value_t::object: + return (*lhs.m_value.object) < (*rhs.m_value.object); + + case value_t::null: + return false; + + case value_t::string: + return (*lhs.m_value.string) < (*rhs.m_value.string); + + case value_t::boolean: + return (lhs.m_value.boolean) < (rhs.m_value.boolean); + + case value_t::number_integer: + return (lhs.m_value.number_integer) < (rhs.m_value.number_integer); + + case value_t::number_unsigned: + return (lhs.m_value.number_unsigned) < (rhs.m_value.number_unsigned); + + case value_t::number_float: + return (lhs.m_value.number_float) < (rhs.m_value.number_float); + + case value_t::binary: + return (*lhs.m_value.binary) < (*rhs.m_value.binary); + + case value_t::discarded: + default: + return false; + } + } + else if (lhs_type == value_t::number_integer && rhs_type == value_t::number_float) + { + return static_cast(lhs.m_value.number_integer) < rhs.m_value.number_float; + } + else if (lhs_type == value_t::number_float && rhs_type == value_t::number_integer) + { + return lhs.m_value.number_float < static_cast(rhs.m_value.number_integer); + } + else if (lhs_type == value_t::number_unsigned && rhs_type == value_t::number_float) + { + return static_cast(lhs.m_value.number_unsigned) < rhs.m_value.number_float; + } + else if (lhs_type == value_t::number_float && rhs_type == value_t::number_unsigned) + { + return lhs.m_value.number_float < static_cast(rhs.m_value.number_unsigned); + } + else if (lhs_type == value_t::number_integer && rhs_type == value_t::number_unsigned) + { + return lhs.m_value.number_integer < static_cast(rhs.m_value.number_unsigned); + } + else if (lhs_type == value_t::number_unsigned && rhs_type == value_t::number_integer) + { + return static_cast(lhs.m_value.number_unsigned) < rhs.m_value.number_integer; + } + + // We only reach this line if we cannot compare values. In that case, + // we compare types. Note we have to call the operator explicitly, + // because MSVC has problems otherwise. + return operator<(lhs_type, rhs_type); + } + + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(const_reference lhs, ScalarType rhs) noexcept + { + return lhs < basic_json(rhs); + } + + /*! + @brief comparison: less than + @copydoc operator<(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) < rhs; + } + + /*! + @brief comparison: less than or equal + + Compares whether one JSON value @a lhs is less than or equal to another + JSON value by calculating `not (rhs < lhs)`. + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether @a lhs is less than or equal to @a rhs + + @complexity Linear. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__greater} + + @since version 1.0.0 + */ + friend bool operator<=(const_reference lhs, const_reference rhs) noexcept + { + return !(rhs < lhs); + } + + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(const_reference lhs, ScalarType rhs) noexcept + { + return lhs <= basic_json(rhs); + } + + /*! + @brief comparison: less than or equal + @copydoc operator<=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator<=(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) <= rhs; + } + + /*! + @brief comparison: greater than + + Compares whether one JSON value @a lhs is greater than another + JSON value by calculating `not (lhs <= rhs)`. + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether @a lhs is greater than to @a rhs + + @complexity Linear. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__lessequal} + + @since version 1.0.0 + */ + friend bool operator>(const_reference lhs, const_reference rhs) noexcept + { + return !(lhs <= rhs); + } + + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(const_reference lhs, ScalarType rhs) noexcept + { + return lhs > basic_json(rhs); + } + + /*! + @brief comparison: greater than + @copydoc operator>(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) > rhs; + } + + /*! + @brief comparison: greater than or equal + + Compares whether one JSON value @a lhs is greater than or equal to another + JSON value by calculating `not (lhs < rhs)`. + + @param[in] lhs first JSON value to consider + @param[in] rhs second JSON value to consider + @return whether @a lhs is greater than or equal to @a rhs + + @complexity Linear. + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @liveexample{The example demonstrates comparing several JSON + types.,operator__greaterequal} + + @since version 1.0.0 + */ + friend bool operator>=(const_reference lhs, const_reference rhs) noexcept + { + return !(lhs < rhs); + } + + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(const_reference lhs, ScalarType rhs) noexcept + { + return lhs >= basic_json(rhs); + } + + /*! + @brief comparison: greater than or equal + @copydoc operator>=(const_reference, const_reference) + */ + template::value, int>::type = 0> + friend bool operator>=(ScalarType lhs, const_reference rhs) noexcept + { + return basic_json(lhs) >= rhs; + } + + /// @} + + /////////////////// + // serialization // + /////////////////// + + /// @name serialization + /// @{ +#ifndef JSON_NO_IO + /*! + @brief serialize to stream + + Serialize the given JSON value @a j to the output stream @a o. The JSON + value will be serialized using the @ref dump member function. + + - The indentation of the output can be controlled with the member variable + `width` of the output stream @a o. For instance, using the manipulator + `std::setw(4)` on @a o sets the indentation level to `4` and the + serialization result is the same as calling `dump(4)`. + + - The indentation character can be controlled with the member variable + `fill` of the output stream @a o. For instance, the manipulator + `std::setfill('\\t')` sets indentation to use a tab character rather than + the default space character. + + @param[in,out] o stream to serialize to + @param[in] j JSON value to serialize + + @return the stream @a o + + @throw type_error.316 if a string stored inside the JSON value is not + UTF-8 encoded + + @complexity Linear. + + @liveexample{The example below shows the serialization with different + parameters to `width` to adjust the indentation level.,operator_serialize} + + @since version 1.0.0; indentation character added in version 3.0.0 + */ + friend std::ostream& operator<<(std::ostream& o, const basic_json& j) + { + // read width member and use it as indentation parameter if nonzero + const bool pretty_print = o.width() > 0; + const auto indentation = pretty_print ? o.width() : 0; + + // reset width to 0 for subsequent calls to this stream + o.width(0); + + // do the actual serialization + serializer s(detail::output_adapter(o), o.fill()); + s.dump(j, pretty_print, false, static_cast(indentation)); + return o; + } + + /*! + @brief serialize to stream + @deprecated This stream operator is deprecated and will be removed in + future 4.0.0 of the library. Please use + @ref operator<<(std::ostream&, const basic_json&) + instead; that is, replace calls like `j >> o;` with `o << j;`. + @since version 1.0.0; deprecated since version 3.0.0 + */ + JSON_HEDLEY_DEPRECATED_FOR(3.0.0, operator<<(std::ostream&, const basic_json&)) + friend std::ostream& operator>>(const basic_json& j, std::ostream& o) + { + return o << j; + } +#endif // JSON_NO_IO + /// @} + + + ///////////////////// + // deserialization // + ///////////////////// + + /// @name deserialization + /// @{ + + /*! + @brief deserialize from a compatible input + + @tparam InputType A compatible input, for instance + - an std::istream object + - a FILE pointer + - a C-style array of characters + - a pointer to a null-terminated string of single byte characters + - an object obj for which begin(obj) and end(obj) produces a valid pair of + iterators. + + @param[in] i input to read from + @param[in] cb a parser callback function of type @ref parser_callback_t + which is used to control the deserialization by filtering unwanted values + (optional) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + @param[in] ignore_comments whether comments should be ignored and treated + like whitespace (true) or yield a parse error (true); (optional, false by + default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.101 if a parse error occurs; example: `""unexpected end + of input; expected string literal""` + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + + @complexity Linear in the length of the input. The parser is a predictive + LL(1) parser. The complexity can be higher if the parser callback function + @a cb or reading from the input @a i has a super-linear complexity. + + @note A UTF-8 byte order mark is silently ignored. + + @liveexample{The example below demonstrates the `parse()` function reading + from an array.,parse__array__parser_callback_t} + + @liveexample{The example below demonstrates the `parse()` function with + and without callback function.,parse__string__parser_callback_t} + + @liveexample{The example below demonstrates the `parse()` function with + and without callback function.,parse__istream__parser_callback_t} + + @liveexample{The example below demonstrates the `parse()` function reading + from a contiguous container.,parse__contiguouscontainer__parser_callback_t} + + @since version 2.0.3 (contiguous containers); version 3.9.0 allowed to + ignore comments. + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json parse(InputType&& i, + const parser_callback_t cb = nullptr, + const bool allow_exceptions = true, + const bool ignore_comments = false) + { + basic_json result; + parser(detail::input_adapter(std::forward(i)), cb, allow_exceptions, ignore_comments).parse(true, result); + return result; + } + + /*! + @brief deserialize from a pair of character iterators + + The value_type of the iterator must be a integral type with size of 1, 2 or + 4 bytes, which will be interpreted respectively as UTF-8, UTF-16 and UTF-32. + + @param[in] first iterator to start of character range + @param[in] last iterator to end of character range + @param[in] cb a parser callback function of type @ref parser_callback_t + which is used to control the deserialization by filtering unwanted values + (optional) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + @param[in] ignore_comments whether comments should be ignored and treated + like whitespace (true) or yield a parse error (true); (optional, false by + default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.101 if a parse error occurs; example: `""unexpected end + of input; expected string literal""` + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json parse(IteratorType first, + IteratorType last, + const parser_callback_t cb = nullptr, + const bool allow_exceptions = true, + const bool ignore_comments = false) + { + basic_json result; + parser(detail::input_adapter(std::move(first), std::move(last)), cb, allow_exceptions, ignore_comments).parse(true, result); + return result; + } + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, parse(ptr, ptr + len)) + static basic_json parse(detail::span_input_adapter&& i, + const parser_callback_t cb = nullptr, + const bool allow_exceptions = true, + const bool ignore_comments = false) + { + basic_json result; + parser(i.get(), cb, allow_exceptions, ignore_comments).parse(true, result); + return result; + } + + /*! + @brief check if the input is valid JSON + + Unlike the @ref parse(InputType&&, const parser_callback_t,const bool) + function, this function neither throws an exception in case of invalid JSON + input (i.e., a parse error) nor creates diagnostic information. + + @tparam InputType A compatible input, for instance + - an std::istream object + - a FILE pointer + - a C-style array of characters + - a pointer to a null-terminated string of single byte characters + - an object obj for which begin(obj) and end(obj) produces a valid pair of + iterators. + + @param[in] i input to read from + @param[in] ignore_comments whether comments should be ignored and treated + like whitespace (true) or yield a parse error (true); (optional, false by + default) + + @return Whether the input read from @a i is valid JSON. + + @complexity Linear in the length of the input. The parser is a predictive + LL(1) parser. + + @note A UTF-8 byte order mark is silently ignored. + + @liveexample{The example below demonstrates the `accept()` function reading + from a string.,accept__string} + */ + template + static bool accept(InputType&& i, + const bool ignore_comments = false) + { + return parser(detail::input_adapter(std::forward(i)), nullptr, false, ignore_comments).accept(true); + } + + template + static bool accept(IteratorType first, IteratorType last, + const bool ignore_comments = false) + { + return parser(detail::input_adapter(std::move(first), std::move(last)), nullptr, false, ignore_comments).accept(true); + } + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, accept(ptr, ptr + len)) + static bool accept(detail::span_input_adapter&& i, + const bool ignore_comments = false) + { + return parser(i.get(), nullptr, false, ignore_comments).accept(true); + } + + /*! + @brief generate SAX events + + The SAX event lister must follow the interface of @ref json_sax. + + This function reads from a compatible input. Examples are: + - an std::istream object + - a FILE pointer + - a C-style array of characters + - a pointer to a null-terminated string of single byte characters + - an object obj for which begin(obj) and end(obj) produces a valid pair of + iterators. + + @param[in] i input to read from + @param[in,out] sax SAX event listener + @param[in] format the format to parse (JSON, CBOR, MessagePack, or UBJSON) + @param[in] strict whether the input has to be consumed completely + @param[in] ignore_comments whether comments should be ignored and treated + like whitespace (true) or yield a parse error (true); (optional, false by + default); only applies to the JSON file format. + + @return return value of the last processed SAX event + + @throw parse_error.101 if a parse error occurs; example: `""unexpected end + of input; expected string literal""` + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + + @complexity Linear in the length of the input. The parser is a predictive + LL(1) parser. The complexity can be higher if the SAX consumer @a sax has + a super-linear complexity. + + @note A UTF-8 byte order mark is silently ignored. + + @liveexample{The example below demonstrates the `sax_parse()` function + reading from string and processing the events with a user-defined SAX + event consumer.,sax_parse} + + @since version 3.2.0 + */ + template + JSON_HEDLEY_NON_NULL(2) + static bool sax_parse(InputType&& i, SAX* sax, + input_format_t format = input_format_t::json, + const bool strict = true, + const bool ignore_comments = false) + { + auto ia = detail::input_adapter(std::forward(i)); + return format == input_format_t::json + ? parser(std::move(ia), nullptr, true, ignore_comments).sax_parse(sax, strict) + : detail::binary_reader(std::move(ia)).sax_parse(format, sax, strict); + } + + template + JSON_HEDLEY_NON_NULL(3) + static bool sax_parse(IteratorType first, IteratorType last, SAX* sax, + input_format_t format = input_format_t::json, + const bool strict = true, + const bool ignore_comments = false) + { + auto ia = detail::input_adapter(std::move(first), std::move(last)); + return format == input_format_t::json + ? parser(std::move(ia), nullptr, true, ignore_comments).sax_parse(sax, strict) + : detail::binary_reader(std::move(ia)).sax_parse(format, sax, strict); + } + + template + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, sax_parse(ptr, ptr + len, ...)) + JSON_HEDLEY_NON_NULL(2) + static bool sax_parse(detail::span_input_adapter&& i, SAX* sax, + input_format_t format = input_format_t::json, + const bool strict = true, + const bool ignore_comments = false) + { + auto ia = i.get(); + return format == input_format_t::json + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + ? parser(std::move(ia), nullptr, true, ignore_comments).sax_parse(sax, strict) + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + : detail::binary_reader(std::move(ia)).sax_parse(format, sax, strict); + } +#ifndef JSON_NO_IO + /*! + @brief deserialize from stream + @deprecated This stream operator is deprecated and will be removed in + version 4.0.0 of the library. Please use + @ref operator>>(std::istream&, basic_json&) + instead; that is, replace calls like `j << i;` with `i >> j;`. + @since version 1.0.0; deprecated since version 3.0.0 + */ + JSON_HEDLEY_DEPRECATED_FOR(3.0.0, operator>>(std::istream&, basic_json&)) + friend std::istream& operator<<(basic_json& j, std::istream& i) + { + return operator>>(i, j); + } + + /*! + @brief deserialize from stream + + Deserializes an input stream to a JSON value. + + @param[in,out] i input stream to read a serialized JSON value from + @param[in,out] j JSON value to write the deserialized input to + + @throw parse_error.101 in case of an unexpected token + @throw parse_error.102 if to_unicode fails or surrogate error + @throw parse_error.103 if to_unicode fails + + @complexity Linear in the length of the input. The parser is a predictive + LL(1) parser. + + @note A UTF-8 byte order mark is silently ignored. + + @liveexample{The example below shows how a JSON value is constructed by + reading a serialization from a stream.,operator_deserialize} + + @sa parse(std::istream&, const parser_callback_t) for a variant with a + parser callback function to filter values while parsing + + @since version 1.0.0 + */ + friend std::istream& operator>>(std::istream& i, basic_json& j) + { + parser(detail::input_adapter(i)).parse(false, j); + return i; + } +#endif // JSON_NO_IO + /// @} + + /////////////////////////// + // convenience functions // + /////////////////////////// + + /*! + @brief return the type as string + + Returns the type name as string to be used in error messages - usually to + indicate that a function was called on a wrong JSON type. + + @return a string representation of a the @a m_type member: + Value type | return value + ----------- | ------------- + null | `"null"` + boolean | `"boolean"` + string | `"string"` + number | `"number"` (for all number types) + object | `"object"` + array | `"array"` + binary | `"binary"` + discarded | `"discarded"` + + @exceptionsafety No-throw guarantee: this function never throws exceptions. + + @complexity Constant. + + @liveexample{The following code exemplifies `type_name()` for all JSON + types.,type_name} + + @sa see @ref type() -- return the type of the JSON value + @sa see @ref operator value_t() -- return the type of the JSON value (implicit) + + @since version 1.0.0, public since 2.1.0, `const char*` and `noexcept` + since 3.0.0 + */ + JSON_HEDLEY_RETURNS_NON_NULL + const char* type_name() const noexcept + { + { + switch (m_type) + { + case value_t::null: + return "null"; + case value_t::object: + return "object"; + case value_t::array: + return "array"; + case value_t::string: + return "string"; + case value_t::boolean: + return "boolean"; + case value_t::binary: + return "binary"; + case value_t::discarded: + return "discarded"; + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + default: + return "number"; + } + } + } + + + JSON_PRIVATE_UNLESS_TESTED: + ////////////////////// + // member variables // + ////////////////////// + + /// the type of the current element + value_t m_type = value_t::null; + + /// the value of the current element + json_value m_value = {}; + +#if JSON_DIAGNOSTICS + /// a pointer to a parent value (for debugging purposes) + basic_json* m_parent = nullptr; +#endif + + ////////////////////////////////////////// + // binary serialization/deserialization // + ////////////////////////////////////////// + + /// @name binary serialization/deserialization support + /// @{ + + public: + /*! + @brief create a CBOR serialization of a given JSON value + + Serializes a given JSON value @a j to a byte vector using the CBOR (Concise + Binary Object Representation) serialization format. CBOR is a binary + serialization format which aims to be more compact than JSON itself, yet + more efficient to parse. + + The library uses the following mapping from JSON values types to + CBOR types according to the CBOR specification (RFC 7049): + + JSON value type | value/range | CBOR type | first byte + --------------- | ------------------------------------------ | ---------------------------------- | --------------- + null | `null` | Null | 0xF6 + boolean | `true` | True | 0xF5 + boolean | `false` | False | 0xF4 + number_integer | -9223372036854775808..-2147483649 | Negative integer (8 bytes follow) | 0x3B + number_integer | -2147483648..-32769 | Negative integer (4 bytes follow) | 0x3A + number_integer | -32768..-129 | Negative integer (2 bytes follow) | 0x39 + number_integer | -128..-25 | Negative integer (1 byte follow) | 0x38 + number_integer | -24..-1 | Negative integer | 0x20..0x37 + number_integer | 0..23 | Integer | 0x00..0x17 + number_integer | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_integer | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_integer | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1A + number_integer | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1B + number_unsigned | 0..23 | Integer | 0x00..0x17 + number_unsigned | 24..255 | Unsigned integer (1 byte follow) | 0x18 + number_unsigned | 256..65535 | Unsigned integer (2 bytes follow) | 0x19 + number_unsigned | 65536..4294967295 | Unsigned integer (4 bytes follow) | 0x1A + number_unsigned | 4294967296..18446744073709551615 | Unsigned integer (8 bytes follow) | 0x1B + number_float | *any value representable by a float* | Single-Precision Float | 0xFA + number_float | *any value NOT representable by a float* | Double-Precision Float | 0xFB + string | *length*: 0..23 | UTF-8 string | 0x60..0x77 + string | *length*: 23..255 | UTF-8 string (1 byte follow) | 0x78 + string | *length*: 256..65535 | UTF-8 string (2 bytes follow) | 0x79 + string | *length*: 65536..4294967295 | UTF-8 string (4 bytes follow) | 0x7A + string | *length*: 4294967296..18446744073709551615 | UTF-8 string (8 bytes follow) | 0x7B + array | *size*: 0..23 | array | 0x80..0x97 + array | *size*: 23..255 | array (1 byte follow) | 0x98 + array | *size*: 256..65535 | array (2 bytes follow) | 0x99 + array | *size*: 65536..4294967295 | array (4 bytes follow) | 0x9A + array | *size*: 4294967296..18446744073709551615 | array (8 bytes follow) | 0x9B + object | *size*: 0..23 | map | 0xA0..0xB7 + object | *size*: 23..255 | map (1 byte follow) | 0xB8 + object | *size*: 256..65535 | map (2 bytes follow) | 0xB9 + object | *size*: 65536..4294967295 | map (4 bytes follow) | 0xBA + object | *size*: 4294967296..18446744073709551615 | map (8 bytes follow) | 0xBB + binary | *size*: 0..23 | byte string | 0x40..0x57 + binary | *size*: 23..255 | byte string (1 byte follow) | 0x58 + binary | *size*: 256..65535 | byte string (2 bytes follow) | 0x59 + binary | *size*: 65536..4294967295 | byte string (4 bytes follow) | 0x5A + binary | *size*: 4294967296..18446744073709551615 | byte string (8 bytes follow) | 0x5B + + Binary values with subtype are mapped to tagged values (0xD8..0xDB) + depending on the subtype, followed by a byte string, see "binary" cells + in the table above. + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a CBOR value. + + @note If NaN or Infinity are stored inside a JSON number, they are + serialized properly. This behavior differs from the @ref dump() + function which serializes NaN or Infinity to `null`. + + @note The following CBOR types are not used in the conversion: + - UTF-8 strings terminated by "break" (0x7F) + - arrays terminated by "break" (0x9F) + - maps terminated by "break" (0xBF) + - byte strings terminated by "break" (0x5F) + - date/time (0xC0..0xC1) + - bignum (0xC2..0xC3) + - decimal fraction (0xC4) + - bigfloat (0xC5) + - expected conversions (0xD5..0xD7) + - simple values (0xE0..0xF3, 0xF8) + - undefined (0xF7) + - half-precision floats (0xF9) + - break (0xFF) + + @param[in] j JSON value to serialize + @return CBOR serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in CBOR format.,to_cbor} + + @sa http://cbor.io + @sa see @ref from_cbor(InputType&&, const bool, const bool, const cbor_tag_handler_t) for the + analogous deserialization + @sa see @ref to_msgpack(const basic_json&) for the related MessagePack format + @sa see @ref to_ubjson(const basic_json&, const bool, const bool) for the + related UBJSON format + + @since version 2.0.9; compact representation of floating-point numbers + since version 3.8.0 + */ + static std::vector to_cbor(const basic_json& j) + { + std::vector result; + to_cbor(j, result); + return result; + } + + static void to_cbor(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_cbor(j); + } + + static void to_cbor(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_cbor(j); + } + + /*! + @brief create a MessagePack serialization of a given JSON value + + Serializes a given JSON value @a j to a byte vector using the MessagePack + serialization format. MessagePack is a binary serialization format which + aims to be more compact than JSON itself, yet more efficient to parse. + + The library uses the following mapping from JSON values types to + MessagePack types according to the MessagePack specification: + + JSON value type | value/range | MessagePack type | first byte + --------------- | --------------------------------- | ---------------- | ---------- + null | `null` | nil | 0xC0 + boolean | `true` | true | 0xC3 + boolean | `false` | false | 0xC2 + number_integer | -9223372036854775808..-2147483649 | int64 | 0xD3 + number_integer | -2147483648..-32769 | int32 | 0xD2 + number_integer | -32768..-129 | int16 | 0xD1 + number_integer | -128..-33 | int8 | 0xD0 + number_integer | -32..-1 | negative fixint | 0xE0..0xFF + number_integer | 0..127 | positive fixint | 0x00..0x7F + number_integer | 128..255 | uint 8 | 0xCC + number_integer | 256..65535 | uint 16 | 0xCD + number_integer | 65536..4294967295 | uint 32 | 0xCE + number_integer | 4294967296..18446744073709551615 | uint 64 | 0xCF + number_unsigned | 0..127 | positive fixint | 0x00..0x7F + number_unsigned | 128..255 | uint 8 | 0xCC + number_unsigned | 256..65535 | uint 16 | 0xCD + number_unsigned | 65536..4294967295 | uint 32 | 0xCE + number_unsigned | 4294967296..18446744073709551615 | uint 64 | 0xCF + number_float | *any value representable by a float* | float 32 | 0xCA + number_float | *any value NOT representable by a float* | float 64 | 0xCB + string | *length*: 0..31 | fixstr | 0xA0..0xBF + string | *length*: 32..255 | str 8 | 0xD9 + string | *length*: 256..65535 | str 16 | 0xDA + string | *length*: 65536..4294967295 | str 32 | 0xDB + array | *size*: 0..15 | fixarray | 0x90..0x9F + array | *size*: 16..65535 | array 16 | 0xDC + array | *size*: 65536..4294967295 | array 32 | 0xDD + object | *size*: 0..15 | fix map | 0x80..0x8F + object | *size*: 16..65535 | map 16 | 0xDE + object | *size*: 65536..4294967295 | map 32 | 0xDF + binary | *size*: 0..255 | bin 8 | 0xC4 + binary | *size*: 256..65535 | bin 16 | 0xC5 + binary | *size*: 65536..4294967295 | bin 32 | 0xC6 + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a MessagePack value. + + @note The following values can **not** be converted to a MessagePack value: + - strings with more than 4294967295 bytes + - byte strings with more than 4294967295 bytes + - arrays with more than 4294967295 elements + - objects with more than 4294967295 elements + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + + @note If NaN or Infinity are stored inside a JSON number, they are + serialized properly. This behavior differs from the @ref dump() + function which serializes NaN or Infinity to `null`. + + @param[in] j JSON value to serialize + @return MessagePack serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in MessagePack format.,to_msgpack} + + @sa http://msgpack.org + @sa see @ref from_msgpack for the analogous deserialization + @sa see @ref to_cbor(const basic_json& for the related CBOR format + @sa see @ref to_ubjson(const basic_json&, const bool, const bool) for the + related UBJSON format + + @since version 2.0.9 + */ + static std::vector to_msgpack(const basic_json& j) + { + std::vector result; + to_msgpack(j, result); + return result; + } + + static void to_msgpack(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_msgpack(j); + } + + static void to_msgpack(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_msgpack(j); + } + + /*! + @brief create a UBJSON serialization of a given JSON value + + Serializes a given JSON value @a j to a byte vector using the UBJSON + (Universal Binary JSON) serialization format. UBJSON aims to be more compact + than JSON itself, yet more efficient to parse. + + The library uses the following mapping from JSON values types to + UBJSON types according to the UBJSON specification: + + JSON value type | value/range | UBJSON type | marker + --------------- | --------------------------------- | ----------- | ------ + null | `null` | null | `Z` + boolean | `true` | true | `T` + boolean | `false` | false | `F` + number_integer | -9223372036854775808..-2147483649 | int64 | `L` + number_integer | -2147483648..-32769 | int32 | `l` + number_integer | -32768..-129 | int16 | `I` + number_integer | -128..127 | int8 | `i` + number_integer | 128..255 | uint8 | `U` + number_integer | 256..32767 | int16 | `I` + number_integer | 32768..2147483647 | int32 | `l` + number_integer | 2147483648..9223372036854775807 | int64 | `L` + number_unsigned | 0..127 | int8 | `i` + number_unsigned | 128..255 | uint8 | `U` + number_unsigned | 256..32767 | int16 | `I` + number_unsigned | 32768..2147483647 | int32 | `l` + number_unsigned | 2147483648..9223372036854775807 | int64 | `L` + number_unsigned | 2147483649..18446744073709551615 | high-precision | `H` + number_float | *any value* | float64 | `D` + string | *with shortest length indicator* | string | `S` + array | *see notes on optimized format* | array | `[` + object | *see notes on optimized format* | map | `{` + + @note The mapping is **complete** in the sense that any JSON value type + can be converted to a UBJSON value. + + @note The following values can **not** be converted to a UBJSON value: + - strings with more than 9223372036854775807 bytes (theoretical) + + @note The following markers are not used in the conversion: + - `Z`: no-op values are not created. + - `C`: single-byte strings are serialized with `S` markers. + + @note Any UBJSON output created @ref to_ubjson can be successfully parsed + by @ref from_ubjson. + + @note If NaN or Infinity are stored inside a JSON number, they are + serialized properly. This behavior differs from the @ref dump() + function which serializes NaN or Infinity to `null`. + + @note The optimized formats for containers are supported: Parameter + @a use_size adds size information to the beginning of a container and + removes the closing marker. Parameter @a use_type further checks + whether all elements of a container have the same type and adds the + type marker to the beginning of the container. The @a use_type + parameter must only be used together with @a use_size = true. Note + that @a use_size = true alone may result in larger representations - + the benefit of this parameter is that the receiving side is + immediately informed on the number of elements of the container. + + @note If the JSON data contains the binary type, the value stored is a list + of integers, as suggested by the UBJSON documentation. In particular, + this means that serialization and the deserialization of a JSON + containing binary values into UBJSON and back will result in a + different JSON object. + + @param[in] j JSON value to serialize + @param[in] use_size whether to add size annotations to container types + @param[in] use_type whether to add type annotations to container types + (must be combined with @a use_size = true) + @return UBJSON serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in UBJSON format.,to_ubjson} + + @sa http://ubjson.org + @sa see @ref from_ubjson(InputType&&, const bool, const bool) for the + analogous deserialization + @sa see @ref to_cbor(const basic_json& for the related CBOR format + @sa see @ref to_msgpack(const basic_json&) for the related MessagePack format + + @since version 3.1.0 + */ + static std::vector to_ubjson(const basic_json& j, + const bool use_size = false, + const bool use_type = false) + { + std::vector result; + to_ubjson(j, result, use_size, use_type); + return result; + } + + static void to_ubjson(const basic_json& j, detail::output_adapter o, + const bool use_size = false, const bool use_type = false) + { + binary_writer(o).write_ubjson(j, use_size, use_type); + } + + static void to_ubjson(const basic_json& j, detail::output_adapter o, + const bool use_size = false, const bool use_type = false) + { + binary_writer(o).write_ubjson(j, use_size, use_type); + } + + + /*! + @brief Serializes the given JSON object `j` to BSON and returns a vector + containing the corresponding BSON-representation. + + BSON (Binary JSON) is a binary format in which zero or more ordered key/value pairs are + stored as a single entity (a so-called document). + + The library uses the following mapping from JSON values types to BSON types: + + JSON value type | value/range | BSON type | marker + --------------- | --------------------------------- | ----------- | ------ + null | `null` | null | 0x0A + boolean | `true`, `false` | boolean | 0x08 + number_integer | -9223372036854775808..-2147483649 | int64 | 0x12 + number_integer | -2147483648..2147483647 | int32 | 0x10 + number_integer | 2147483648..9223372036854775807 | int64 | 0x12 + number_unsigned | 0..2147483647 | int32 | 0x10 + number_unsigned | 2147483648..9223372036854775807 | int64 | 0x12 + number_unsigned | 9223372036854775808..18446744073709551615| -- | -- + number_float | *any value* | double | 0x01 + string | *any value* | string | 0x02 + array | *any value* | document | 0x04 + object | *any value* | document | 0x03 + binary | *any value* | binary | 0x05 + + @warning The mapping is **incomplete**, since only JSON-objects (and things + contained therein) can be serialized to BSON. + Also, integers larger than 9223372036854775807 cannot be serialized to BSON, + and the keys may not contain U+0000, since they are serialized a + zero-terminated c-strings. + + @throw out_of_range.407 if `j.is_number_unsigned() && j.get() > 9223372036854775807` + @throw out_of_range.409 if a key in `j` contains a NULL (U+0000) + @throw type_error.317 if `!j.is_object()` + + @pre The input `j` is required to be an object: `j.is_object() == true`. + + @note Any BSON output created via @ref to_bson can be successfully parsed + by @ref from_bson. + + @param[in] j JSON value to serialize + @return BSON serialization as byte vector + + @complexity Linear in the size of the JSON value @a j. + + @liveexample{The example shows the serialization of a JSON value to a byte + vector in BSON format.,to_bson} + + @sa http://bsonspec.org/spec.html + @sa see @ref from_bson(detail::input_adapter&&, const bool strict) for the + analogous deserialization + @sa see @ref to_ubjson(const basic_json&, const bool, const bool) for the + related UBJSON format + @sa see @ref to_cbor(const basic_json&) for the related CBOR format + @sa see @ref to_msgpack(const basic_json&) for the related MessagePack format + */ + static std::vector to_bson(const basic_json& j) + { + std::vector result; + to_bson(j, result); + return result; + } + + /*! + @brief Serializes the given JSON object `j` to BSON and forwards the + corresponding BSON-representation to the given output_adapter `o`. + @param j The JSON object to convert to BSON. + @param o The output adapter that receives the binary BSON representation. + @pre The input `j` shall be an object: `j.is_object() == true` + @sa see @ref to_bson(const basic_json&) + */ + static void to_bson(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_bson(j); + } + + /*! + @copydoc to_bson(const basic_json&, detail::output_adapter) + */ + static void to_bson(const basic_json& j, detail::output_adapter o) + { + binary_writer(o).write_bson(j); + } + + + /*! + @brief create a JSON value from an input in CBOR format + + Deserializes a given input @a i to a JSON value using the CBOR (Concise + Binary Object Representation) serialization format. + + The library maps CBOR types to JSON value types as follows: + + CBOR type | JSON value type | first byte + ---------------------- | --------------- | ---------- + Integer | number_unsigned | 0x00..0x17 + Unsigned integer | number_unsigned | 0x18 + Unsigned integer | number_unsigned | 0x19 + Unsigned integer | number_unsigned | 0x1A + Unsigned integer | number_unsigned | 0x1B + Negative integer | number_integer | 0x20..0x37 + Negative integer | number_integer | 0x38 + Negative integer | number_integer | 0x39 + Negative integer | number_integer | 0x3A + Negative integer | number_integer | 0x3B + Byte string | binary | 0x40..0x57 + Byte string | binary | 0x58 + Byte string | binary | 0x59 + Byte string | binary | 0x5A + Byte string | binary | 0x5B + UTF-8 string | string | 0x60..0x77 + UTF-8 string | string | 0x78 + UTF-8 string | string | 0x79 + UTF-8 string | string | 0x7A + UTF-8 string | string | 0x7B + UTF-8 string | string | 0x7F + array | array | 0x80..0x97 + array | array | 0x98 + array | array | 0x99 + array | array | 0x9A + array | array | 0x9B + array | array | 0x9F + map | object | 0xA0..0xB7 + map | object | 0xB8 + map | object | 0xB9 + map | object | 0xBA + map | object | 0xBB + map | object | 0xBF + False | `false` | 0xF4 + True | `true` | 0xF5 + Null | `null` | 0xF6 + Half-Precision Float | number_float | 0xF9 + Single-Precision Float | number_float | 0xFA + Double-Precision Float | number_float | 0xFB + + @warning The mapping is **incomplete** in the sense that not all CBOR + types can be converted to a JSON value. The following CBOR types + are not supported and will yield parse errors (parse_error.112): + - date/time (0xC0..0xC1) + - bignum (0xC2..0xC3) + - decimal fraction (0xC4) + - bigfloat (0xC5) + - expected conversions (0xD5..0xD7) + - simple values (0xE0..0xF3, 0xF8) + - undefined (0xF7) + + @warning CBOR allows map keys of any type, whereas JSON only allows + strings as keys in object values. Therefore, CBOR maps with keys + other than UTF-8 strings are rejected (parse_error.113). + + @note Any CBOR output created @ref to_cbor can be successfully parsed by + @ref from_cbor. + + @param[in] i an input in CBOR format convertible to an input adapter + @param[in] strict whether to expect the input to be consumed until EOF + (true by default) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + @param[in] tag_handler how to treat CBOR tags (optional, error by default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.110 if the given input ends prematurely or the end of + file was not reached when @a strict was set to true + @throw parse_error.112 if unsupported features from CBOR were + used in the given input @a v or if the input is not valid CBOR + @throw parse_error.113 if a string was expected as map key, but not found + + @complexity Linear in the size of the input @a i. + + @liveexample{The example shows the deserialization of a byte vector in CBOR + format to a JSON value.,from_cbor} + + @sa http://cbor.io + @sa see @ref to_cbor(const basic_json&) for the analogous serialization + @sa see @ref from_msgpack(InputType&&, const bool, const bool) for the + related MessagePack format + @sa see @ref from_ubjson(InputType&&, const bool, const bool) for the + related UBJSON format + + @since version 2.0.9; parameter @a start_index since 2.1.1; changed to + consume input adapters, removed start_index parameter, and added + @a strict parameter since 3.0.0; added @a allow_exceptions parameter + since 3.2.0; added @a tag_handler parameter since 3.9.0. + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_cbor(InputType&& i, + const bool strict = true, + const bool allow_exceptions = true, + const cbor_tag_handler_t tag_handler = cbor_tag_handler_t::error) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::forward(i)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::cbor, &sdp, strict, tag_handler); + return res ? result : basic_json(value_t::discarded); + } + + /*! + @copydoc from_cbor(InputType&&, const bool, const bool, const cbor_tag_handler_t) + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_cbor(IteratorType first, IteratorType last, + const bool strict = true, + const bool allow_exceptions = true, + const cbor_tag_handler_t tag_handler = cbor_tag_handler_t::error) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::move(first), std::move(last)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::cbor, &sdp, strict, tag_handler); + return res ? result : basic_json(value_t::discarded); + } + + template + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_cbor(ptr, ptr + len)) + static basic_json from_cbor(const T* ptr, std::size_t len, + const bool strict = true, + const bool allow_exceptions = true, + const cbor_tag_handler_t tag_handler = cbor_tag_handler_t::error) + { + return from_cbor(ptr, ptr + len, strict, allow_exceptions, tag_handler); + } + + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_cbor(ptr, ptr + len)) + static basic_json from_cbor(detail::span_input_adapter&& i, + const bool strict = true, + const bool allow_exceptions = true, + const cbor_tag_handler_t tag_handler = cbor_tag_handler_t::error) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = i.get(); + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::cbor, &sdp, strict, tag_handler); + return res ? result : basic_json(value_t::discarded); + } + + /*! + @brief create a JSON value from an input in MessagePack format + + Deserializes a given input @a i to a JSON value using the MessagePack + serialization format. + + The library maps MessagePack types to JSON value types as follows: + + MessagePack type | JSON value type | first byte + ---------------- | --------------- | ---------- + positive fixint | number_unsigned | 0x00..0x7F + fixmap | object | 0x80..0x8F + fixarray | array | 0x90..0x9F + fixstr | string | 0xA0..0xBF + nil | `null` | 0xC0 + false | `false` | 0xC2 + true | `true` | 0xC3 + float 32 | number_float | 0xCA + float 64 | number_float | 0xCB + uint 8 | number_unsigned | 0xCC + uint 16 | number_unsigned | 0xCD + uint 32 | number_unsigned | 0xCE + uint 64 | number_unsigned | 0xCF + int 8 | number_integer | 0xD0 + int 16 | number_integer | 0xD1 + int 32 | number_integer | 0xD2 + int 64 | number_integer | 0xD3 + str 8 | string | 0xD9 + str 16 | string | 0xDA + str 32 | string | 0xDB + array 16 | array | 0xDC + array 32 | array | 0xDD + map 16 | object | 0xDE + map 32 | object | 0xDF + bin 8 | binary | 0xC4 + bin 16 | binary | 0xC5 + bin 32 | binary | 0xC6 + ext 8 | binary | 0xC7 + ext 16 | binary | 0xC8 + ext 32 | binary | 0xC9 + fixext 1 | binary | 0xD4 + fixext 2 | binary | 0xD5 + fixext 4 | binary | 0xD6 + fixext 8 | binary | 0xD7 + fixext 16 | binary | 0xD8 + negative fixint | number_integer | 0xE0-0xFF + + @note Any MessagePack output created @ref to_msgpack can be successfully + parsed by @ref from_msgpack. + + @param[in] i an input in MessagePack format convertible to an input + adapter + @param[in] strict whether to expect the input to be consumed until EOF + (true by default) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.110 if the given input ends prematurely or the end of + file was not reached when @a strict was set to true + @throw parse_error.112 if unsupported features from MessagePack were + used in the given input @a i or if the input is not valid MessagePack + @throw parse_error.113 if a string was expected as map key, but not found + + @complexity Linear in the size of the input @a i. + + @liveexample{The example shows the deserialization of a byte vector in + MessagePack format to a JSON value.,from_msgpack} + + @sa http://msgpack.org + @sa see @ref to_msgpack(const basic_json&) for the analogous serialization + @sa see @ref from_cbor(InputType&&, const bool, const bool, const cbor_tag_handler_t) for the + related CBOR format + @sa see @ref from_ubjson(InputType&&, const bool, const bool) for + the related UBJSON format + @sa see @ref from_bson(InputType&&, const bool, const bool) for + the related BSON format + + @since version 2.0.9; parameter @a start_index since 2.1.1; changed to + consume input adapters, removed start_index parameter, and added + @a strict parameter since 3.0.0; added @a allow_exceptions parameter + since 3.2.0 + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_msgpack(InputType&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::forward(i)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::msgpack, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + /*! + @copydoc from_msgpack(InputType&&, const bool, const bool) + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_msgpack(IteratorType first, IteratorType last, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::move(first), std::move(last)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::msgpack, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + + template + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_msgpack(ptr, ptr + len)) + static basic_json from_msgpack(const T* ptr, std::size_t len, + const bool strict = true, + const bool allow_exceptions = true) + { + return from_msgpack(ptr, ptr + len, strict, allow_exceptions); + } + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_msgpack(ptr, ptr + len)) + static basic_json from_msgpack(detail::span_input_adapter&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = i.get(); + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::msgpack, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + + /*! + @brief create a JSON value from an input in UBJSON format + + Deserializes a given input @a i to a JSON value using the UBJSON (Universal + Binary JSON) serialization format. + + The library maps UBJSON types to JSON value types as follows: + + UBJSON type | JSON value type | marker + ----------- | --------------------------------------- | ------ + no-op | *no value, next value is read* | `N` + null | `null` | `Z` + false | `false` | `F` + true | `true` | `T` + float32 | number_float | `d` + float64 | number_float | `D` + uint8 | number_unsigned | `U` + int8 | number_integer | `i` + int16 | number_integer | `I` + int32 | number_integer | `l` + int64 | number_integer | `L` + high-precision number | number_integer, number_unsigned, or number_float - depends on number string | 'H' + string | string | `S` + char | string | `C` + array | array (optimized values are supported) | `[` + object | object (optimized values are supported) | `{` + + @note The mapping is **complete** in the sense that any UBJSON value can + be converted to a JSON value. + + @param[in] i an input in UBJSON format convertible to an input adapter + @param[in] strict whether to expect the input to be consumed until EOF + (true by default) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.110 if the given input ends prematurely or the end of + file was not reached when @a strict was set to true + @throw parse_error.112 if a parse error occurs + @throw parse_error.113 if a string could not be parsed successfully + + @complexity Linear in the size of the input @a i. + + @liveexample{The example shows the deserialization of a byte vector in + UBJSON format to a JSON value.,from_ubjson} + + @sa http://ubjson.org + @sa see @ref to_ubjson(const basic_json&, const bool, const bool) for the + analogous serialization + @sa see @ref from_cbor(InputType&&, const bool, const bool, const cbor_tag_handler_t) for the + related CBOR format + @sa see @ref from_msgpack(InputType&&, const bool, const bool) for + the related MessagePack format + @sa see @ref from_bson(InputType&&, const bool, const bool) for + the related BSON format + + @since version 3.1.0; added @a allow_exceptions parameter since 3.2.0 + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_ubjson(InputType&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::forward(i)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::ubjson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + /*! + @copydoc from_ubjson(InputType&&, const bool, const bool) + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_ubjson(IteratorType first, IteratorType last, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::move(first), std::move(last)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::ubjson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + template + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_ubjson(ptr, ptr + len)) + static basic_json from_ubjson(const T* ptr, std::size_t len, + const bool strict = true, + const bool allow_exceptions = true) + { + return from_ubjson(ptr, ptr + len, strict, allow_exceptions); + } + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_ubjson(ptr, ptr + len)) + static basic_json from_ubjson(detail::span_input_adapter&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = i.get(); + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::ubjson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + + /*! + @brief Create a JSON value from an input in BSON format + + Deserializes a given input @a i to a JSON value using the BSON (Binary JSON) + serialization format. + + The library maps BSON record types to JSON value types as follows: + + BSON type | BSON marker byte | JSON value type + --------------- | ---------------- | --------------------------- + double | 0x01 | number_float + string | 0x02 | string + document | 0x03 | object + array | 0x04 | array + binary | 0x05 | binary + undefined | 0x06 | still unsupported + ObjectId | 0x07 | still unsupported + boolean | 0x08 | boolean + UTC Date-Time | 0x09 | still unsupported + null | 0x0A | null + Regular Expr. | 0x0B | still unsupported + DB Pointer | 0x0C | still unsupported + JavaScript Code | 0x0D | still unsupported + Symbol | 0x0E | still unsupported + JavaScript Code | 0x0F | still unsupported + int32 | 0x10 | number_integer + Timestamp | 0x11 | still unsupported + 128-bit decimal float | 0x13 | still unsupported + Max Key | 0x7F | still unsupported + Min Key | 0xFF | still unsupported + + @warning The mapping is **incomplete**. The unsupported mappings + are indicated in the table above. + + @param[in] i an input in BSON format convertible to an input adapter + @param[in] strict whether to expect the input to be consumed until EOF + (true by default) + @param[in] allow_exceptions whether to throw exceptions in case of a + parse error (optional, true by default) + + @return deserialized JSON value; in case of a parse error and + @a allow_exceptions set to `false`, the return value will be + value_t::discarded. + + @throw parse_error.114 if an unsupported BSON record type is encountered + + @complexity Linear in the size of the input @a i. + + @liveexample{The example shows the deserialization of a byte vector in + BSON format to a JSON value.,from_bson} + + @sa http://bsonspec.org/spec.html + @sa see @ref to_bson(const basic_json&) for the analogous serialization + @sa see @ref from_cbor(InputType&&, const bool, const bool, const cbor_tag_handler_t) for the + related CBOR format + @sa see @ref from_msgpack(InputType&&, const bool, const bool) for + the related MessagePack format + @sa see @ref from_ubjson(InputType&&, const bool, const bool) for the + related UBJSON format + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_bson(InputType&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::forward(i)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::bson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + /*! + @copydoc from_bson(InputType&&, const bool, const bool) + */ + template + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json from_bson(IteratorType first, IteratorType last, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = detail::input_adapter(std::move(first), std::move(last)); + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::bson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + + template + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_bson(ptr, ptr + len)) + static basic_json from_bson(const T* ptr, std::size_t len, + const bool strict = true, + const bool allow_exceptions = true) + { + return from_bson(ptr, ptr + len, strict, allow_exceptions); + } + + JSON_HEDLEY_WARN_UNUSED_RESULT + JSON_HEDLEY_DEPRECATED_FOR(3.8.0, from_bson(ptr, ptr + len)) + static basic_json from_bson(detail::span_input_adapter&& i, + const bool strict = true, + const bool allow_exceptions = true) + { + basic_json result; + detail::json_sax_dom_parser sdp(result, allow_exceptions); + auto ia = i.get(); + // NOLINTNEXTLINE(hicpp-move-const-arg,performance-move-const-arg) + const bool res = binary_reader(std::move(ia)).sax_parse(input_format_t::bson, &sdp, strict); + return res ? result : basic_json(value_t::discarded); + } + /// @} + + ////////////////////////// + // JSON Pointer support // + ////////////////////////// + + /// @name JSON Pointer functions + /// @{ + + /*! + @brief access specified element via JSON Pointer + + Uses a JSON pointer to retrieve a reference to the respective JSON value. + No bound checking is performed. Similar to @ref operator[](const typename + object_t::key_type&), `null` values are created in arrays and objects if + necessary. + + In particular: + - If the JSON pointer points to an object key that does not exist, it + is created an filled with a `null` value before a reference to it + is returned. + - If the JSON pointer points to an array index that does not exist, it + is created an filled with a `null` value before a reference to it + is returned. All indices between the current maximum and the given + index are also filled with `null`. + - The special value `-` is treated as a synonym for the index past the + end. + + @param[in] ptr a JSON pointer + + @return reference to the element pointed to by @a ptr + + @complexity Constant. + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.404 if the JSON pointer can not be resolved + + @liveexample{The behavior is shown in the example.,operatorjson_pointer} + + @since version 2.0.0 + */ + reference operator[](const json_pointer& ptr) + { + return ptr.get_unchecked(this); + } + + /*! + @brief access specified element via JSON Pointer + + Uses a JSON pointer to retrieve a reference to the respective JSON value. + No bound checking is performed. The function does not change the JSON + value; no `null` values are created. In particular, the special value + `-` yields an exception. + + @param[in] ptr JSON pointer to the desired element + + @return const reference to the element pointed to by @a ptr + + @complexity Constant. + + @throw parse_error.106 if an array index begins with '0' + @throw parse_error.109 if an array index was not a number + @throw out_of_range.402 if the array index '-' is used + @throw out_of_range.404 if the JSON pointer can not be resolved + + @liveexample{The behavior is shown in the example.,operatorjson_pointer_const} + + @since version 2.0.0 + */ + const_reference operator[](const json_pointer& ptr) const + { + return ptr.get_unchecked(this); + } + + /*! + @brief access specified element via JSON Pointer + + Returns a reference to the element at with specified JSON pointer @a ptr, + with bounds checking. + + @param[in] ptr JSON pointer to the desired element + + @return reference to the element pointed to by @a ptr + + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0'. See example below. + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number. See example below. + + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index '-' is always invalid. See example below. + + @throw out_of_range.403 if the JSON pointer describes a key of an object + which cannot be found. See example below. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer} + */ + reference at(const json_pointer& ptr) + { + return ptr.get_checked(this); + } + + /*! + @brief access specified element via JSON Pointer + + Returns a const reference to the element at with specified JSON pointer @a + ptr, with bounds checking. + + @param[in] ptr JSON pointer to the desired element + + @return reference to the element pointed to by @a ptr + + @throw parse_error.106 if an array index in the passed JSON pointer @a ptr + begins with '0'. See example below. + + @throw parse_error.109 if an array index in the passed JSON pointer @a ptr + is not a number. See example below. + + @throw out_of_range.401 if an array index in the passed JSON pointer @a ptr + is out of range. See example below. + + @throw out_of_range.402 if the array index '-' is used in the passed JSON + pointer @a ptr. As `at` provides checked access (and no elements are + implicitly inserted), the index '-' is always invalid. See example below. + + @throw out_of_range.403 if the JSON pointer describes a key of an object + which cannot be found. See example below. + + @throw out_of_range.404 if the JSON pointer @a ptr can not be resolved. + See example below. + + @exceptionsafety Strong guarantee: if an exception is thrown, there are no + changes in the JSON value. + + @complexity Constant. + + @since version 2.0.0 + + @liveexample{The behavior is shown in the example.,at_json_pointer_const} + */ + const_reference at(const json_pointer& ptr) const + { + return ptr.get_checked(this); + } + + /*! + @brief return flattened JSON value + + The function creates a JSON object whose keys are JSON pointers (see [RFC + 6901](https://tools.ietf.org/html/rfc6901)) and whose values are all + primitive. The original JSON value can be restored using the @ref + unflatten() function. + + @return an object that maps JSON pointers to primitive values + + @note Empty objects and arrays are flattened to `null` and will not be + reconstructed correctly by the @ref unflatten() function. + + @complexity Linear in the size the JSON value. + + @liveexample{The following code shows how a JSON object is flattened to an + object whose keys consist of JSON pointers.,flatten} + + @sa see @ref unflatten() for the reverse function + + @since version 2.0.0 + */ + basic_json flatten() const + { + basic_json result(value_t::object); + json_pointer::flatten("", *this, result); + return result; + } + + /*! + @brief unflatten a previously flattened JSON value + + The function restores the arbitrary nesting of a JSON value that has been + flattened before using the @ref flatten() function. The JSON value must + meet certain constraints: + 1. The value must be an object. + 2. The keys must be JSON pointers (see + [RFC 6901](https://tools.ietf.org/html/rfc6901)) + 3. The mapped values must be primitive JSON types. + + @return the original JSON from a flattened version + + @note Empty objects and arrays are flattened by @ref flatten() to `null` + values and can not unflattened to their original type. Apart from + this example, for a JSON value `j`, the following is always true: + `j == j.flatten().unflatten()`. + + @complexity Linear in the size the JSON value. + + @throw type_error.314 if value is not an object + @throw type_error.315 if object values are not primitive + + @liveexample{The following code shows how a flattened JSON object is + unflattened into the original nested JSON object.,unflatten} + + @sa see @ref flatten() for the reverse function + + @since version 2.0.0 + */ + basic_json unflatten() const + { + return json_pointer::unflatten(*this); + } + + /// @} + + ////////////////////////// + // JSON Patch functions // + ////////////////////////// + + /// @name JSON Patch functions + /// @{ + + /*! + @brief applies a JSON patch + + [JSON Patch](http://jsonpatch.com) defines a JSON document structure for + expressing a sequence of operations to apply to a JSON) document. With + this function, a JSON Patch is applied to the current JSON value by + executing all operations from the patch. + + @param[in] json_patch JSON patch document + @return patched document + + @note The application of a patch is atomic: Either all operations succeed + and the patched document is returned or an exception is thrown. In + any case, the original value is not changed: the patch is applied + to a copy of the value. + + @throw parse_error.104 if the JSON patch does not consist of an array of + objects + + @throw parse_error.105 if the JSON patch is malformed (e.g., mandatory + attributes are missing); example: `"operation add must have member path"` + + @throw out_of_range.401 if an array index is out of range. + + @throw out_of_range.403 if a JSON pointer inside the patch could not be + resolved successfully in the current JSON value; example: `"key baz not + found"` + + @throw out_of_range.405 if JSON pointer has no parent ("add", "remove", + "move") + + @throw other_error.501 if "test" operation was unsuccessful + + @complexity Linear in the size of the JSON value and the length of the + JSON patch. As usually only a fraction of the JSON value is affected by + the patch, the complexity can usually be neglected. + + @liveexample{The following code shows how a JSON patch is applied to a + value.,patch} + + @sa see @ref diff -- create a JSON patch by comparing two JSON values + + @sa [RFC 6902 (JSON Patch)](https://tools.ietf.org/html/rfc6902) + @sa [RFC 6901 (JSON Pointer)](https://tools.ietf.org/html/rfc6901) + + @since version 2.0.0 + */ + basic_json patch(const basic_json& json_patch) const + { + // make a working copy to apply the patch to + basic_json result = *this; + + // the valid JSON Patch operations + enum class patch_operations {add, remove, replace, move, copy, test, invalid}; + + const auto get_op = [](const std::string & op) + { + if (op == "add") + { + return patch_operations::add; + } + if (op == "remove") + { + return patch_operations::remove; + } + if (op == "replace") + { + return patch_operations::replace; + } + if (op == "move") + { + return patch_operations::move; + } + if (op == "copy") + { + return patch_operations::copy; + } + if (op == "test") + { + return patch_operations::test; + } + + return patch_operations::invalid; + }; + + // wrapper for "add" operation; add value at ptr + const auto operation_add = [&result](json_pointer & ptr, basic_json val) + { + // adding to the root of the target document means replacing it + if (ptr.empty()) + { + result = val; + return; + } + + // make sure the top element of the pointer exists + json_pointer top_pointer = ptr.top(); + if (top_pointer != ptr) + { + result.at(top_pointer); + } + + // get reference to parent of JSON pointer ptr + const auto last_path = ptr.back(); + ptr.pop_back(); + basic_json& parent = result[ptr]; + + switch (parent.m_type) + { + case value_t::null: + case value_t::object: + { + // use operator[] to add value + parent[last_path] = val; + break; + } + + case value_t::array: + { + if (last_path == "-") + { + // special case: append to back + parent.push_back(val); + } + else + { + const auto idx = json_pointer::array_index(last_path); + if (JSON_HEDLEY_UNLIKELY(idx > parent.size())) + { + // avoid undefined behavior + JSON_THROW(out_of_range::create(401, "array index " + std::to_string(idx) + " is out of range", parent)); + } + + // default case: insert add offset + parent.insert(parent.begin() + static_cast(idx), val); + } + break; + } + + // if there exists a parent it cannot be primitive + case value_t::string: // LCOV_EXCL_LINE + case value_t::boolean: // LCOV_EXCL_LINE + case value_t::number_integer: // LCOV_EXCL_LINE + case value_t::number_unsigned: // LCOV_EXCL_LINE + case value_t::number_float: // LCOV_EXCL_LINE + case value_t::binary: // LCOV_EXCL_LINE + case value_t::discarded: // LCOV_EXCL_LINE + default: // LCOV_EXCL_LINE + JSON_ASSERT(false); // NOLINT(cert-dcl03-c,hicpp-static-assert,misc-static-assert) LCOV_EXCL_LINE + } + }; + + // wrapper for "remove" operation; remove value at ptr + const auto operation_remove = [this, &result](json_pointer & ptr) + { + // get reference to parent of JSON pointer ptr + const auto last_path = ptr.back(); + ptr.pop_back(); + basic_json& parent = result.at(ptr); + + // remove child + if (parent.is_object()) + { + // perform range check + auto it = parent.find(last_path); + if (JSON_HEDLEY_LIKELY(it != parent.end())) + { + parent.erase(it); + } + else + { + JSON_THROW(out_of_range::create(403, "key '" + last_path + "' not found", *this)); + } + } + else if (parent.is_array()) + { + // note erase performs range check + parent.erase(json_pointer::array_index(last_path)); + } + }; + + // type check: top level value must be an array + if (JSON_HEDLEY_UNLIKELY(!json_patch.is_array())) + { + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects", json_patch)); + } + + // iterate and apply the operations + for (const auto& val : json_patch) + { + // wrapper to get a value for an operation + const auto get_value = [&val](const std::string & op, + const std::string & member, + bool string_type) -> basic_json & + { + // find value + auto it = val.m_value.object->find(member); + + // context-sensitive error message + const auto error_msg = (op == "op") ? "operation" : "operation '" + op + "'"; + + // check if desired value is present + if (JSON_HEDLEY_UNLIKELY(it == val.m_value.object->end())) + { + // NOLINTNEXTLINE(performance-inefficient-string-concatenation) + JSON_THROW(parse_error::create(105, 0, error_msg + " must have member '" + member + "'", val)); + } + + // check if result is of type string + if (JSON_HEDLEY_UNLIKELY(string_type && !it->second.is_string())) + { + // NOLINTNEXTLINE(performance-inefficient-string-concatenation) + JSON_THROW(parse_error::create(105, 0, error_msg + " must have string member '" + member + "'", val)); + } + + // no error: return value + return it->second; + }; + + // type check: every element of the array must be an object + if (JSON_HEDLEY_UNLIKELY(!val.is_object())) + { + JSON_THROW(parse_error::create(104, 0, "JSON patch must be an array of objects", val)); + } + + // collect mandatory members + const auto op = get_value("op", "op", true).template get(); + const auto path = get_value(op, "path", true).template get(); + json_pointer ptr(path); + + switch (get_op(op)) + { + case patch_operations::add: + { + operation_add(ptr, get_value("add", "value", false)); + break; + } + + case patch_operations::remove: + { + operation_remove(ptr); + break; + } + + case patch_operations::replace: + { + // the "path" location must exist - use at() + result.at(ptr) = get_value("replace", "value", false); + break; + } + + case patch_operations::move: + { + const auto from_path = get_value("move", "from", true).template get(); + json_pointer from_ptr(from_path); + + // the "from" location must exist - use at() + basic_json v = result.at(from_ptr); + + // The move operation is functionally identical to a + // "remove" operation on the "from" location, followed + // immediately by an "add" operation at the target + // location with the value that was just removed. + operation_remove(from_ptr); + operation_add(ptr, v); + break; + } + + case patch_operations::copy: + { + const auto from_path = get_value("copy", "from", true).template get(); + const json_pointer from_ptr(from_path); + + // the "from" location must exist - use at() + basic_json v = result.at(from_ptr); + + // The copy is functionally identical to an "add" + // operation at the target location using the value + // specified in the "from" member. + operation_add(ptr, v); + break; + } + + case patch_operations::test: + { + bool success = false; + JSON_TRY + { + // check if "value" matches the one at "path" + // the "path" location must exist - use at() + success = (result.at(ptr) == get_value("test", "value", false)); + } + JSON_INTERNAL_CATCH (out_of_range&) + { + // ignore out of range errors: success remains false + } + + // throw an exception if test fails + if (JSON_HEDLEY_UNLIKELY(!success)) + { + JSON_THROW(other_error::create(501, "unsuccessful: " + val.dump(), val)); + } + + break; + } + + case patch_operations::invalid: + default: + { + // op must be "add", "remove", "replace", "move", "copy", or + // "test" + JSON_THROW(parse_error::create(105, 0, "operation value '" + op + "' is invalid", val)); + } + } + } + + return result; + } + + /*! + @brief creates a diff as a JSON patch + + Creates a [JSON Patch](http://jsonpatch.com) so that value @a source can + be changed into the value @a target by calling @ref patch function. + + @invariant For two JSON values @a source and @a target, the following code + yields always `true`: + @code {.cpp} + source.patch(diff(source, target)) == target; + @endcode + + @note Currently, only `remove`, `add`, and `replace` operations are + generated. + + @param[in] source JSON value to compare from + @param[in] target JSON value to compare against + @param[in] path helper value to create JSON pointers + + @return a JSON patch to convert the @a source to @a target + + @complexity Linear in the lengths of @a source and @a target. + + @liveexample{The following code shows how a JSON patch is created as a + diff for two JSON values.,diff} + + @sa see @ref patch -- apply a JSON patch + @sa see @ref merge_patch -- apply a JSON Merge Patch + + @sa [RFC 6902 (JSON Patch)](https://tools.ietf.org/html/rfc6902) + + @since version 2.0.0 + */ + JSON_HEDLEY_WARN_UNUSED_RESULT + static basic_json diff(const basic_json& source, const basic_json& target, + const std::string& path = "") + { + // the patch + basic_json result(value_t::array); + + // if the values are the same, return empty patch + if (source == target) + { + return result; + } + + if (source.type() != target.type()) + { + // different types: replace value + result.push_back( + { + {"op", "replace"}, {"path", path}, {"value", target} + }); + return result; + } + + switch (source.type()) + { + case value_t::array: + { + // first pass: traverse common elements + std::size_t i = 0; + while (i < source.size() && i < target.size()) + { + // recursive call to compare array values at index i + auto temp_diff = diff(source[i], target[i], path + "/" + std::to_string(i)); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + ++i; + } + + // i now reached the end of at least one array + // in a second pass, traverse the remaining elements + + // remove my remaining elements + const auto end_index = static_cast(result.size()); + while (i < source.size()) + { + // add operations in reverse order to avoid invalid + // indices + result.insert(result.begin() + end_index, object( + { + {"op", "remove"}, + {"path", path + "/" + std::to_string(i)} + })); + ++i; + } + + // add other remaining elements + while (i < target.size()) + { + result.push_back( + { + {"op", "add"}, + {"path", path + "/-"}, + {"value", target[i]} + }); + ++i; + } + + break; + } + + case value_t::object: + { + // first pass: traverse this object's elements + for (auto it = source.cbegin(); it != source.cend(); ++it) + { + // escape the key name to be used in a JSON patch + const auto path_key = path + "/" + detail::escape(it.key()); + + if (target.find(it.key()) != target.end()) + { + // recursive call to compare object values at key it + auto temp_diff = diff(it.value(), target[it.key()], path_key); + result.insert(result.end(), temp_diff.begin(), temp_diff.end()); + } + else + { + // found a key that is not in o -> remove it + result.push_back(object( + { + {"op", "remove"}, {"path", path_key} + })); + } + } + + // second pass: traverse other object's elements + for (auto it = target.cbegin(); it != target.cend(); ++it) + { + if (source.find(it.key()) == source.end()) + { + // found a key that is not in this -> add it + const auto path_key = path + "/" + detail::escape(it.key()); + result.push_back( + { + {"op", "add"}, {"path", path_key}, + {"value", it.value()} + }); + } + } + + break; + } + + case value_t::null: + case value_t::string: + case value_t::boolean: + case value_t::number_integer: + case value_t::number_unsigned: + case value_t::number_float: + case value_t::binary: + case value_t::discarded: + default: + { + // both primitive type: replace value + result.push_back( + { + {"op", "replace"}, {"path", path}, {"value", target} + }); + break; + } + } + + return result; + } + + /// @} + + //////////////////////////////// + // JSON Merge Patch functions // + //////////////////////////////// + + /// @name JSON Merge Patch functions + /// @{ + + /*! + @brief applies a JSON Merge Patch + + The merge patch format is primarily intended for use with the HTTP PATCH + method as a means of describing a set of modifications to a target + resource's content. This function applies a merge patch to the current + JSON value. + + The function implements the following algorithm from Section 2 of + [RFC 7396 (JSON Merge Patch)](https://tools.ietf.org/html/rfc7396): + + ``` + define MergePatch(Target, Patch): + if Patch is an Object: + if Target is not an Object: + Target = {} // Ignore the contents and set it to an empty Object + for each Name/Value pair in Patch: + if Value is null: + if Name exists in Target: + remove the Name/Value pair from Target + else: + Target[Name] = MergePatch(Target[Name], Value) + return Target + else: + return Patch + ``` + + Thereby, `Target` is the current object; that is, the patch is applied to + the current value. + + @param[in] apply_patch the patch to apply + + @complexity Linear in the lengths of @a patch. + + @liveexample{The following code shows how a JSON Merge Patch is applied to + a JSON document.,merge_patch} + + @sa see @ref patch -- apply a JSON patch + @sa [RFC 7396 (JSON Merge Patch)](https://tools.ietf.org/html/rfc7396) + + @since version 3.0.0 + */ + void merge_patch(const basic_json& apply_patch) + { + if (apply_patch.is_object()) + { + if (!is_object()) + { + *this = object(); + } + for (auto it = apply_patch.begin(); it != apply_patch.end(); ++it) + { + if (it.value().is_null()) + { + erase(it.key()); + } + else + { + operator[](it.key()).merge_patch(it.value()); + } + } + } + else + { + *this = apply_patch; + } + } + + /// @} +}; + +/*! +@brief user-defined to_string function for JSON values + +This function implements a user-defined to_string for JSON objects. + +@param[in] j a JSON object +@return a std::string object +*/ + +NLOHMANN_BASIC_JSON_TPL_DECLARATION +std::string to_string(const NLOHMANN_BASIC_JSON_TPL& j) +{ + return j.dump(); +} +} // namespace nlohmann + +/////////////////////// +// nonmember support // +/////////////////////// + +// specialization of std::swap, and std::hash +namespace std +{ + +/// hash value for JSON objects +template<> +struct hash +{ + /*! + @brief return a hash value for a JSON object + + @since version 1.0.0 + */ + std::size_t operator()(const nlohmann::json& j) const + { + return nlohmann::detail::hash(j); + } +}; + +/// specialization for std::less +/// @note: do not remove the space after '<', +/// see https://github.com/nlohmann/json/pull/679 +template<> +struct less<::nlohmann::detail::value_t> +{ + /*! + @brief compare two value_t enum values + @since version 3.0.0 + */ + bool operator()(nlohmann::detail::value_t lhs, + nlohmann::detail::value_t rhs) const noexcept + { + return nlohmann::detail::operator<(lhs, rhs); + } +}; + +// C++20 prohibit function specialization in the std namespace. +#ifndef JSON_HAS_CPP_20 + +/*! +@brief exchanges the values of two JSON objects + +@since version 1.0.0 +*/ +template<> +inline void swap(nlohmann::json& j1, nlohmann::json& j2) noexcept( // NOLINT(readability-inconsistent-declaration-parameter-name) + is_nothrow_move_constructible::value&& // NOLINT(misc-redundant-expression) + is_nothrow_move_assignable::value + ) +{ + j1.swap(j2); +} + +#endif + +} // namespace std + +/*! +@brief user-defined string literal for JSON values + +This operator implements a user-defined string literal for JSON objects. It +can be used by adding `"_json"` to a string literal and returns a JSON object +if no parse error occurred. + +@param[in] s a string representation of a JSON object +@param[in] n the length of string @a s +@return a JSON object + +@since version 1.0.0 +*/ +JSON_HEDLEY_NON_NULL(1) +inline nlohmann::json operator "" _json(const char* s, std::size_t n) +{ + return nlohmann::json::parse(s, s + n); +} + +/*! +@brief user-defined string literal for JSON pointer + +This operator implements a user-defined string literal for JSON Pointers. It +can be used by adding `"_json_pointer"` to a string literal and returns a JSON pointer +object if no parse error occurred. + +@param[in] s a string representation of a JSON Pointer +@param[in] n the length of string @a s +@return a JSON pointer object + +@since version 2.0.0 +*/ +JSON_HEDLEY_NON_NULL(1) +inline nlohmann::json::json_pointer operator "" _json_pointer(const char* s, std::size_t n) +{ + return nlohmann::json::json_pointer(std::string(s, n)); +} + +// #include + + +// restore clang diagnostic settings +#if defined(__clang__) + #pragma clang diagnostic pop +#endif + +// clean up +#undef JSON_ASSERT +#undef JSON_INTERNAL_CATCH +#undef JSON_CATCH +#undef JSON_THROW +#undef JSON_TRY +#undef JSON_PRIVATE_UNLESS_TESTED +#undef JSON_HAS_CPP_11 +#undef JSON_HAS_CPP_14 +#undef JSON_HAS_CPP_17 +#undef JSON_HAS_CPP_20 +#undef NLOHMANN_BASIC_JSON_TPL_DECLARATION +#undef NLOHMANN_BASIC_JSON_TPL +#undef JSON_EXPLICIT +#undef NLOHMANN_CAN_CALL_STD_FUNC_IMPL + +// #include + + +#undef JSON_HEDLEY_ALWAYS_INLINE +#undef JSON_HEDLEY_ARM_VERSION +#undef JSON_HEDLEY_ARM_VERSION_CHECK +#undef JSON_HEDLEY_ARRAY_PARAM +#undef JSON_HEDLEY_ASSUME +#undef JSON_HEDLEY_BEGIN_C_DECLS +#undef JSON_HEDLEY_CLANG_HAS_ATTRIBUTE +#undef JSON_HEDLEY_CLANG_HAS_BUILTIN +#undef JSON_HEDLEY_CLANG_HAS_CPP_ATTRIBUTE +#undef JSON_HEDLEY_CLANG_HAS_DECLSPEC_DECLSPEC_ATTRIBUTE +#undef JSON_HEDLEY_CLANG_HAS_EXTENSION +#undef JSON_HEDLEY_CLANG_HAS_FEATURE +#undef JSON_HEDLEY_CLANG_HAS_WARNING +#undef JSON_HEDLEY_COMPCERT_VERSION +#undef JSON_HEDLEY_COMPCERT_VERSION_CHECK +#undef JSON_HEDLEY_CONCAT +#undef JSON_HEDLEY_CONCAT3 +#undef JSON_HEDLEY_CONCAT3_EX +#undef JSON_HEDLEY_CONCAT_EX +#undef JSON_HEDLEY_CONST +#undef JSON_HEDLEY_CONSTEXPR +#undef JSON_HEDLEY_CONST_CAST +#undef JSON_HEDLEY_CPP_CAST +#undef JSON_HEDLEY_CRAY_VERSION +#undef JSON_HEDLEY_CRAY_VERSION_CHECK +#undef JSON_HEDLEY_C_DECL +#undef JSON_HEDLEY_DEPRECATED +#undef JSON_HEDLEY_DEPRECATED_FOR +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_CAST_QUAL +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_CPP98_COMPAT_WRAP_ +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_DEPRECATED +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_CPP_ATTRIBUTES +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNKNOWN_PRAGMAS +#undef JSON_HEDLEY_DIAGNOSTIC_DISABLE_UNUSED_FUNCTION +#undef JSON_HEDLEY_DIAGNOSTIC_POP +#undef JSON_HEDLEY_DIAGNOSTIC_PUSH +#undef JSON_HEDLEY_DMC_VERSION +#undef JSON_HEDLEY_DMC_VERSION_CHECK +#undef JSON_HEDLEY_EMPTY_BASES +#undef JSON_HEDLEY_EMSCRIPTEN_VERSION +#undef JSON_HEDLEY_EMSCRIPTEN_VERSION_CHECK +#undef JSON_HEDLEY_END_C_DECLS +#undef JSON_HEDLEY_FLAGS +#undef JSON_HEDLEY_FLAGS_CAST +#undef JSON_HEDLEY_GCC_HAS_ATTRIBUTE +#undef JSON_HEDLEY_GCC_HAS_BUILTIN +#undef JSON_HEDLEY_GCC_HAS_CPP_ATTRIBUTE +#undef JSON_HEDLEY_GCC_HAS_DECLSPEC_ATTRIBUTE +#undef JSON_HEDLEY_GCC_HAS_EXTENSION +#undef JSON_HEDLEY_GCC_HAS_FEATURE +#undef JSON_HEDLEY_GCC_HAS_WARNING +#undef JSON_HEDLEY_GCC_NOT_CLANG_VERSION_CHECK +#undef JSON_HEDLEY_GCC_VERSION +#undef JSON_HEDLEY_GCC_VERSION_CHECK +#undef JSON_HEDLEY_GNUC_HAS_ATTRIBUTE +#undef JSON_HEDLEY_GNUC_HAS_BUILTIN +#undef JSON_HEDLEY_GNUC_HAS_CPP_ATTRIBUTE +#undef JSON_HEDLEY_GNUC_HAS_DECLSPEC_ATTRIBUTE +#undef JSON_HEDLEY_GNUC_HAS_EXTENSION +#undef JSON_HEDLEY_GNUC_HAS_FEATURE +#undef JSON_HEDLEY_GNUC_HAS_WARNING +#undef JSON_HEDLEY_GNUC_VERSION +#undef JSON_HEDLEY_GNUC_VERSION_CHECK +#undef JSON_HEDLEY_HAS_ATTRIBUTE +#undef JSON_HEDLEY_HAS_BUILTIN +#undef JSON_HEDLEY_HAS_CPP_ATTRIBUTE +#undef JSON_HEDLEY_HAS_CPP_ATTRIBUTE_NS +#undef JSON_HEDLEY_HAS_DECLSPEC_ATTRIBUTE +#undef JSON_HEDLEY_HAS_EXTENSION +#undef JSON_HEDLEY_HAS_FEATURE +#undef JSON_HEDLEY_HAS_WARNING +#undef JSON_HEDLEY_IAR_VERSION +#undef JSON_HEDLEY_IAR_VERSION_CHECK +#undef JSON_HEDLEY_IBM_VERSION +#undef JSON_HEDLEY_IBM_VERSION_CHECK +#undef JSON_HEDLEY_IMPORT +#undef JSON_HEDLEY_INLINE +#undef JSON_HEDLEY_INTEL_CL_VERSION +#undef JSON_HEDLEY_INTEL_CL_VERSION_CHECK +#undef JSON_HEDLEY_INTEL_VERSION +#undef JSON_HEDLEY_INTEL_VERSION_CHECK +#undef JSON_HEDLEY_IS_CONSTANT +#undef JSON_HEDLEY_IS_CONSTEXPR_ +#undef JSON_HEDLEY_LIKELY +#undef JSON_HEDLEY_MALLOC +#undef JSON_HEDLEY_MCST_LCC_VERSION +#undef JSON_HEDLEY_MCST_LCC_VERSION_CHECK +#undef JSON_HEDLEY_MESSAGE +#undef JSON_HEDLEY_MSVC_VERSION +#undef JSON_HEDLEY_MSVC_VERSION_CHECK +#undef JSON_HEDLEY_NEVER_INLINE +#undef JSON_HEDLEY_NON_NULL +#undef JSON_HEDLEY_NO_ESCAPE +#undef JSON_HEDLEY_NO_RETURN +#undef JSON_HEDLEY_NO_THROW +#undef JSON_HEDLEY_NULL +#undef JSON_HEDLEY_PELLES_VERSION +#undef JSON_HEDLEY_PELLES_VERSION_CHECK +#undef JSON_HEDLEY_PGI_VERSION +#undef JSON_HEDLEY_PGI_VERSION_CHECK +#undef JSON_HEDLEY_PREDICT +#undef JSON_HEDLEY_PRINTF_FORMAT +#undef JSON_HEDLEY_PRIVATE +#undef JSON_HEDLEY_PUBLIC +#undef JSON_HEDLEY_PURE +#undef JSON_HEDLEY_REINTERPRET_CAST +#undef JSON_HEDLEY_REQUIRE +#undef JSON_HEDLEY_REQUIRE_CONSTEXPR +#undef JSON_HEDLEY_REQUIRE_MSG +#undef JSON_HEDLEY_RESTRICT +#undef JSON_HEDLEY_RETURNS_NON_NULL +#undef JSON_HEDLEY_SENTINEL +#undef JSON_HEDLEY_STATIC_ASSERT +#undef JSON_HEDLEY_STATIC_CAST +#undef JSON_HEDLEY_STRINGIFY +#undef JSON_HEDLEY_STRINGIFY_EX +#undef JSON_HEDLEY_SUNPRO_VERSION +#undef JSON_HEDLEY_SUNPRO_VERSION_CHECK +#undef JSON_HEDLEY_TINYC_VERSION +#undef JSON_HEDLEY_TINYC_VERSION_CHECK +#undef JSON_HEDLEY_TI_ARMCL_VERSION +#undef JSON_HEDLEY_TI_ARMCL_VERSION_CHECK +#undef JSON_HEDLEY_TI_CL2000_VERSION +#undef JSON_HEDLEY_TI_CL2000_VERSION_CHECK +#undef JSON_HEDLEY_TI_CL430_VERSION +#undef JSON_HEDLEY_TI_CL430_VERSION_CHECK +#undef JSON_HEDLEY_TI_CL6X_VERSION +#undef JSON_HEDLEY_TI_CL6X_VERSION_CHECK +#undef JSON_HEDLEY_TI_CL7X_VERSION +#undef JSON_HEDLEY_TI_CL7X_VERSION_CHECK +#undef JSON_HEDLEY_TI_CLPRU_VERSION +#undef JSON_HEDLEY_TI_CLPRU_VERSION_CHECK +#undef JSON_HEDLEY_TI_VERSION +#undef JSON_HEDLEY_TI_VERSION_CHECK +#undef JSON_HEDLEY_UNAVAILABLE +#undef JSON_HEDLEY_UNLIKELY +#undef JSON_HEDLEY_UNPREDICTABLE +#undef JSON_HEDLEY_UNREACHABLE +#undef JSON_HEDLEY_UNREACHABLE_RETURN +#undef JSON_HEDLEY_VERSION +#undef JSON_HEDLEY_VERSION_DECODE_MAJOR +#undef JSON_HEDLEY_VERSION_DECODE_MINOR +#undef JSON_HEDLEY_VERSION_DECODE_REVISION +#undef JSON_HEDLEY_VERSION_ENCODE +#undef JSON_HEDLEY_WARNING +#undef JSON_HEDLEY_WARN_UNUSED_RESULT +#undef JSON_HEDLEY_WARN_UNUSED_RESULT_MSG +#undef JSON_HEDLEY_FALL_THROUGH + + + +#endif // INCLUDE_NLOHMANN_JSON_HPP_ diff --git a/src/external/tinygltf/tiny_gltf.cc b/src/external/tinygltf/tiny_gltf.cc new file mode 100644 index 00000000..3f279152 --- /dev/null +++ b/src/external/tinygltf/tiny_gltf.cc @@ -0,0 +1,4 @@ +#define TINYGLTF_IMPLEMENTATION +#define STB_IMAGE_IMPLEMENTATION +#define STB_IMAGE_WRITE_IMPLEMENTATION +#include "tiny_gltf.h" diff --git a/src/external/tinygltf/tiny_gltf.h b/src/external/tinygltf/tiny_gltf.h new file mode 100644 index 00000000..0f69ad27 --- /dev/null +++ b/src/external/tinygltf/tiny_gltf.h @@ -0,0 +1,8249 @@ +// +// Header-only tiny glTF 2.0 loader and serializer. +// +// +// The MIT License (MIT) +// +// Copyright (c) 2015 - Present Syoyo Fujita, Aurélien Chatelain and many +// contributors. +// +// 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. + +// Version: +// - v2.8.1 Missed serialization texture sampler name fixed. PR#399. +// - v2.8.0 Add URICallbacks for custom URI handling in Buffer and Image. PR#397. +// - v2.7.0 Change WriteImageDataFunction user callback function signature. PR#393. +// - v2.6.3 Fix GLB file with empty BIN chunk was not handled. PR#382 and PR#383. +// - v2.6.2 Fix out-of-bounds access of accessors. PR#379. +// - v2.6.1 Better GLB validation check when loading. +// - v2.6.0 Support serializing sparse accessor(Thanks to @fynv). +// Disable expanding file path for security(no use of awkward `wordexp` anymore). +// - v2.5.0 Add SetPreserveImageChannels() option to load image data as is. +// - v2.4.3 Fix null object output when material has all default +// parameters. +// - v2.4.2 Decode percent-encoded URI. +// - v2.4.1 Fix some glTF object class does not have `extensions` and/or +// `extras` property. +// - v2.4.0 Experimental RapidJSON and C++14 support(Thanks to @jrkoone). +// - v2.3.1 Set default value of minFilter and magFilter in Sampler to -1. +// - v2.3.0 Modified Material representation according to glTF 2.0 schema +// (and introduced TextureInfo class) +// Change the behavior of `Value::IsNumber`. It return true either the +// value is int or real. +// - v2.2.0 Add loading 16bit PNG support. Add Sparse accessor support(Thanks +// to @Ybalrid) +// - v2.1.0 Add draco compression. +// - v2.0.1 Add comparison feature(Thanks to @Selmar). +// - v2.0.0 glTF 2.0!. +// +// Tiny glTF loader is using following third party libraries: +// +// - jsonhpp: C++ JSON library. +// - base64: base64 decode/encode library. +// - stb_image: Image loading library. +// +#ifndef TINY_GLTF_H_ +#define TINY_GLTF_H_ + +#include +#include +#include // std::fabs +#include +#include +#include +#include +#include +#include +#include + +//Auto-detect C++14 standard version +#if !defined(TINYGLTF_USE_CPP14) && defined(__cplusplus) && (__cplusplus >= 201402L) +#define TINYGLTF_USE_CPP14 +#endif + +#ifndef TINYGLTF_USE_CPP14 +#include +#endif + +#ifdef __ANDROID__ +#ifdef TINYGLTF_ANDROID_LOAD_FROM_ASSETS +#include +#endif +#endif + +#ifdef __GNUC__ +#if (__GNUC__ < 4) || ((__GNUC__ == 4) && (__GNUC_MINOR__ <= 8)) +#define TINYGLTF_NOEXCEPT +#else +#define TINYGLTF_NOEXCEPT noexcept +#endif +#else +#define TINYGLTF_NOEXCEPT noexcept +#endif + +#define DEFAULT_METHODS(x) \ + ~x() = default; \ + x(const x &) = default; \ + x(x &&) TINYGLTF_NOEXCEPT = default; \ + x &operator=(const x &) = default; \ + x &operator=(x &&) TINYGLTF_NOEXCEPT = default; + +namespace tinygltf { + +#define TINYGLTF_MODE_POINTS (0) +#define TINYGLTF_MODE_LINE (1) +#define TINYGLTF_MODE_LINE_LOOP (2) +#define TINYGLTF_MODE_LINE_STRIP (3) +#define TINYGLTF_MODE_TRIANGLES (4) +#define TINYGLTF_MODE_TRIANGLE_STRIP (5) +#define TINYGLTF_MODE_TRIANGLE_FAN (6) + +#define TINYGLTF_COMPONENT_TYPE_BYTE (5120) +#define TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE (5121) +#define TINYGLTF_COMPONENT_TYPE_SHORT (5122) +#define TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT (5123) +#define TINYGLTF_COMPONENT_TYPE_INT (5124) +#define TINYGLTF_COMPONENT_TYPE_UNSIGNED_INT (5125) +#define TINYGLTF_COMPONENT_TYPE_FLOAT (5126) +#define TINYGLTF_COMPONENT_TYPE_DOUBLE \ + (5130) // OpenGL double type. Note that some of glTF 2.0 validator does not + // support double type even the schema seems allow any value of + // integer: + // https://github.com/KhronosGroup/glTF/blob/b9884a2fd45130b4d673dd6c8a706ee21ee5c5f7/specification/2.0/schema/accessor.schema.json#L22 + +#define TINYGLTF_TEXTURE_FILTER_NEAREST (9728) +#define TINYGLTF_TEXTURE_FILTER_LINEAR (9729) +#define TINYGLTF_TEXTURE_FILTER_NEAREST_MIPMAP_NEAREST (9984) +#define TINYGLTF_TEXTURE_FILTER_LINEAR_MIPMAP_NEAREST (9985) +#define TINYGLTF_TEXTURE_FILTER_NEAREST_MIPMAP_LINEAR (9986) +#define TINYGLTF_TEXTURE_FILTER_LINEAR_MIPMAP_LINEAR (9987) + +#define TINYGLTF_TEXTURE_WRAP_REPEAT (10497) +#define TINYGLTF_TEXTURE_WRAP_CLAMP_TO_EDGE (33071) +#define TINYGLTF_TEXTURE_WRAP_MIRRORED_REPEAT (33648) + +// Redeclarations of the above for technique.parameters. +#define TINYGLTF_PARAMETER_TYPE_BYTE (5120) +#define TINYGLTF_PARAMETER_TYPE_UNSIGNED_BYTE (5121) +#define TINYGLTF_PARAMETER_TYPE_SHORT (5122) +#define TINYGLTF_PARAMETER_TYPE_UNSIGNED_SHORT (5123) +#define TINYGLTF_PARAMETER_TYPE_INT (5124) +#define TINYGLTF_PARAMETER_TYPE_UNSIGNED_INT (5125) +#define TINYGLTF_PARAMETER_TYPE_FLOAT (5126) + +#define TINYGLTF_PARAMETER_TYPE_FLOAT_VEC2 (35664) +#define TINYGLTF_PARAMETER_TYPE_FLOAT_VEC3 (35665) +#define TINYGLTF_PARAMETER_TYPE_FLOAT_VEC4 (35666) + +#define TINYGLTF_PARAMETER_TYPE_INT_VEC2 (35667) +#define TINYGLTF_PARAMETER_TYPE_INT_VEC3 (35668) +#define TINYGLTF_PARAMETER_TYPE_INT_VEC4 (35669) + +#define TINYGLTF_PARAMETER_TYPE_BOOL (35670) +#define TINYGLTF_PARAMETER_TYPE_BOOL_VEC2 (35671) +#define TINYGLTF_PARAMETER_TYPE_BOOL_VEC3 (35672) +#define TINYGLTF_PARAMETER_TYPE_BOOL_VEC4 (35673) + +#define TINYGLTF_PARAMETER_TYPE_FLOAT_MAT2 (35674) +#define TINYGLTF_PARAMETER_TYPE_FLOAT_MAT3 (35675) +#define TINYGLTF_PARAMETER_TYPE_FLOAT_MAT4 (35676) + +#define TINYGLTF_PARAMETER_TYPE_SAMPLER_2D (35678) + +// End parameter types + +#define TINYGLTF_TYPE_VEC2 (2) +#define TINYGLTF_TYPE_VEC3 (3) +#define TINYGLTF_TYPE_VEC4 (4) +#define TINYGLTF_TYPE_MAT2 (32 + 2) +#define TINYGLTF_TYPE_MAT3 (32 + 3) +#define TINYGLTF_TYPE_MAT4 (32 + 4) +#define TINYGLTF_TYPE_SCALAR (64 + 1) +#define TINYGLTF_TYPE_VECTOR (64 + 4) +#define TINYGLTF_TYPE_MATRIX (64 + 16) + +#define TINYGLTF_IMAGE_FORMAT_JPEG (0) +#define TINYGLTF_IMAGE_FORMAT_PNG (1) +#define TINYGLTF_IMAGE_FORMAT_BMP (2) +#define TINYGLTF_IMAGE_FORMAT_GIF (3) + +#define TINYGLTF_TEXTURE_FORMAT_ALPHA (6406) +#define TINYGLTF_TEXTURE_FORMAT_RGB (6407) +#define TINYGLTF_TEXTURE_FORMAT_RGBA (6408) +#define TINYGLTF_TEXTURE_FORMAT_LUMINANCE (6409) +#define TINYGLTF_TEXTURE_FORMAT_LUMINANCE_ALPHA (6410) + +#define TINYGLTF_TEXTURE_TARGET_TEXTURE2D (3553) +#define TINYGLTF_TEXTURE_TYPE_UNSIGNED_BYTE (5121) + +#define TINYGLTF_TARGET_ARRAY_BUFFER (34962) +#define TINYGLTF_TARGET_ELEMENT_ARRAY_BUFFER (34963) + +#define TINYGLTF_SHADER_TYPE_VERTEX_SHADER (35633) +#define TINYGLTF_SHADER_TYPE_FRAGMENT_SHADER (35632) + +#define TINYGLTF_DOUBLE_EPS (1.e-12) +#define TINYGLTF_DOUBLE_EQUAL(a, b) (std::fabs((b) - (a)) < TINYGLTF_DOUBLE_EPS) + +#ifdef __ANDROID__ +#ifdef TINYGLTF_ANDROID_LOAD_FROM_ASSETS +#ifdef TINYGLTF_IMPLEMENTATION +AAssetManager *asset_manager = nullptr; +#else +extern AAssetManager *asset_manager; +#endif +#endif +#endif + +typedef enum { + NULL_TYPE, + REAL_TYPE, + INT_TYPE, + BOOL_TYPE, + STRING_TYPE, + ARRAY_TYPE, + BINARY_TYPE, + OBJECT_TYPE +} Type; + +static inline int32_t GetComponentSizeInBytes(uint32_t componentType) { + if (componentType == TINYGLTF_COMPONENT_TYPE_BYTE) { + return 1; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE) { + return 1; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_SHORT) { + return 2; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT) { + return 2; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_INT) { + return 4; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_UNSIGNED_INT) { + return 4; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_FLOAT) { + return 4; + } else if (componentType == TINYGLTF_COMPONENT_TYPE_DOUBLE) { + return 8; + } else { + // Unknown component type + return -1; + } +} + +static inline int32_t GetNumComponentsInType(uint32_t ty) { + if (ty == TINYGLTF_TYPE_SCALAR) { + return 1; + } else if (ty == TINYGLTF_TYPE_VEC2) { + return 2; + } else if (ty == TINYGLTF_TYPE_VEC3) { + return 3; + } else if (ty == TINYGLTF_TYPE_VEC4) { + return 4; + } else if (ty == TINYGLTF_TYPE_MAT2) { + return 4; + } else if (ty == TINYGLTF_TYPE_MAT3) { + return 9; + } else if (ty == TINYGLTF_TYPE_MAT4) { + return 16; + } else { + // Unknown component type + return -1; + } +} + +// TODO(syoyo): Move these functions to TinyGLTF class +bool IsDataURI(const std::string &in); +bool DecodeDataURI(std::vector *out, std::string &mime_type, + const std::string &in, size_t reqBytes, bool checkSize); + +#ifdef __clang__ +#pragma clang diagnostic push +// Suppress warning for : static Value null_value +#pragma clang diagnostic ignored "-Wexit-time-destructors" +#pragma clang diagnostic ignored "-Wpadded" +#endif + +// Simple class to represent JSON object +class Value { + public: + typedef std::vector Array; + typedef std::map Object; + + Value() + : type_(NULL_TYPE), + int_value_(0), + real_value_(0.0), + boolean_value_(false) {} + + explicit Value(bool b) : type_(BOOL_TYPE) { boolean_value_ = b; } + explicit Value(int i) : type_(INT_TYPE) { + int_value_ = i; + real_value_ = i; + } + explicit Value(double n) : type_(REAL_TYPE) { real_value_ = n; } + explicit Value(const std::string &s) : type_(STRING_TYPE) { + string_value_ = s; + } + explicit Value(std::string &&s) + : type_(STRING_TYPE), string_value_(std::move(s)) {} + explicit Value(const char *s) : type_(STRING_TYPE) { + string_value_ = s; + } + explicit Value(const unsigned char *p, size_t n) : type_(BINARY_TYPE) { + binary_value_.resize(n); + memcpy(binary_value_.data(), p, n); + } + explicit Value(std::vector &&v) noexcept + : type_(BINARY_TYPE), + binary_value_(std::move(v)) {} + explicit Value(const Array &a) : type_(ARRAY_TYPE) { array_value_ = a; } + explicit Value(Array &&a) noexcept : type_(ARRAY_TYPE), + array_value_(std::move(a)) {} + + explicit Value(const Object &o) : type_(OBJECT_TYPE) { object_value_ = o; } + explicit Value(Object &&o) noexcept : type_(OBJECT_TYPE), + object_value_(std::move(o)) {} + + DEFAULT_METHODS(Value) + + char Type() const { return static_cast(type_); } + + bool IsBool() const { return (type_ == BOOL_TYPE); } + + bool IsInt() const { return (type_ == INT_TYPE); } + + bool IsNumber() const { return (type_ == REAL_TYPE) || (type_ == INT_TYPE); } + + bool IsReal() const { return (type_ == REAL_TYPE); } + + bool IsString() const { return (type_ == STRING_TYPE); } + + bool IsBinary() const { return (type_ == BINARY_TYPE); } + + bool IsArray() const { return (type_ == ARRAY_TYPE); } + + bool IsObject() const { return (type_ == OBJECT_TYPE); } + + // Use this function if you want to have number value as double. + double GetNumberAsDouble() const { + if (type_ == INT_TYPE) { + return double(int_value_); + } else { + return real_value_; + } + } + + // Use this function if you want to have number value as int. + // TODO(syoyo): Support int value larger than 32 bits + int GetNumberAsInt() const { + if (type_ == REAL_TYPE) { + return int(real_value_); + } else { + return int_value_; + } + } + + // Accessor + template + const T &Get() const; + template + T &Get(); + + // Lookup value from an array + const Value &Get(int idx) const { + static Value null_value; + assert(IsArray()); + assert(idx >= 0); + return (static_cast(idx) < array_value_.size()) + ? array_value_[static_cast(idx)] + : null_value; + } + + // Lookup value from a key-value pair + const Value &Get(const std::string &key) const { + static Value null_value; + assert(IsObject()); + Object::const_iterator it = object_value_.find(key); + return (it != object_value_.end()) ? it->second : null_value; + } + + size_t ArrayLen() const { + if (!IsArray()) return 0; + return array_value_.size(); + } + + // Valid only for object type. + bool Has(const std::string &key) const { + if (!IsObject()) return false; + Object::const_iterator it = object_value_.find(key); + return (it != object_value_.end()) ? true : false; + } + + // List keys + std::vector Keys() const { + std::vector keys; + if (!IsObject()) return keys; // empty + + for (Object::const_iterator it = object_value_.begin(); + it != object_value_.end(); ++it) { + keys.push_back(it->first); + } + + return keys; + } + + size_t Size() const { return (IsArray() ? ArrayLen() : Keys().size()); } + + bool operator==(const tinygltf::Value &other) const; + + protected: + int type_ = NULL_TYPE; + + int int_value_ = 0; + double real_value_ = 0.0; + std::string string_value_; + std::vector binary_value_; + Array array_value_; + Object object_value_; + bool boolean_value_ = false; +}; + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +#define TINYGLTF_VALUE_GET(ctype, var) \ + template <> \ + inline const ctype &Value::Get() const { \ + return var; \ + } \ + template <> \ + inline ctype &Value::Get() { \ + return var; \ + } +TINYGLTF_VALUE_GET(bool, boolean_value_) +TINYGLTF_VALUE_GET(double, real_value_) +TINYGLTF_VALUE_GET(int, int_value_) +TINYGLTF_VALUE_GET(std::string, string_value_) +TINYGLTF_VALUE_GET(std::vector, binary_value_) +TINYGLTF_VALUE_GET(Value::Array, array_value_) +TINYGLTF_VALUE_GET(Value::Object, object_value_) +#undef TINYGLTF_VALUE_GET + +#ifdef __clang__ +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wc++98-compat" +#pragma clang diagnostic ignored "-Wpadded" +#endif + +/// Aggregate object for representing a color +using ColorValue = std::array; + +// === legacy interface ==== +// TODO(syoyo): Deprecate `Parameter` class. +struct Parameter { + bool bool_value = false; + bool has_number_value = false; + std::string string_value; + std::vector number_array; + std::map json_double_value; + double number_value = 0.0; + + // context sensitive methods. depending the type of the Parameter you are + // accessing, these are either valid or not + // If this parameter represent a texture map in a material, will return the + // texture index + + /// Return the index of a texture if this Parameter is a texture map. + /// Returned value is only valid if the parameter represent a texture from a + /// material + int TextureIndex() const { + const auto it = json_double_value.find("index"); + if (it != std::end(json_double_value)) { + return int(it->second); + } + return -1; + } + + /// Return the index of a texture coordinate set if this Parameter is a + /// texture map. Returned value is only valid if the parameter represent a + /// texture from a material + int TextureTexCoord() const { + const auto it = json_double_value.find("texCoord"); + if (it != std::end(json_double_value)) { + return int(it->second); + } + // As per the spec, if texCoord is omitted, this parameter is 0 + return 0; + } + + /// Return the scale of a texture if this Parameter is a normal texture map. + /// Returned value is only valid if the parameter represent a normal texture + /// from a material + double TextureScale() const { + const auto it = json_double_value.find("scale"); + if (it != std::end(json_double_value)) { + return it->second; + } + // As per the spec, if scale is omitted, this parameter is 1 + return 1; + } + + /// Return the strength of a texture if this Parameter is a an occlusion map. + /// Returned value is only valid if the parameter represent an occlusion map + /// from a material + double TextureStrength() const { + const auto it = json_double_value.find("strength"); + if (it != std::end(json_double_value)) { + return it->second; + } + // As per the spec, if strength is omitted, this parameter is 1 + return 1; + } + + /// Material factor, like the roughness or metalness of a material + /// Returned value is only valid if the parameter represent a texture from a + /// material + double Factor() const { return number_value; } + + /// Return the color of a material + /// Returned value is only valid if the parameter represent a texture from a + /// material + ColorValue ColorFactor() const { + return { + {// this aggregate initialize the std::array object, and uses C++11 RVO. + number_array[0], number_array[1], number_array[2], + (number_array.size() > 3 ? number_array[3] : 1.0)}}; + } + + Parameter() = default; + DEFAULT_METHODS(Parameter) + bool operator==(const Parameter &) const; +}; + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +#ifdef __clang__ +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wpadded" +#endif + +typedef std::map ParameterMap; +typedef std::map ExtensionMap; + +struct AnimationChannel { + int sampler; // required + int target_node; // optional index of the node to target (alternative + // target should be provided by extension) + std::string target_path; // required with standard values of ["translation", + // "rotation", "scale", "weights"] + Value extras; + ExtensionMap extensions; + ExtensionMap target_extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + std::string target_extensions_json_string; + + AnimationChannel() : sampler(-1), target_node(-1) {} + DEFAULT_METHODS(AnimationChannel) + bool operator==(const AnimationChannel &) const; +}; + +struct AnimationSampler { + int input; // required + int output; // required + std::string interpolation; // "LINEAR", "STEP","CUBICSPLINE" or user defined + // string. default "LINEAR" + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + AnimationSampler() : input(-1), output(-1), interpolation("LINEAR") {} + DEFAULT_METHODS(AnimationSampler) + bool operator==(const AnimationSampler &) const; +}; + +struct Animation { + std::string name; + std::vector channels; + std::vector samplers; + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Animation() = default; + DEFAULT_METHODS(Animation) + bool operator==(const Animation &) const; +}; + +struct Skin { + std::string name; + int inverseBindMatrices; // required here but not in the spec + int skeleton; // The index of the node used as a skeleton root + std::vector joints; // Indices of skeleton nodes + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Skin() { + inverseBindMatrices = -1; + skeleton = -1; + } + DEFAULT_METHODS(Skin) + bool operator==(const Skin &) const; +}; + +struct Sampler { + std::string name; + // glTF 2.0 spec does not define default value for `minFilter` and + // `magFilter`. Set -1 in TinyGLTF(issue #186) + int minFilter = + -1; // optional. -1 = no filter defined. ["NEAREST", "LINEAR", + // "NEAREST_MIPMAP_NEAREST", "LINEAR_MIPMAP_NEAREST", + // "NEAREST_MIPMAP_LINEAR", "LINEAR_MIPMAP_LINEAR"] + int magFilter = + -1; // optional. -1 = no filter defined. ["NEAREST", "LINEAR"] + int wrapS = + TINYGLTF_TEXTURE_WRAP_REPEAT; // ["CLAMP_TO_EDGE", "MIRRORED_REPEAT", + // "REPEAT"], default "REPEAT" + int wrapT = + TINYGLTF_TEXTURE_WRAP_REPEAT; // ["CLAMP_TO_EDGE", "MIRRORED_REPEAT", + // "REPEAT"], default "REPEAT" + // int wrapR = TINYGLTF_TEXTURE_WRAP_REPEAT; // TinyGLTF extension. currently + // not used. + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Sampler() + : minFilter(-1), + magFilter(-1), + wrapS(TINYGLTF_TEXTURE_WRAP_REPEAT), + wrapT(TINYGLTF_TEXTURE_WRAP_REPEAT) {} + DEFAULT_METHODS(Sampler) + bool operator==(const Sampler &) const; +}; + +struct Image { + std::string name; + int width; + int height; + int component; + int bits; // bit depth per channel. 8(byte), 16 or 32. + int pixel_type; // pixel type(TINYGLTF_COMPONENT_TYPE_***). usually + // UBYTE(bits = 8) or USHORT(bits = 16) + std::vector image; + int bufferView; // (required if no uri) + std::string mimeType; // (required if no uri) ["image/jpeg", "image/png", + // "image/bmp", "image/gif"] + std::string uri; // (required if no mimeType) uri is not decoded(e.g. + // whitespace may be represented as %20) + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + // When this flag is true, data is stored to `image` in as-is format(e.g. jpeg + // compressed for "image/jpeg" mime) This feature is good if you use custom + // image loader function. (e.g. delayed decoding of images for faster glTF + // parsing) Default parser for Image does not provide as-is loading feature at + // the moment. (You can manipulate this by providing your own LoadImageData + // function) + bool as_is; + + Image() : as_is(false) { + bufferView = -1; + width = -1; + height = -1; + component = -1; + bits = -1; + pixel_type = -1; + } + DEFAULT_METHODS(Image) + + bool operator==(const Image &) const; +}; + +struct Texture { + std::string name; + + int sampler; + int source; + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Texture() : sampler(-1), source(-1) {} + DEFAULT_METHODS(Texture) + + bool operator==(const Texture &) const; +}; + +struct TextureInfo { + int index = -1; // required. + int texCoord; // The set index of texture's TEXCOORD attribute used for + // texture coordinate mapping. + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + TextureInfo() : index(-1), texCoord(0) {} + DEFAULT_METHODS(TextureInfo) + bool operator==(const TextureInfo &) const; +}; + +struct NormalTextureInfo { + int index = -1; // required + int texCoord; // The set index of texture's TEXCOORD attribute used for + // texture coordinate mapping. + double scale; // scaledNormal = normalize(( + // * 2.0 - 1.0) * vec3(, , 1.0)) + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + NormalTextureInfo() : index(-1), texCoord(0), scale(1.0) {} + DEFAULT_METHODS(NormalTextureInfo) + bool operator==(const NormalTextureInfo &) const; +}; + +struct OcclusionTextureInfo { + int index = -1; // required + int texCoord; // The set index of texture's TEXCOORD attribute used for + // texture coordinate mapping. + double strength; // occludedColor = lerp(color, color * , ) + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + OcclusionTextureInfo() : index(-1), texCoord(0), strength(1.0) {} + DEFAULT_METHODS(OcclusionTextureInfo) + bool operator==(const OcclusionTextureInfo &) const; +}; + +// pbrMetallicRoughness class defined in glTF 2.0 spec. +struct PbrMetallicRoughness { + std::vector baseColorFactor; // len = 4. default [1,1,1,1] + TextureInfo baseColorTexture; + double metallicFactor; // default 1 + double roughnessFactor; // default 1 + TextureInfo metallicRoughnessTexture; + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + PbrMetallicRoughness() + : baseColorFactor(std::vector{1.0, 1.0, 1.0, 1.0}), + metallicFactor(1.0), + roughnessFactor(1.0) {} + DEFAULT_METHODS(PbrMetallicRoughness) + bool operator==(const PbrMetallicRoughness &) const; +}; + +// Each extension should be stored in a ParameterMap. +// members not in the values could be included in the ParameterMap +// to keep a single material model +struct Material { + std::string name; + + std::vector emissiveFactor; // length 3. default [0, 0, 0] + std::string alphaMode; // default "OPAQUE" + double alphaCutoff; // default 0.5 + bool doubleSided; // default false; + + PbrMetallicRoughness pbrMetallicRoughness; + + NormalTextureInfo normalTexture; + OcclusionTextureInfo occlusionTexture; + TextureInfo emissiveTexture; + + // For backward compatibility + // TODO(syoyo): Remove `values` and `additionalValues` in the next release. + ParameterMap values; + ParameterMap additionalValues; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Material() : alphaMode("OPAQUE"), alphaCutoff(0.5), doubleSided(false) {} + DEFAULT_METHODS(Material) + + bool operator==(const Material &) const; +}; + +struct BufferView { + std::string name; + int buffer{-1}; // Required + size_t byteOffset{0}; // minimum 0, default 0 + size_t byteLength{0}; // required, minimum 1. 0 = invalid + size_t byteStride{0}; // minimum 4, maximum 252 (multiple of 4), default 0 = + // understood to be tightly packed + int target{0}; // ["ARRAY_BUFFER", "ELEMENT_ARRAY_BUFFER"] for vertex indices + // or attribs. Could be 0 for other data + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + bool dracoDecoded{false}; // Flag indicating this has been draco decoded + + BufferView() + : buffer(-1), + byteOffset(0), + byteLength(0), + byteStride(0), + target(0), + dracoDecoded(false) {} + DEFAULT_METHODS(BufferView) + bool operator==(const BufferView &) const; +}; + +struct Accessor { + int bufferView; // optional in spec but required here since sparse accessor + // are not supported + std::string name; + size_t byteOffset; + bool normalized; // optional. + int componentType; // (required) One of TINYGLTF_COMPONENT_TYPE_*** + size_t count; // required + int type; // (required) One of TINYGLTF_TYPE_*** .. + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + std::vector + minValues; // optional. integer value is promoted to double + std::vector + maxValues; // optional. integer value is promoted to double + + struct { + int count; + bool isSparse; + struct { + int byteOffset; + int bufferView; + int componentType; // a TINYGLTF_COMPONENT_TYPE_ value + } indices; + struct { + int bufferView; + int byteOffset; + } values; + } sparse; + + /// + /// Utility function to compute byteStride for a given bufferView object. + /// Returns -1 upon invalid glTF value or parameter configuration. + /// + int ByteStride(const BufferView &bufferViewObject) const { + if (bufferViewObject.byteStride == 0) { + // Assume data is tightly packed. + int componentSizeInBytes = + GetComponentSizeInBytes(static_cast(componentType)); + if (componentSizeInBytes <= 0) { + return -1; + } + + int numComponents = GetNumComponentsInType(static_cast(type)); + if (numComponents <= 0) { + return -1; + } + + return componentSizeInBytes * numComponents; + } else { + // Check if byteStride is a multiple of the size of the accessor's component + // type. + int componentSizeInBytes = + GetComponentSizeInBytes(static_cast(componentType)); + if (componentSizeInBytes <= 0) { + return -1; + } + + if ((bufferViewObject.byteStride % uint32_t(componentSizeInBytes)) != 0) { + return -1; + } + return static_cast(bufferViewObject.byteStride); + } + + // unreachable return 0; + } + + Accessor() + : bufferView(-1), + byteOffset(0), + normalized(false), + componentType(-1), + count(0), + type(-1) { + sparse.isSparse = false; + } + DEFAULT_METHODS(Accessor) + bool operator==(const tinygltf::Accessor &) const; +}; + +struct PerspectiveCamera { + double aspectRatio; // min > 0 + double yfov; // required. min > 0 + double zfar; // min > 0 + double znear; // required. min > 0 + + PerspectiveCamera() + : aspectRatio(0.0), + yfov(0.0), + zfar(0.0) // 0 = use infinite projection matrix + , + znear(0.0) {} + DEFAULT_METHODS(PerspectiveCamera) + bool operator==(const PerspectiveCamera &) const; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +struct OrthographicCamera { + double xmag; // required. must not be zero. + double ymag; // required. must not be zero. + double zfar; // required. `zfar` must be greater than `znear`. + double znear; // required + + OrthographicCamera() : xmag(0.0), ymag(0.0), zfar(0.0), znear(0.0) {} + DEFAULT_METHODS(OrthographicCamera) + bool operator==(const OrthographicCamera &) const; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +struct Camera { + std::string type; // required. "perspective" or "orthographic" + std::string name; + + PerspectiveCamera perspective; + OrthographicCamera orthographic; + + Camera() {} + DEFAULT_METHODS(Camera) + bool operator==(const Camera &) const; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +struct Primitive { + std::map attributes; // (required) A dictionary object of + // integer, where each integer + // is the index of the accessor + // containing an attribute. + int material; // The index of the material to apply to this primitive + // when rendering. + int indices; // The index of the accessor that contains the indices. + int mode; // one of TINYGLTF_MODE_*** + std::vector > targets; // array of morph targets, + // where each target is a dict with attributes in ["POSITION, "NORMAL", + // "TANGENT"] pointing + // to their corresponding accessors + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Primitive() { + material = -1; + indices = -1; + mode = -1; + } + DEFAULT_METHODS(Primitive) + bool operator==(const Primitive &) const; +}; + +struct Mesh { + std::string name; + std::vector primitives; + std::vector weights; // weights to be applied to the Morph Targets + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Mesh() = default; + DEFAULT_METHODS(Mesh) + bool operator==(const Mesh &) const; +}; + +class Node { + public: + Node() : camera(-1), skin(-1), mesh(-1) {} + + DEFAULT_METHODS(Node) + + bool operator==(const Node &) const; + + int camera; // the index of the camera referenced by this node + + std::string name; + int skin; + int mesh; + std::vector children; + std::vector rotation; // length must be 0 or 4 + std::vector scale; // length must be 0 or 3 + std::vector translation; // length must be 0 or 3 + std::vector matrix; // length must be 0 or 16 + std::vector weights; // The weights of the instantiated Morph Target + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +struct Buffer { + std::string name; + std::vector data; + std::string + uri; // considered as required here but not in the spec (need to clarify) + // uri is not decoded(e.g. whitespace may be represented as %20) + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Buffer() = default; + DEFAULT_METHODS(Buffer) + bool operator==(const Buffer &) const; +}; + +struct Asset { + std::string version = "2.0"; // required + std::string generator; + std::string minVersion; + std::string copyright; + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Asset() = default; + DEFAULT_METHODS(Asset) + bool operator==(const Asset &) const; +}; + +struct Scene { + std::string name; + std::vector nodes; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; + + Scene() = default; + DEFAULT_METHODS(Scene) + bool operator==(const Scene &) const; +}; + +struct SpotLight { + double innerConeAngle; + double outerConeAngle; + + SpotLight() : innerConeAngle(0.0), outerConeAngle(0.7853981634) {} + DEFAULT_METHODS(SpotLight) + bool operator==(const SpotLight &) const; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +struct Light { + std::string name; + std::vector color; + double intensity{1.0}; + std::string type; + double range{0.0}; // 0.0 = infinite + SpotLight spot; + + Light() : intensity(1.0), range(0.0) {} + DEFAULT_METHODS(Light) + + bool operator==(const Light &) const; + + ExtensionMap extensions; + Value extras; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +class Model { + public: + Model() = default; + DEFAULT_METHODS(Model) + + bool operator==(const Model &) const; + + std::vector accessors; + std::vector animations; + std::vector buffers; + std::vector bufferViews; + std::vector materials; + std::vector meshes; + std::vector nodes; + std::vector textures; + std::vector images; + std::vector skins; + std::vector samplers; + std::vector cameras; + std::vector scenes; + std::vector lights; + + int defaultScene = -1; + std::vector extensionsUsed; + std::vector extensionsRequired; + + Asset asset; + + Value extras; + ExtensionMap extensions; + + // Filled when SetStoreOriginalJSONForExtrasAndExtensions is enabled. + std::string extras_json_string; + std::string extensions_json_string; +}; + +enum SectionCheck { + NO_REQUIRE = 0x00, + REQUIRE_VERSION = 0x01, + REQUIRE_SCENE = 0x02, + REQUIRE_SCENES = 0x04, + REQUIRE_NODES = 0x08, + REQUIRE_ACCESSORS = 0x10, + REQUIRE_BUFFERS = 0x20, + REQUIRE_BUFFER_VIEWS = 0x40, + REQUIRE_ALL = 0x7f +}; + +/// +/// URIEncodeFunction type. Signature for custom URI encoding of external +/// resources such as .bin and image files. Used by tinygltf to re-encode the +/// final location of saved files. object_type may be used to encode buffer and +/// image URIs differently, for example. See +/// https://registry.khronos.org/glTF/specs/2.0/glTF-2.0.html#uris +/// +typedef bool (*URIEncodeFunction)(const std::string &in_uri, + const std::string &object_type, + std::string *out_uri, void *user_data); + +/// +/// URIDecodeFunction type. Signature for custom URI decoding of external +/// resources such as .bin and image files. Used by tinygltf when computing +/// filenames to write resources. +/// +typedef bool (*URIDecodeFunction)(const std::string &in_uri, + std::string *out_uri, void *user_data); + +// Declaration of default uri decode function +bool URIDecode(const std::string &in_uri, std::string *out_uri, + void *user_data); + +/// +/// A structure containing URI callbacks and a pointer to their user data. +/// +struct URICallbacks { + URIEncodeFunction encode; // Optional encode method + URIDecodeFunction decode; // Required decode method + + void *user_data; // An argument that is passed to all uri callbacks +}; + +/// +/// LoadImageDataFunction type. Signature for custom image loading callbacks. +/// +typedef bool (*LoadImageDataFunction)(Image *, const int, std::string *, + std::string *, int, int, + const unsigned char *, int, + void *user_pointer); + +/// +/// WriteImageDataFunction type. Signature for custom image writing callbacks. +/// The out_uri parameter becomes the URI written to the gltf and may reference +/// a file or contain a data URI. +/// +typedef bool (*WriteImageDataFunction)(const std::string *basepath, + const std::string *filename, + const Image *image, bool embedImages, + const URICallbacks *uri_cb, + std::string *out_uri, + void *user_pointer); + +#ifndef TINYGLTF_NO_STB_IMAGE +// Declaration of default image loader callback +bool LoadImageData(Image *image, const int image_idx, std::string *err, + std::string *warn, int req_width, int req_height, + const unsigned char *bytes, int size, void *); +#endif + +#ifndef TINYGLTF_NO_STB_IMAGE_WRITE +// Declaration of default image writer callback +bool WriteImageData(const std::string *basepath, const std::string *filename, + const Image *image, bool embedImages, + const URICallbacks *uri_cb, std::string *out_uri, void *); +#endif + +/// +/// FilExistsFunction type. Signature for custom filesystem callbacks. +/// +typedef bool (*FileExistsFunction)(const std::string &abs_filename, void *); + +/// +/// ExpandFilePathFunction type. Signature for custom filesystem callbacks. +/// +typedef std::string (*ExpandFilePathFunction)(const std::string &, void *); + +/// +/// ReadWholeFileFunction type. Signature for custom filesystem callbacks. +/// +typedef bool (*ReadWholeFileFunction)(std::vector *, + std::string *, const std::string &, + void *); + +/// +/// WriteWholeFileFunction type. Signature for custom filesystem callbacks. +/// +typedef bool (*WriteWholeFileFunction)(std::string *, const std::string &, + const std::vector &, + void *); + +/// +/// GetFileSizeFunction type. Signature for custom filesystem callbacks. +/// +typedef bool (*GetFileSizeFunction)(size_t *filesize_out, std::string *err, const std::string &abs_filename, + void *userdata); + +/// +/// A structure containing all required filesystem callbacks and a pointer to +/// their user data. +/// +struct FsCallbacks { + FileExistsFunction FileExists; + ExpandFilePathFunction ExpandFilePath; + ReadWholeFileFunction ReadWholeFile; + WriteWholeFileFunction WriteWholeFile; + GetFileSizeFunction GetFileSizeInBytes; // To avoid GetFileSize Win32 API, add `InBytes` suffix. + + void *user_data; // An argument that is passed to all fs callbacks +}; + +#ifndef TINYGLTF_NO_FS +// Declaration of default filesystem callbacks + +bool FileExists(const std::string &abs_filename, void *); + +/// +/// Expand file path(e.g. `~` to home directory on posix, `%APPDATA%` to +/// `C:\\Users\\tinygltf\\AppData`) +/// +/// @param[in] filepath File path string. Assume UTF-8 +/// @param[in] userdata User data. Set to `nullptr` if you don't need it. +/// +std::string ExpandFilePath(const std::string &filepath, void *userdata); + +bool ReadWholeFile(std::vector *out, std::string *err, + const std::string &filepath, void *); + +bool WriteWholeFile(std::string *err, const std::string &filepath, + const std::vector &contents, void *); + +bool GetFileSizeInBytes(size_t *filesize_out, std::string *err, const std::string &filepath, + void *); +#endif + +/// +/// glTF Parser/Serializer context. +/// +class TinyGLTF { + public: +#ifdef __clang__ +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wc++98-compat" +#endif + + TinyGLTF() : bin_data_(nullptr), bin_size_(0), is_binary_(false) {} + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + + ~TinyGLTF() {} + + /// + /// Loads glTF ASCII asset from a file. + /// Set warning message to `warn` for example it fails to load asserts. + /// Returns false and set error string to `err` if there's an error. + /// + bool LoadASCIIFromFile(Model *model, std::string *err, std::string *warn, + const std::string &filename, + unsigned int check_sections = REQUIRE_VERSION); + + /// + /// Loads glTF ASCII asset from string(memory). + /// `length` = strlen(str); + /// `base_dir` is a search path of glTF asset(e.g. images). Path Must be an + /// expanded path (e.g. no tilde(`~`), no environment variables). Set warning + /// message to `warn` for example it fails to load asserts. Returns false and + /// set error string to `err` if there's an error. + /// + bool LoadASCIIFromString(Model *model, std::string *err, std::string *warn, + const char *str, const unsigned int length, + const std::string &base_dir, + unsigned int check_sections = REQUIRE_VERSION); + + /// + /// Loads glTF binary asset from a file. + /// Set warning message to `warn` for example it fails to load asserts. + /// Returns false and set error string to `err` if there's an error. + /// + bool LoadBinaryFromFile(Model *model, std::string *err, std::string *warn, + const std::string &filename, + unsigned int check_sections = REQUIRE_VERSION); + + /// + /// Loads glTF binary asset from memory. + /// `length` = strlen(str); + /// `base_dir` is a search path of glTF asset(e.g. images). Path Must be an + /// expanded path (e.g. no tilde(`~`), no environment variables). + /// Set warning message to `warn` for example it fails to load asserts. + /// Returns false and set error string to `err` if there's an error. + /// + bool LoadBinaryFromMemory(Model *model, std::string *err, std::string *warn, + const unsigned char *bytes, + const unsigned int length, + const std::string &base_dir = "", + unsigned int check_sections = REQUIRE_VERSION); + + /// + /// Write glTF to stream, buffers and images will be embedded + /// + bool WriteGltfSceneToStream(const Model *model, std::ostream &stream, + bool prettyPrint, bool writeBinary); + + /// + /// Write glTF to file. + /// + bool WriteGltfSceneToFile(const Model *model, const std::string &filename, + bool embedImages, bool embedBuffers, + bool prettyPrint, bool writeBinary); + + /// + /// Set callback to use for loading image data + /// + void SetImageLoader(LoadImageDataFunction LoadImageData, void *user_data); + + /// + /// Unset(remove) callback of loading image data + /// + void RemoveImageLoader(); + + /// + /// Set callback to use for writing image data + /// + void SetImageWriter(WriteImageDataFunction WriteImageData, void *user_data); + + /// + /// Set callbacks to use for URI encoding and decoding and their user data + /// + void SetURICallbacks(URICallbacks callbacks); + + /// + /// Set callbacks to use for filesystem (fs) access and their user data + /// + void SetFsCallbacks(FsCallbacks callbacks); + + /// + /// Set serializing default values(default = false). + /// When true, default values are force serialized to .glTF. + /// This may be helpful if you want to serialize a full description of glTF + /// data. + /// + /// TODO(LTE): Supply parsing option as function arguments to + /// `LoadASCIIFromFile()` and others, not by a class method + /// + void SetSerializeDefaultValues(const bool enabled) { + serialize_default_values_ = enabled; + } + + bool GetSerializeDefaultValues() const { return serialize_default_values_; } + + /// + /// Store original JSON string for `extras` and `extensions`. + /// This feature will be useful when the user want to reconstruct custom data + /// structure from JSON string. + /// + void SetStoreOriginalJSONForExtrasAndExtensions(const bool enabled) { + store_original_json_for_extras_and_extensions_ = enabled; + } + + bool GetStoreOriginalJSONForExtrasAndExtensions() const { + return store_original_json_for_extras_and_extensions_; + } + + /// + /// Specify whether preserve image channels when loading images or not. + /// (Not effective when the user supplies their own LoadImageData callbacks) + /// + void SetPreserveImageChannels(bool onoff) { + preserve_image_channels_ = onoff; + } + + /// + /// Set maximum allowed external file size in bytes. + /// Default: 2GB + /// Only effective for built-in ReadWholeFileFunction FS function. + /// + void SetMaxExternalFileSize(size_t max_bytes) { + max_external_file_size_ = max_bytes; + } + + size_t GetMaxExternalFileSize() const { + return max_external_file_size_; + } + + bool GetPreserveImageChannels() const { return preserve_image_channels_; } + + private: + /// + /// Loads glTF asset from string(memory). + /// `length` = strlen(str); + /// Set warning message to `warn` for example it fails to load asserts + /// Returns false and set error string to `err` if there's an error. + /// + bool LoadFromString(Model *model, std::string *err, std::string *warn, + const char *str, const unsigned int length, + const std::string &base_dir, unsigned int check_sections); + + const unsigned char *bin_data_ = nullptr; + size_t bin_size_ = 0; + bool is_binary_ = false; + + bool serialize_default_values_ = false; ///< Serialize default values? + + bool store_original_json_for_extras_and_extensions_ = false; + + bool preserve_image_channels_ = false; /// Default false(expand channels to + /// RGBA) for backward compatibility. + + size_t max_external_file_size_{size_t((std::numeric_limits::max)())}; // Default 2GB + + // Warning & error messages + std::string warn_; + std::string err_; + + FsCallbacks fs = { +#ifndef TINYGLTF_NO_FS + &tinygltf::FileExists, &tinygltf::ExpandFilePath, + &tinygltf::ReadWholeFile, &tinygltf::WriteWholeFile, &tinygltf::GetFileSizeInBytes, + + nullptr // Fs callback user data +#else + nullptr, nullptr, nullptr, nullptr, nullptr, + + nullptr // Fs callback user data +#endif + }; + + URICallbacks uri_cb = { + // Use paths as-is by default. This will use JSON string escaping. + nullptr, + // Decode all URIs before using them as paths as the application may have + // percent encoded them. + &tinygltf::URIDecode, + // URI callback user data + nullptr}; + + LoadImageDataFunction LoadImageData = +#ifndef TINYGLTF_NO_STB_IMAGE + &tinygltf::LoadImageData; +#else + nullptr; +#endif + void *load_image_user_data_{nullptr}; + bool user_image_loader_{false}; + + WriteImageDataFunction WriteImageData = +#ifndef TINYGLTF_NO_STB_IMAGE_WRITE + &tinygltf::WriteImageData; +#else + nullptr; +#endif + void *write_image_user_data_{nullptr}; +}; + +#ifdef __clang__ +#pragma clang diagnostic pop // -Wpadded +#endif + +} // namespace tinygltf + +#endif // TINY_GLTF_H_ + +#if defined(TINYGLTF_IMPLEMENTATION) || defined(__INTELLISENSE__) +#include +//#include +#ifndef TINYGLTF_NO_FS +#include +#include +#include // for is_directory check +#endif +#include + +#ifdef __clang__ +// Disable some warnings for external files. +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wfloat-equal" +#pragma clang diagnostic ignored "-Wexit-time-destructors" +#pragma clang diagnostic ignored "-Wconversion" +#pragma clang diagnostic ignored "-Wold-style-cast" +#pragma clang diagnostic ignored "-Wglobal-constructors" +#if __has_warning("-Wreserved-id-macro") +#pragma clang diagnostic ignored "-Wreserved-id-macro" +#endif +#pragma clang diagnostic ignored "-Wdisabled-macro-expansion" +#pragma clang diagnostic ignored "-Wpadded" +#pragma clang diagnostic ignored "-Wc++98-compat" +#pragma clang diagnostic ignored "-Wc++98-compat-pedantic" +#pragma clang diagnostic ignored "-Wdocumentation-unknown-command" +#pragma clang diagnostic ignored "-Wswitch-enum" +#pragma clang diagnostic ignored "-Wimplicit-fallthrough" +#pragma clang diagnostic ignored "-Wweak-vtables" +#pragma clang diagnostic ignored "-Wcovered-switch-default" +#if __has_warning("-Wdouble-promotion") +#pragma clang diagnostic ignored "-Wdouble-promotion" +#endif +#if __has_warning("-Wcomma") +#pragma clang diagnostic ignored "-Wcomma" +#endif +#if __has_warning("-Wzero-as-null-pointer-constant") +#pragma clang diagnostic ignored "-Wzero-as-null-pointer-constant" +#endif +#if __has_warning("-Wcast-qual") +#pragma clang diagnostic ignored "-Wcast-qual" +#endif +#if __has_warning("-Wmissing-variable-declarations") +#pragma clang diagnostic ignored "-Wmissing-variable-declarations" +#endif +#if __has_warning("-Wmissing-prototypes") +#pragma clang diagnostic ignored "-Wmissing-prototypes" +#endif +#if __has_warning("-Wcast-align") +#pragma clang diagnostic ignored "-Wcast-align" +#endif +#if __has_warning("-Wnewline-eof") +#pragma clang diagnostic ignored "-Wnewline-eof" +#endif +#if __has_warning("-Wunused-parameter") +#pragma clang diagnostic ignored "-Wunused-parameter" +#endif +#if __has_warning("-Wmismatched-tags") +#pragma clang diagnostic ignored "-Wmismatched-tags" +#endif +#if __has_warning("-Wextra-semi-stmt") +#pragma clang diagnostic ignored "-Wextra-semi-stmt" +#endif +#endif + +// Disable GCC warnings +#ifdef __GNUC__ +#pragma GCC diagnostic push +#pragma GCC diagnostic ignored "-Wtype-limits" +#endif // __GNUC__ + +#ifndef TINYGLTF_NO_INCLUDE_JSON +#ifndef TINYGLTF_USE_RAPIDJSON +#include "json.hpp" +#else +#ifndef TINYGLTF_NO_INCLUDE_RAPIDJSON +#include "document.h" +#include "prettywriter.h" +#include "rapidjson.h" +#include "stringbuffer.h" +#include "writer.h" +#endif +#endif +#endif + +#ifdef TINYGLTF_ENABLE_DRACO +#include "draco/compression/decode.h" +#include "draco/core/decoder_buffer.h" +#endif + +#ifndef TINYGLTF_NO_STB_IMAGE +#ifndef TINYGLTF_NO_INCLUDE_STB_IMAGE +#include "stb_image.h" +#endif +#endif + +#ifndef TINYGLTF_NO_STB_IMAGE_WRITE +#ifndef TINYGLTF_NO_INCLUDE_STB_IMAGE_WRITE +#include "stb_image_write.h" +#endif +#endif + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +#ifdef __GNUC__ +#pragma GCC diagnostic pop +#endif + +#ifdef _WIN32 + +// issue 143. +// Define NOMINMAX to avoid min/max defines, +// but undef it after included Windows.h +#ifndef NOMINMAX +#define TINYGLTF_INTERNAL_NOMINMAX +#define NOMINMAX +#endif + +#ifndef WIN32_LEAN_AND_MEAN +#define WIN32_LEAN_AND_MEAN +#define TINYGLTF_INTERNAL_WIN32_LEAN_AND_MEAN +#endif +#ifndef __MINGW32__ +#include // include API for expanding a file path +#else +#include +#endif + +#ifdef TINYGLTF_INTERNAL_WIN32_LEAN_AND_MEAN +#undef WIN32_LEAN_AND_MEAN +#endif + +#if defined(TINYGLTF_INTERNAL_NOMINMAX) +#undef NOMINMAX +#endif + +#if defined(__GLIBCXX__) // mingw + +#include // _O_RDONLY + +#include // fstream (all sorts of IO stuff) + stdio_filebuf (=streambuf) + +#endif + +#elif !defined(__ANDROID__) && !defined(__OpenBSD__) +//#include +#endif + +#if defined(__sparcv9) || defined(__powerpc__) +// Big endian +#else +#if (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__) || MINIZ_X86_OR_X64_CPU +#define TINYGLTF_LITTLE_ENDIAN 1 +#endif +#endif + +namespace tinygltf { +namespace detail { +#ifdef TINYGLTF_USE_RAPIDJSON + +#ifdef TINYGLTF_USE_RAPIDJSON_CRTALLOCATOR +// This uses the RapidJSON CRTAllocator. It is thread safe and multiple +// documents may be active at once. +using json = + rapidjson::GenericValue, rapidjson::CrtAllocator>; +using json_const_iterator = json::ConstMemberIterator; +using json_const_array_iterator = json const *; +using JsonDocument = + rapidjson::GenericDocument, rapidjson::CrtAllocator>; +rapidjson::CrtAllocator s_CrtAllocator; // stateless and thread safe +rapidjson::CrtAllocator &GetAllocator() { return s_CrtAllocator; } +#else +// This uses the default RapidJSON MemoryPoolAllocator. It is very fast, but +// not thread safe. Only a single JsonDocument may be active at any one time, +// meaning only a single gltf load/save can be active any one time. +using json = rapidjson::Value; +using json_const_iterator = json::ConstMemberIterator; +using json_const_array_iterator = json const *; +rapidjson::Document *s_pActiveDocument = nullptr; +rapidjson::Document::AllocatorType &GetAllocator() { + assert(s_pActiveDocument); // Root json node must be JsonDocument type + return s_pActiveDocument->GetAllocator(); +} + +#ifdef __clang__ +#pragma clang diagnostic push +// Suppress JsonDocument(JsonDocument &&rhs) noexcept +#pragma clang diagnostic ignored "-Wunused-member-function" +#endif + +struct JsonDocument : public rapidjson::Document { + JsonDocument() { + assert(s_pActiveDocument == + nullptr); // When using default allocator, only one document can be + // active at a time, if you need multiple active at once, + // define TINYGLTF_USE_RAPIDJSON_CRTALLOCATOR + s_pActiveDocument = this; + } + JsonDocument(const JsonDocument &) = delete; + JsonDocument(JsonDocument &&rhs) noexcept + : rapidjson::Document(std::move(rhs)) { + s_pActiveDocument = this; + rhs.isNil = true; + } + ~JsonDocument() { + if (!isNil) { + s_pActiveDocument = nullptr; + } + } + + private: + bool isNil = false; +}; + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +#endif // TINYGLTF_USE_RAPIDJSON_CRTALLOCATOR + +#else +using nlohmann::json; +using json_const_iterator = json::const_iterator; +using json_const_array_iterator = json_const_iterator; +using JsonDocument = json; +#endif + +void JsonParse(JsonDocument &doc, const char *str, size_t length, + bool throwExc = false) { +#ifdef TINYGLTF_USE_RAPIDJSON + (void)throwExc; + doc.Parse(str, length); +#else + doc = detail::json::parse(str, str + length, nullptr, throwExc); +#endif +} +} // namespace +} + +#ifdef __APPLE__ +#include "TargetConditionals.h" +#endif + +#ifdef __clang__ +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wc++98-compat" +#endif + +namespace tinygltf { + +/// +/// Internal LoadImageDataOption struct. +/// This struct is passed through `user_pointer` in LoadImageData. +/// The struct is not passed when the user supply their own LoadImageData +/// callbacks. +/// +struct LoadImageDataOption { + // true: preserve image channels(e.g. load as RGB image if the image has RGB + // channels) default `false`(channels are expanded to RGBA for backward + // compatibility). + bool preserve_channels{false}; +}; + +// Equals function for Value, for recursivity +static bool Equals(const tinygltf::Value &one, const tinygltf::Value &other) { + if (one.Type() != other.Type()) return false; + + switch (one.Type()) { + case NULL_TYPE: + return true; + case BOOL_TYPE: + return one.Get() == other.Get(); + case REAL_TYPE: + return TINYGLTF_DOUBLE_EQUAL(one.Get(), other.Get()); + case INT_TYPE: + return one.Get() == other.Get(); + case OBJECT_TYPE: { + auto oneObj = one.Get(); + auto otherObj = other.Get(); + if (oneObj.size() != otherObj.size()) return false; + for (auto &it : oneObj) { + auto otherIt = otherObj.find(it.first); + if (otherIt == otherObj.end()) return false; + + if (!Equals(it.second, otherIt->second)) return false; + } + return true; + } + case ARRAY_TYPE: { + if (one.Size() != other.Size()) return false; + for (int i = 0; i < int(one.Size()); ++i) + if (!Equals(one.Get(i), other.Get(i))) return false; + return true; + } + case STRING_TYPE: + return one.Get() == other.Get(); + case BINARY_TYPE: + return one.Get >() == + other.Get >(); + default: { + // unhandled type + return false; + } + } +} + +// Equals function for std::vector using TINYGLTF_DOUBLE_EPSILON +static bool Equals(const std::vector &one, + const std::vector &other) { + if (one.size() != other.size()) return false; + for (int i = 0; i < int(one.size()); ++i) { + if (!TINYGLTF_DOUBLE_EQUAL(one[size_t(i)], other[size_t(i)])) return false; + } + return true; +} + +bool Accessor::operator==(const Accessor &other) const { + return this->bufferView == other.bufferView && + this->byteOffset == other.byteOffset && + this->componentType == other.componentType && + this->count == other.count && this->extensions == other.extensions && + this->extras == other.extras && + Equals(this->maxValues, other.maxValues) && + Equals(this->minValues, other.minValues) && this->name == other.name && + this->normalized == other.normalized && this->type == other.type; +} +bool Animation::operator==(const Animation &other) const { + return this->channels == other.channels && + this->extensions == other.extensions && this->extras == other.extras && + this->name == other.name && this->samplers == other.samplers; +} +bool AnimationChannel::operator==(const AnimationChannel &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->target_node == other.target_node && + this->target_path == other.target_path && + this->sampler == other.sampler; +} +bool AnimationSampler::operator==(const AnimationSampler &other) const { + return this->extras == other.extras && this->extensions == other.extensions && + this->input == other.input && + this->interpolation == other.interpolation && + this->output == other.output; +} +bool Asset::operator==(const Asset &other) const { + return this->copyright == other.copyright && + this->extensions == other.extensions && this->extras == other.extras && + this->generator == other.generator && + this->minVersion == other.minVersion && this->version == other.version; +} +bool Buffer::operator==(const Buffer &other) const { + return this->data == other.data && this->extensions == other.extensions && + this->extras == other.extras && this->name == other.name && + this->uri == other.uri; +} +bool BufferView::operator==(const BufferView &other) const { + return this->buffer == other.buffer && this->byteLength == other.byteLength && + this->byteOffset == other.byteOffset && + this->byteStride == other.byteStride && this->name == other.name && + this->target == other.target && this->extensions == other.extensions && + this->extras == other.extras && + this->dracoDecoded == other.dracoDecoded; +} +bool Camera::operator==(const Camera &other) const { + return this->name == other.name && this->extensions == other.extensions && + this->extras == other.extras && + this->orthographic == other.orthographic && + this->perspective == other.perspective && this->type == other.type; +} +bool Image::operator==(const Image &other) const { + return this->bufferView == other.bufferView && + this->component == other.component && + this->extensions == other.extensions && this->extras == other.extras && + this->height == other.height && this->image == other.image && + this->mimeType == other.mimeType && this->name == other.name && + this->uri == other.uri && this->width == other.width; +} +bool Light::operator==(const Light &other) const { + return Equals(this->color, other.color) && this->name == other.name && + this->type == other.type; +} +bool Material::operator==(const Material &other) const { + return (this->pbrMetallicRoughness == other.pbrMetallicRoughness) && + (this->normalTexture == other.normalTexture) && + (this->occlusionTexture == other.occlusionTexture) && + (this->emissiveTexture == other.emissiveTexture) && + Equals(this->emissiveFactor, other.emissiveFactor) && + (this->alphaMode == other.alphaMode) && + TINYGLTF_DOUBLE_EQUAL(this->alphaCutoff, other.alphaCutoff) && + (this->doubleSided == other.doubleSided) && + (this->extensions == other.extensions) && + (this->extras == other.extras) && (this->values == other.values) && + (this->additionalValues == other.additionalValues) && + (this->name == other.name); +} +bool Mesh::operator==(const Mesh &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->name == other.name && Equals(this->weights, other.weights) && + this->primitives == other.primitives; +} +bool Model::operator==(const Model &other) const { + return this->accessors == other.accessors && + this->animations == other.animations && this->asset == other.asset && + this->buffers == other.buffers && + this->bufferViews == other.bufferViews && + this->cameras == other.cameras && + this->defaultScene == other.defaultScene && + this->extensions == other.extensions && + this->extensionsRequired == other.extensionsRequired && + this->extensionsUsed == other.extensionsUsed && + this->extras == other.extras && this->images == other.images && + this->lights == other.lights && this->materials == other.materials && + this->meshes == other.meshes && this->nodes == other.nodes && + this->samplers == other.samplers && this->scenes == other.scenes && + this->skins == other.skins && this->textures == other.textures; +} +bool Node::operator==(const Node &other) const { + return this->camera == other.camera && this->children == other.children && + this->extensions == other.extensions && this->extras == other.extras && + Equals(this->matrix, other.matrix) && this->mesh == other.mesh && + this->name == other.name && Equals(this->rotation, other.rotation) && + Equals(this->scale, other.scale) && this->skin == other.skin && + Equals(this->translation, other.translation) && + Equals(this->weights, other.weights); +} +bool SpotLight::operator==(const SpotLight &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + TINYGLTF_DOUBLE_EQUAL(this->innerConeAngle, other.innerConeAngle) && + TINYGLTF_DOUBLE_EQUAL(this->outerConeAngle, other.outerConeAngle); +} +bool OrthographicCamera::operator==(const OrthographicCamera &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + TINYGLTF_DOUBLE_EQUAL(this->xmag, other.xmag) && + TINYGLTF_DOUBLE_EQUAL(this->ymag, other.ymag) && + TINYGLTF_DOUBLE_EQUAL(this->zfar, other.zfar) && + TINYGLTF_DOUBLE_EQUAL(this->znear, other.znear); +} +bool Parameter::operator==(const Parameter &other) const { + if (this->bool_value != other.bool_value || + this->has_number_value != other.has_number_value) + return false; + + if (!TINYGLTF_DOUBLE_EQUAL(this->number_value, other.number_value)) + return false; + + if (this->json_double_value.size() != other.json_double_value.size()) + return false; + for (auto &it : this->json_double_value) { + auto otherIt = other.json_double_value.find(it.first); + if (otherIt == other.json_double_value.end()) return false; + + if (!TINYGLTF_DOUBLE_EQUAL(it.second, otherIt->second)) return false; + } + + if (!Equals(this->number_array, other.number_array)) return false; + + if (this->string_value != other.string_value) return false; + + return true; +} +bool PerspectiveCamera::operator==(const PerspectiveCamera &other) const { + return TINYGLTF_DOUBLE_EQUAL(this->aspectRatio, other.aspectRatio) && + this->extensions == other.extensions && this->extras == other.extras && + TINYGLTF_DOUBLE_EQUAL(this->yfov, other.yfov) && + TINYGLTF_DOUBLE_EQUAL(this->zfar, other.zfar) && + TINYGLTF_DOUBLE_EQUAL(this->znear, other.znear); +} +bool Primitive::operator==(const Primitive &other) const { + return this->attributes == other.attributes && this->extras == other.extras && + this->indices == other.indices && this->material == other.material && + this->mode == other.mode && this->targets == other.targets; +} +bool Sampler::operator==(const Sampler &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->magFilter == other.magFilter && + this->minFilter == other.minFilter && this->name == other.name && + this->wrapS == other.wrapS && this->wrapT == other.wrapT; + + // this->wrapR == other.wrapR +} +bool Scene::operator==(const Scene &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->name == other.name && this->nodes == other.nodes; +} +bool Skin::operator==(const Skin &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->inverseBindMatrices == other.inverseBindMatrices && + this->joints == other.joints && this->name == other.name && + this->skeleton == other.skeleton; +} +bool Texture::operator==(const Texture &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->name == other.name && this->sampler == other.sampler && + this->source == other.source; +} +bool TextureInfo::operator==(const TextureInfo &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->index == other.index && this->texCoord == other.texCoord; +} +bool NormalTextureInfo::operator==(const NormalTextureInfo &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->index == other.index && this->texCoord == other.texCoord && + TINYGLTF_DOUBLE_EQUAL(this->scale, other.scale); +} +bool OcclusionTextureInfo::operator==(const OcclusionTextureInfo &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + this->index == other.index && this->texCoord == other.texCoord && + TINYGLTF_DOUBLE_EQUAL(this->strength, other.strength); +} +bool PbrMetallicRoughness::operator==(const PbrMetallicRoughness &other) const { + return this->extensions == other.extensions && this->extras == other.extras && + (this->baseColorTexture == other.baseColorTexture) && + (this->metallicRoughnessTexture == other.metallicRoughnessTexture) && + Equals(this->baseColorFactor, other.baseColorFactor) && + TINYGLTF_DOUBLE_EQUAL(this->metallicFactor, other.metallicFactor) && + TINYGLTF_DOUBLE_EQUAL(this->roughnessFactor, other.roughnessFactor); +} +bool Value::operator==(const Value &other) const { + return Equals(*this, other); +} + +static void swap4(unsigned int *val) { +#ifdef TINYGLTF_LITTLE_ENDIAN + (void)val; +#else + unsigned int tmp = *val; + unsigned char *dst = reinterpret_cast(val); + unsigned char *src = reinterpret_cast(&tmp); + + dst[0] = src[3]; + dst[1] = src[2]; + dst[2] = src[1]; + dst[3] = src[0]; +#endif +} + +static std::string JoinPath(const std::string &path0, + const std::string &path1) { + if (path0.empty()) { + return path1; + } else { + // check '/' + char lastChar = *path0.rbegin(); + if (lastChar != '/') { + return path0 + std::string("/") + path1; + } else { + return path0 + path1; + } + } +} + +static std::string FindFile(const std::vector &paths, + const std::string &filepath, FsCallbacks *fs) { + if (fs == nullptr || fs->ExpandFilePath == nullptr || + fs->FileExists == nullptr) { + // Error, fs callback[s] missing + return std::string(); + } + + // https://github.com/syoyo/tinygltf/issues/416 + // Use strlen() since std::string's size/length reports the number of elements in the buffer, not the length of string(null-terminated) + // strip null-character in the middle of string. + size_t slength = strlen(filepath.c_str()); + if (slength == 0) { + return std::string(); + } + + std::string cleaned_filepath = std::string(filepath.c_str()); + + for (size_t i = 0; i < paths.size(); i++) { + std::string absPath = + fs->ExpandFilePath(JoinPath(paths[i], cleaned_filepath), fs->user_data); + if (fs->FileExists(absPath, fs->user_data)) { + return absPath; + } + } + + return std::string(); +} + +static std::string GetFilePathExtension(const std::string &FileName) { + if (FileName.find_last_of(".") != std::string::npos) + return FileName.substr(FileName.find_last_of(".") + 1); + return ""; +} + +static std::string GetBaseDir(const std::string &filepath) { + if (filepath.find_last_of("/\\") != std::string::npos) + return filepath.substr(0, filepath.find_last_of("/\\")); + return ""; +} + +static std::string GetBaseFilename(const std::string &filepath) { + auto idx = filepath.find_last_of("/\\"); + if (idx != std::string::npos) return filepath.substr(idx + 1); + return filepath; +} + +std::string base64_encode(unsigned char const *, unsigned int len); +std::string base64_decode(std::string const &s); + +/* + base64.cpp and base64.h + + Copyright (C) 2004-2008 René Nyffenegger + + This source code is provided 'as-is', without any express or implied + warranty. In no event will the author be held liable for any damages + arising from the use of this software. + + Permission is granted to anyone to use this software for any purpose, + including commercial applications, and to alter it and redistribute it + freely, subject to the following restrictions: + + 1. The origin of this source code must not be misrepresented; you must not + claim that you wrote the original source code. If you use this source code + in a product, an acknowledgment in the product documentation would be + appreciated but is not required. + + 2. Altered source versions must be plainly marked as such, and must not be + misrepresented as being the original source code. + + 3. This notice may not be removed or altered from any source distribution. + + René Nyffenegger rene.nyffenegger@adp-gmbh.ch + +*/ + +#ifdef __clang__ +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Wsign-conversion" +#pragma clang diagnostic ignored "-Wconversion" +#endif + +static inline bool is_base64(unsigned char c) { + return (isalnum(c) || (c == '+') || (c == '/')); +} + +std::string base64_encode(unsigned char const *bytes_to_encode, + unsigned int in_len) { + std::string ret; + int i = 0; + int j = 0; + unsigned char char_array_3[3]; + unsigned char char_array_4[4]; + + const char *base64_chars = + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz" + "0123456789+/"; + + while (in_len--) { + char_array_3[i++] = *(bytes_to_encode++); + if (i == 3) { + char_array_4[0] = (char_array_3[0] & 0xfc) >> 2; + char_array_4[1] = + ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4); + char_array_4[2] = + ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6); + char_array_4[3] = char_array_3[2] & 0x3f; + + for (i = 0; (i < 4); i++) ret += base64_chars[char_array_4[i]]; + i = 0; + } + } + + if (i) { + for (j = i; j < 3; j++) char_array_3[j] = '\0'; + + char_array_4[0] = (char_array_3[0] & 0xfc) >> 2; + char_array_4[1] = + ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4); + char_array_4[2] = + ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6); + + for (j = 0; (j < i + 1); j++) ret += base64_chars[char_array_4[j]]; + + while ((i++ < 3)) ret += '='; + } + + return ret; +} + +std::string base64_decode(std::string const &encoded_string) { + int in_len = static_cast(encoded_string.size()); + int i = 0; + int j = 0; + int in_ = 0; + unsigned char char_array_4[4], char_array_3[3]; + std::string ret; + + const std::string base64_chars = + "ABCDEFGHIJKLMNOPQRSTUVWXYZ" + "abcdefghijklmnopqrstuvwxyz" + "0123456789+/"; + + while (in_len-- && (encoded_string[in_] != '=') && + is_base64(encoded_string[in_])) { + char_array_4[i++] = encoded_string[in_]; + in_++; + if (i == 4) { + for (i = 0; i < 4; i++) + char_array_4[i] = + static_cast(base64_chars.find(char_array_4[i])); + + char_array_3[0] = + (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); + char_array_3[1] = + ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); + char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; + + for (i = 0; (i < 3); i++) ret += char_array_3[i]; + i = 0; + } + } + + if (i) { + for (j = i; j < 4; j++) char_array_4[j] = 0; + + for (j = 0; j < 4; j++) + char_array_4[j] = + static_cast(base64_chars.find(char_array_4[j])); + + char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4); + char_array_3[1] = + ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2); + char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3]; + + for (j = 0; (j < i - 1); j++) ret += char_array_3[j]; + } + + return ret; +} +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +// https://github.com/syoyo/tinygltf/issues/228 +// TODO(syoyo): Use uriparser https://uriparser.github.io/ for stricter Uri +// decoding? +// +// Uri Decoding from DLIB +// http://dlib.net/dlib/server/server_http.cpp.html +// --- dlib begin ------------------------------------------------------------ +// Copyright (C) 2003 Davis E. King (davis@dlib.net) +// License: Boost Software License +// Boost Software License - Version 1.0 - August 17th, 2003 + +// Permission is hereby granted, free of charge, to any person or organization +// obtaining a copy of the software and accompanying documentation covered by +// this license (the "Software") to use, reproduce, display, distribute, +// execute, and transmit the Software, and to prepare derivative works of the +// Software, and to permit third-parties to whom the Software is furnished to +// do so, all subject to the following: +// The copyright notices in the Software and this entire statement, including +// the above license grant, this restriction and the following disclaimer, +// must be included in all copies of the Software, in whole or in part, and +// all derivative works of the Software, unless such copies or derivative +// works are solely in the form of machine-executable object code generated by +// a source language processor. +// 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, TITLE AND NON-INFRINGEMENT. IN NO EVENT +// SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE +// FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, +// ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +// DEALINGS IN THE SOFTWARE. +// +namespace dlib { + +inline unsigned char from_hex(unsigned char ch) { + if (ch <= '9' && ch >= '0') + ch -= '0'; + else if (ch <= 'f' && ch >= 'a') + ch -= 'a' - 10; + else if (ch <= 'F' && ch >= 'A') + ch -= 'A' - 10; + else + ch = 0; + return ch; +} + +static const std::string urldecode(const std::string &str) { + using namespace std; + string result; + string::size_type i; + for (i = 0; i < str.size(); ++i) { + if (str[i] == '+') { + result += ' '; + } else if (str[i] == '%' && str.size() > i + 2) { + const unsigned char ch1 = + from_hex(static_cast(str[i + 1])); + const unsigned char ch2 = + from_hex(static_cast(str[i + 2])); + const unsigned char ch = static_cast((ch1 << 4) | ch2); + result += static_cast(ch); + i += 2; + } else { + result += str[i]; + } + } + return result; +} + +} // namespace dlib +// --- dlib end -------------------------------------------------------------- + +bool URIDecode(const std::string &in_uri, std::string *out_uri, + void *user_data) { + (void)user_data; + *out_uri = dlib::urldecode(in_uri); + return true; +} + +static bool LoadExternalFile(std::vector *out, std::string *err, + std::string *warn, const std::string &filename, + const std::string &basedir, bool required, + size_t reqBytes, bool checkSize, size_t maxFileSize, FsCallbacks *fs) { + if (fs == nullptr || fs->FileExists == nullptr || + fs->ExpandFilePath == nullptr || fs->ReadWholeFile == nullptr) { + // This is a developer error, assert() ? + if (err) { + (*err) += "FS callback[s] not set\n"; + } + return false; + } + + std::string *failMsgOut = required ? err : warn; + + out->clear(); + + std::vector paths; + paths.push_back(basedir); + paths.push_back("."); + + std::string filepath = FindFile(paths, filename, fs); + if (filepath.empty() || filename.empty()) { + if (failMsgOut) { + (*failMsgOut) += "File not found : " + filename + "\n"; + } + return false; + } + + // Check file size + if (fs->GetFileSizeInBytes) { + + size_t file_size{0}; + std::string _err; + bool ok = fs->GetFileSizeInBytes(&file_size, &_err, filepath, fs->user_data); + if (!ok) { + if (_err.size()) { + if (failMsgOut) { + (*failMsgOut) += "Getting file size failed : " + filename + ", err = " + _err + "\n"; + } + } + return false; + } + + if (file_size > maxFileSize) { + if (failMsgOut) { + (*failMsgOut) += "File size " + std::to_string(file_size) + " exceeds maximum allowed file size " + std::to_string(maxFileSize) + " : " + filepath + "\n"; + } + return false; + } + } + + std::vector buf; + std::string fileReadErr; + bool fileRead = + fs->ReadWholeFile(&buf, &fileReadErr, filepath, fs->user_data); + if (!fileRead) { + if (failMsgOut) { + (*failMsgOut) += + "File read error : " + filepath + " : " + fileReadErr + "\n"; + } + return false; + } + + size_t sz = buf.size(); + if (sz == 0) { + if (failMsgOut) { + (*failMsgOut) += "File is empty : " + filepath + "\n"; + } + return false; + } + + if (checkSize) { + if (reqBytes == sz) { + out->swap(buf); + return true; + } else { + std::stringstream ss; + ss << "File size mismatch : " << filepath << ", requestedBytes " + << reqBytes << ", but got " << sz << std::endl; + if (failMsgOut) { + (*failMsgOut) += ss.str(); + } + return false; + } + } + + out->swap(buf); + return true; +} + +void TinyGLTF::SetImageLoader(LoadImageDataFunction func, void *user_data) { + LoadImageData = func; + load_image_user_data_ = user_data; + user_image_loader_ = true; +} + +void TinyGLTF::RemoveImageLoader() { + LoadImageData = +#ifndef TINYGLTF_NO_STB_IMAGE + &tinygltf::LoadImageData; +#else + nullptr; +#endif + + load_image_user_data_ = nullptr; + user_image_loader_ = false; +} + +#ifndef TINYGLTF_NO_STB_IMAGE +bool LoadImageData(Image *image, const int image_idx, std::string *err, + std::string *warn, int req_width, int req_height, + const unsigned char *bytes, int size, void *user_data) { + (void)warn; + + LoadImageDataOption option; + if (user_data) { + option = *reinterpret_cast(user_data); + } + + int w = 0, h = 0, comp = 0, req_comp = 0; + + unsigned char *data = nullptr; + + // preserve_channels true: Use channels stored in the image file. + // false: force 32-bit textures for common Vulkan compatibility. It appears + // that some GPU drivers do not support 24-bit images for Vulkan + req_comp = option.preserve_channels ? 0 : 4; + int bits = 8; + int pixel_type = TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE; + + // It is possible that the image we want to load is a 16bit per channel image + // We are going to attempt to load it as 16bit per channel, and if it worked, + // set the image data accordingly. We are casting the returned pointer into + // unsigned char, because we are representing "bytes". But we are updating + // the Image metadata to signal that this image uses 2 bytes (16bits) per + // channel: + if (stbi_is_16_bit_from_memory(bytes, size)) { + data = reinterpret_cast( + stbi_load_16_from_memory(bytes, size, &w, &h, &comp, req_comp)); + if (data) { + bits = 16; + pixel_type = TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT; + } + } + + // at this point, if data is still NULL, it means that the image wasn't + // 16bit per channel, we are going to load it as a normal 8bit per channel + // image as we used to do: + // if image cannot be decoded, ignore parsing and keep it by its path + // don't break in this case + // FIXME we should only enter this function if the image is embedded. If + // image->uri references + // an image file, it should be left as it is. Image loading should not be + // mandatory (to support other formats) + if (!data) data = stbi_load_from_memory(bytes, size, &w, &h, &comp, req_comp); + if (!data) { + // NOTE: you can use `warn` instead of `err` + if (err) { + (*err) += + "Unknown image format. STB cannot decode image data for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + "\".\n"; + } + return false; + } + + if ((w < 1) || (h < 1)) { + stbi_image_free(data); + if (err) { + (*err) += "Invalid image data for image[" + std::to_string(image_idx) + + "] name = \"" + image->name + "\"\n"; + } + return false; + } + + if (req_width > 0) { + if (req_width != w) { + stbi_image_free(data); + if (err) { + (*err) += "Image width mismatch for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + return false; + } + } + + if (req_height > 0) { + if (req_height != h) { + stbi_image_free(data); + if (err) { + (*err) += "Image height mismatch. for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + return false; + } + } + + if (req_comp != 0) { + // loaded data has `req_comp` channels(components) + comp = req_comp; + } + + image->width = w; + image->height = h; + image->component = comp; + image->bits = bits; + image->pixel_type = pixel_type; + image->image.resize(static_cast(w * h * comp) * size_t(bits / 8)); + std::copy(data, data + w * h * comp * (bits / 8), image->image.begin()); + stbi_image_free(data); + + return true; +} +#endif + +void TinyGLTF::SetImageWriter(WriteImageDataFunction func, void *user_data) { + WriteImageData = func; + write_image_user_data_ = user_data; +} + +#ifndef TINYGLTF_NO_STB_IMAGE_WRITE +static void WriteToMemory_stbi(void *context, void *data, int size) { + std::vector *buffer = + reinterpret_cast *>(context); + + unsigned char *pData = reinterpret_cast(data); + + buffer->insert(buffer->end(), pData, pData + size); +} + +bool WriteImageData(const std::string *basepath, const std::string *filename, + const Image *image, bool embedImages, + const URICallbacks *uri_cb, std::string *out_uri, + void *fsPtr) { + const std::string ext = GetFilePathExtension(*filename); + + // Write image to temporary buffer + std::string header; + std::vector data; + + if (ext == "png") { + if ((image->bits != 8) || + (image->pixel_type != TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE)) { + // Unsupported pixel format + return false; + } + + if (!stbi_write_png_to_func(WriteToMemory_stbi, &data, image->width, + image->height, image->component, + &image->image[0], 0)) { + return false; + } + header = "data:image/png;base64,"; + } else if (ext == "jpg") { + if (!stbi_write_jpg_to_func(WriteToMemory_stbi, &data, image->width, + image->height, image->component, + &image->image[0], 100)) { + return false; + } + header = "data:image/jpeg;base64,"; + } else if (ext == "bmp") { + if (!stbi_write_bmp_to_func(WriteToMemory_stbi, &data, image->width, + image->height, image->component, + &image->image[0])) { + return false; + } + header = "data:image/bmp;base64,"; + } else if (!embedImages) { + // Error: can't output requested format to file + return false; + } + + if (embedImages) { + // Embed base64-encoded image into URI + if (data.size()) { + *out_uri = header + + base64_encode(&data[0], static_cast(data.size())); + } else { + // Throw error? + } + } else { + // Write image to disc + FsCallbacks *fs = reinterpret_cast(fsPtr); + if ((fs != nullptr) && (fs->WriteWholeFile != nullptr)) { + const std::string imagefilepath = JoinPath(*basepath, *filename); + std::string writeError; + if (!fs->WriteWholeFile(&writeError, imagefilepath, data, + fs->user_data)) { + // Could not write image file to disc; Throw error ? + return false; + } + } else { + // Throw error? + } + if (uri_cb->encode) { + if (!uri_cb->encode(*filename, "image", out_uri, uri_cb->user_data)) { + return false; + } + } else { + *out_uri = *filename; + } + } + + return true; +} +#endif + +void TinyGLTF::SetURICallbacks(URICallbacks callbacks) { + assert(callbacks.decode); + if (callbacks.decode) { + uri_cb = callbacks; + } +} + +void TinyGLTF::SetFsCallbacks(FsCallbacks callbacks) { fs = callbacks; } + +#ifdef _WIN32 +static inline std::wstring UTF8ToWchar(const std::string &str) { + int wstr_size = + MultiByteToWideChar(CP_UTF8, 0, str.data(), (int)str.size(), nullptr, 0); + std::wstring wstr((size_t)wstr_size, 0); + MultiByteToWideChar(CP_UTF8, 0, str.data(), (int)str.size(), &wstr[0], + (int)wstr.size()); + return wstr; +} + +static inline std::string WcharToUTF8(const std::wstring &wstr) { + int str_size = WideCharToMultiByte(CP_UTF8, 0, wstr.data(), (int)wstr.size(), + nullptr, 0, nullptr, nullptr); + std::string str((size_t)str_size, 0); + WideCharToMultiByte(CP_UTF8, 0, wstr.data(), (int)wstr.size(), &str[0], + (int)str.size(), nullptr, nullptr); + return str; +} +#endif + +#ifndef TINYGLTF_NO_FS +// Default implementations of filesystem functions + +bool FileExists(const std::string &abs_filename, void *) { + bool ret; +#ifdef TINYGLTF_ANDROID_LOAD_FROM_ASSETS + if (asset_manager) { + AAsset *asset = AAssetManager_open(asset_manager, abs_filename.c_str(), + AASSET_MODE_STREAMING); + if (!asset) { + return false; + } + AAsset_close(asset); + ret = true; + } else { + return false; + } +#else +#ifdef _WIN32 +#if defined(_MSC_VER) || defined(__GLIBCXX__) || defined(_LIBCPP_VERSION) + + // First check if a file is a directory. + DWORD result = GetFileAttributesW(UTF8ToWchar(abs_filename).c_str()); + if (result == INVALID_FILE_ATTRIBUTES) { + return false; + } + if (result & FILE_ATTRIBUTE_DIRECTORY) { + return false; + } + + FILE *fp = nullptr; + errno_t err = _wfopen_s(&fp, UTF8ToWchar(abs_filename).c_str(), L"rb"); + if (err != 0) { + return false; + } +#else + // TODO: is_directory check + FILE *fp = nullptr; + errno_t err = fopen_s(&fp, abs_filename.c_str(), "rb"); + if (err != 0) { + return false; + } +#endif + +#else + struct stat sb; + if (stat(abs_filename.c_str(), &sb)) { + return false; + } + if (S_ISDIR(sb.st_mode)) { + return false; + } + + FILE *fp = fopen(abs_filename.c_str(), "rb"); +#endif + if (fp) { + ret = true; + fclose(fp); + } else { + ret = false; + } +#endif + + return ret; +} + +std::string ExpandFilePath(const std::string &filepath, void *) { + // https://github.com/syoyo/tinygltf/issues/368 + // + // No file path expansion in built-in FS function anymore, since glTF URI + // should not contain tilde('~') and environment variables, and for security + // reason(`wordexp`). + // + // Users need to supply `base_dir`(in `LoadASCIIFromString`, + // `LoadBinaryFromMemory`) in expanded absolute path. + + return filepath; + +#if 0 +#ifdef _WIN32 + // Assume input `filepath` is encoded in UTF-8 + std::wstring wfilepath = UTF8ToWchar(filepath); + DWORD wlen = ExpandEnvironmentStringsW(wfilepath.c_str(), nullptr, 0); + wchar_t *wstr = new wchar_t[wlen]; + ExpandEnvironmentStringsW(wfilepath.c_str(), wstr, wlen); + + std::wstring ws(wstr); + delete[] wstr; + return WcharToUTF8(ws); + +#else + +#if defined(TARGET_OS_IPHONE) || defined(TARGET_IPHONE_SIMULATOR) || \ + defined(__ANDROID__) || defined(__EMSCRIPTEN__) || defined(__OpenBSD__) + // no expansion + std::string s = filepath; +#else + std::string s; + wordexp_t p; + + if (filepath.empty()) { + return ""; + } + + // Quote the string to keep any spaces in filepath intact. + std::string quoted_path = "\"" + filepath + "\""; + // char** w; + int ret = wordexp(quoted_path.c_str(), &p, 0); + if (ret) { + // err + s = filepath; + return s; + } + + // Use first element only. + if (p.we_wordv) { + s = std::string(p.we_wordv[0]); + wordfree(&p); + } else { + s = filepath; + } + +#endif + + return s; +#endif +#endif +} + +bool GetFileSizeInBytes(size_t *filesize_out, std::string *err, + const std::string &filepath, void *userdata) { + (void)userdata; + +#ifdef TINYGLTF_ANDROID_LOAD_FROM_ASSETS + if (asset_manager) { + AAsset *asset = AAssetManager_open(asset_manager, filepath.c_str(), + AASSET_MODE_STREAMING); + if (!asset) { + if (err) { + (*err) += "File open error : " + filepath + "\n"; + } + return false; + } + size_t size = AAsset_getLength(asset); + + if (size == 0) { + if (err) { + (*err) += "Invalid file size : " + filepath + + " (does the path point to a directory?)"; + } + return false; + } + + return true; + } else { + if (err) { + (*err) += "No asset manager specified : " + filepath + "\n"; + } + return false; + } +#else +#ifdef _WIN32 +#if defined(__GLIBCXX__) // mingw + int file_descriptor = + _wopen(UTF8ToWchar(filepath).c_str(), _O_RDONLY | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf(file_descriptor, std::ios_base::in); + std::istream f(&wfile_buf); +#elif defined(_MSC_VER) || defined(_LIBCPP_VERSION) + // For libcxx, assume _LIBCPP_HAS_OPEN_WITH_WCHAR is defined to accept + // `wchar_t *` + std::ifstream f(UTF8ToWchar(filepath).c_str(), std::ifstream::binary); +#else + // Unknown compiler/runtime + std::ifstream f(filepath.c_str(), std::ifstream::binary); +#endif +#else + std::ifstream f(filepath.c_str(), std::ifstream::binary); +#endif + if (!f) { + if (err) { + (*err) += "File open error : " + filepath + "\n"; + } + return false; + } + + // For directory(and pipe?), peek() will fail(Posix gnustl/libc++ only) + int buf = f.peek(); + if (!f) { + if (err) { + (*err) += "File read error. Maybe empty file or invalid file : " + filepath + "\n"; + } + return false; + } + + f.seekg(0, f.end); + size_t sz = static_cast(f.tellg()); + + //std::cout << "sz = " << sz << "\n"; + f.seekg(0, f.beg); + + if (int64_t(sz) < 0) { + if (err) { + (*err) += "Invalid file size : " + filepath + + " (does the path point to a directory?)"; + } + return false; + } else if (sz == 0) { + if (err) { + (*err) += "File is empty : " + filepath + "\n"; + } + return false; + } else if (sz >= (std::numeric_limits::max)()) { + if (err) { + (*err) += "Invalid file size : " + filepath + "\n"; + } + return false; + } + + (*filesize_out) = sz; + return true; +#endif +} + +bool ReadWholeFile(std::vector *out, std::string *err, + const std::string &filepath, void *) { +#ifdef TINYGLTF_ANDROID_LOAD_FROM_ASSETS + if (asset_manager) { + AAsset *asset = AAssetManager_open(asset_manager, filepath.c_str(), + AASSET_MODE_STREAMING); + if (!asset) { + if (err) { + (*err) += "File open error : " + filepath + "\n"; + } + return false; + } + size_t size = AAsset_getLength(asset); + if (size == 0) { + if (err) { + (*err) += "Invalid file size : " + filepath + + " (does the path point to a directory?)"; + } + return false; + } + out->resize(size); + AAsset_read(asset, reinterpret_cast(&out->at(0)), size); + AAsset_close(asset); + return true; + } else { + if (err) { + (*err) += "No asset manager specified : " + filepath + "\n"; + } + return false; + } +#else +#ifdef _WIN32 +#if defined(__GLIBCXX__) // mingw + int file_descriptor = + _wopen(UTF8ToWchar(filepath).c_str(), _O_RDONLY | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf(file_descriptor, std::ios_base::in); + std::istream f(&wfile_buf); +#elif defined(_MSC_VER) || defined(_LIBCPP_VERSION) + // For libcxx, assume _LIBCPP_HAS_OPEN_WITH_WCHAR is defined to accept + // `wchar_t *` + std::ifstream f(UTF8ToWchar(filepath).c_str(), std::ifstream::binary); +#else + // Unknown compiler/runtime + std::ifstream f(filepath.c_str(), std::ifstream::binary); +#endif +#else + std::ifstream f(filepath.c_str(), std::ifstream::binary); +#endif + if (!f) { + if (err) { + (*err) += "File open error : " + filepath + "\n"; + } + return false; + } + + // For directory(and pipe?), peek() will fail(Posix gnustl/libc++ only) + int buf = f.peek(); + if (!f) { + if (err) { + (*err) += "File read error. Maybe empty file or invalid file : " + filepath + "\n"; + } + return false; + } + + f.seekg(0, f.end); + size_t sz = static_cast(f.tellg()); + + //std::cout << "sz = " << sz << "\n"; + f.seekg(0, f.beg); + + if (int64_t(sz) < 0) { + if (err) { + (*err) += "Invalid file size : " + filepath + + " (does the path point to a directory?)"; + } + return false; + } else if (sz == 0) { + if (err) { + (*err) += "File is empty : " + filepath + "\n"; + } + return false; + } else if (sz >= (std::numeric_limits::max)()) { + if (err) { + (*err) += "Invalid file size : " + filepath + "\n"; + } + return false; + } + + out->resize(sz); + f.read(reinterpret_cast(&out->at(0)), + static_cast(sz)); + + return true; +#endif +} + +bool WriteWholeFile(std::string *err, const std::string &filepath, + const std::vector &contents, void *) { +#ifdef _WIN32 +#if defined(__GLIBCXX__) // mingw + int file_descriptor = _wopen(UTF8ToWchar(filepath).c_str(), + _O_CREAT | _O_WRONLY | _O_TRUNC | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf( + file_descriptor, std::ios_base::out | std::ios_base::binary); + std::ostream f(&wfile_buf); +#elif defined(_MSC_VER) + std::ofstream f(UTF8ToWchar(filepath).c_str(), std::ofstream::binary); +#else // clang? + std::ofstream f(filepath.c_str(), std::ofstream::binary); +#endif +#else + std::ofstream f(filepath.c_str(), std::ofstream::binary); +#endif + if (!f) { + if (err) { + (*err) += "File open error for writing : " + filepath + "\n"; + } + return false; + } + + f.write(reinterpret_cast(&contents.at(0)), + static_cast(contents.size())); + if (!f) { + if (err) { + (*err) += "File write error: " + filepath + "\n"; + } + return false; + } + + return true; +} + +#endif // TINYGLTF_NO_FS + +static std::string MimeToExt(const std::string &mimeType) { + if (mimeType == "image/jpeg") { + return "jpg"; + } else if (mimeType == "image/png") { + return "png"; + } else if (mimeType == "image/bmp") { + return "bmp"; + } else if (mimeType == "image/gif") { + return "gif"; + } + + return ""; +} + +static bool UpdateImageObject(const Image &image, std::string &baseDir, + int index, bool embedImages, + const URICallbacks *uri_cb, + WriteImageDataFunction *WriteImageData, + void *user_data, std::string *out_uri) { + std::string filename; + std::string ext; + // If image has uri, use it as a filename + if (image.uri.size()) { + std::string decoded_uri; + if (!uri_cb->decode(image.uri, &decoded_uri, uri_cb->user_data)) { + // A decode failure results in a failure to write the gltf. + return false; + } + filename = GetBaseFilename(decoded_uri); + ext = GetFilePathExtension(filename); + } else if (image.bufferView != -1) { + // If there's no URI and the data exists in a buffer, + // don't change properties or write images + } else if (image.name.size()) { + ext = MimeToExt(image.mimeType); + // Otherwise use name as filename + filename = image.name + "." + ext; + } else { + ext = MimeToExt(image.mimeType); + // Fallback to index of image as filename + filename = std::to_string(index) + "." + ext; + } + + // If callback is set and image data exists, modify image data object. If + // image data does not exist, this is not considered a failure and the + // original uri should be maintained. + bool imageWritten = false; + if (*WriteImageData != nullptr && !filename.empty() && !image.image.empty()) { + imageWritten = (*WriteImageData)(&baseDir, &filename, &image, embedImages, + uri_cb, out_uri, user_data); + if (!imageWritten) { + return false; + } + } + + // Use the original uri if the image was not written. + if (!imageWritten) { + *out_uri = image.uri; + } + + return true; +} + +bool IsDataURI(const std::string &in) { + std::string header = "data:application/octet-stream;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:image/jpeg;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:image/png;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:image/bmp;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:image/gif;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:text/plain;base64,"; + if (in.find(header) == 0) { + return true; + } + + header = "data:application/gltf-buffer;base64,"; + if (in.find(header) == 0) { + return true; + } + + return false; +} + +bool DecodeDataURI(std::vector *out, std::string &mime_type, + const std::string &in, size_t reqBytes, bool checkSize) { + std::string header = "data:application/octet-stream;base64,"; + std::string data; + if (in.find(header) == 0) { + data = base64_decode(in.substr(header.size())); // cut mime string. + } + + if (data.empty()) { + header = "data:image/jpeg;base64,"; + if (in.find(header) == 0) { + mime_type = "image/jpeg"; + data = base64_decode(in.substr(header.size())); // cut mime string. + } + } + + if (data.empty()) { + header = "data:image/png;base64,"; + if (in.find(header) == 0) { + mime_type = "image/png"; + data = base64_decode(in.substr(header.size())); // cut mime string. + } + } + + if (data.empty()) { + header = "data:image/bmp;base64,"; + if (in.find(header) == 0) { + mime_type = "image/bmp"; + data = base64_decode(in.substr(header.size())); // cut mime string. + } + } + + if (data.empty()) { + header = "data:image/gif;base64,"; + if (in.find(header) == 0) { + mime_type = "image/gif"; + data = base64_decode(in.substr(header.size())); // cut mime string. + } + } + + if (data.empty()) { + header = "data:text/plain;base64,"; + if (in.find(header) == 0) { + mime_type = "text/plain"; + data = base64_decode(in.substr(header.size())); + } + } + + if (data.empty()) { + header = "data:application/gltf-buffer;base64,"; + if (in.find(header) == 0) { + data = base64_decode(in.substr(header.size())); + } + } + + // TODO(syoyo): Allow empty buffer? #229 + if (data.empty()) { + return false; + } + + if (checkSize) { + if (data.size() != reqBytes) { + return false; + } + out->resize(reqBytes); + } else { + out->resize(data.size()); + } + std::copy(data.begin(), data.end(), out->begin()); + return true; +} + +namespace detail { +bool GetInt(const detail::json &o, int &val) { +#ifdef TINYGLTF_USE_RAPIDJSON + if (!o.IsDouble()) { + if (o.IsInt()) { + val = o.GetInt(); + return true; + } else if (o.IsUint()) { + val = static_cast(o.GetUint()); + return true; + } else if (o.IsInt64()) { + val = static_cast(o.GetInt64()); + return true; + } else if (o.IsUint64()) { + val = static_cast(o.GetUint64()); + return true; + } + } + + return false; +#else + auto type = o.type(); + + if ((type == detail::json::value_t::number_integer) || + (type == detail::json::value_t::number_unsigned)) { + val = static_cast(o.get()); + return true; + } + + return false; +#endif +} + +#ifdef TINYGLTF_USE_RAPIDJSON +bool GetDouble(const detail::json &o, double &val) { + if (o.IsDouble()) { + val = o.GetDouble(); + return true; + } + + return false; +} +#endif + +bool GetNumber(const detail::json &o, double &val) { +#ifdef TINYGLTF_USE_RAPIDJSON + if (o.IsNumber()) { + val = o.GetDouble(); + return true; + } + + return false; +#else + if (o.is_number()) { + val = o.get(); + return true; + } + + return false; +#endif +} + +bool GetString(const detail::json &o, std::string &val) { +#ifdef TINYGLTF_USE_RAPIDJSON + if (o.IsString()) { + val = o.GetString(); + return true; + } + + return false; +#else + if (o.type() == detail::json::value_t::string) { + val = o.get(); + return true; + } + + return false; +#endif +} + +bool IsArray(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.IsArray(); +#else + return o.is_array(); +#endif +} + +detail::json_const_array_iterator ArrayBegin(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.Begin(); +#else + return o.begin(); +#endif +} + +detail::json_const_array_iterator ArrayEnd(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.End(); +#else + return o.end(); +#endif +} + +bool IsObject(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.IsObject(); +#else + return o.is_object(); +#endif +} + +detail::json_const_iterator ObjectBegin(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.MemberBegin(); +#else + return o.begin(); +#endif +} + +detail::json_const_iterator ObjectEnd(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.MemberEnd(); +#else + return o.end(); +#endif +} + +// Making this a const char* results in a pointer to a temporary when +// TINYGLTF_USE_RAPIDJSON is off. +std::string GetKey(detail::json_const_iterator &it) { +#ifdef TINYGLTF_USE_RAPIDJSON + return it->name.GetString(); +#else + return it.key().c_str(); +#endif +} + +bool FindMember(const detail::json &o, const char *member, detail::json_const_iterator &it) { +#ifdef TINYGLTF_USE_RAPIDJSON + if (!o.IsObject()) { + return false; + } + it = o.FindMember(member); + return it != o.MemberEnd(); +#else + it = o.find(member); + return it != o.end(); +#endif +} + +const detail::json &GetValue(detail::json_const_iterator &it) { +#ifdef TINYGLTF_USE_RAPIDJSON + return it->value; +#else + return it.value(); +#endif +} + +std::string JsonToString(const detail::json &o, int spacing = -1) { +#ifdef TINYGLTF_USE_RAPIDJSON + using namespace rapidjson; + StringBuffer buffer; + if (spacing == -1) { + Writer writer(buffer); + // TODO: Better error handling. + // https://github.com/syoyo/tinygltf/issues/332 + if (!o.Accept(writer)) { + return "tiny_gltf::JsonToString() failed rapidjson conversion"; + } + } else { + PrettyWriter writer(buffer); + writer.SetIndent(' ', uint32_t(spacing)); + if (!o.Accept(writer)) { + return "tiny_gltf::JsonToString() failed rapidjson conversion"; + } + } + return buffer.GetString(); +#else + return o.dump(spacing); +#endif +} + +} // namespace + +static bool ParseJsonAsValue(Value *ret, const detail::json &o) { + Value val{}; +#ifdef TINYGLTF_USE_RAPIDJSON + using rapidjson::Type; + switch (o.GetType()) { + case Type::kObjectType: { + Value::Object value_object; + for (auto it = o.MemberBegin(); it != o.MemberEnd(); ++it) { + Value entry; + ParseJsonAsValue(&entry, it->value); + if (entry.Type() != NULL_TYPE) + value_object.emplace(detail::GetKey(it), std::move(entry)); + } + if (value_object.size() > 0) val = Value(std::move(value_object)); + } break; + case Type::kArrayType: { + Value::Array value_array; + value_array.reserve(o.Size()); + for (auto it = o.Begin(); it != o.End(); ++it) { + Value entry; + ParseJsonAsValue(&entry, *it); + if (entry.Type() != NULL_TYPE) + value_array.emplace_back(std::move(entry)); + } + if (value_array.size() > 0) val = Value(std::move(value_array)); + } break; + case Type::kStringType: + val = Value(std::string(o.GetString())); + break; + case Type::kFalseType: + case Type::kTrueType: + val = Value(o.GetBool()); + break; + case Type::kNumberType: + if (!o.IsDouble()) { + int i = 0; + detail::GetInt(o, i); + val = Value(i); + } else { + double d = 0.0; + detail::GetDouble(o, d); + val = Value(d); + } + break; + case Type::kNullType: + break; + // all types are covered, so no `case default` + } +#else + switch (o.type()) { + case detail::json::value_t::object: { + Value::Object value_object; + for (auto it = o.begin(); it != o.end(); it++) { + Value entry; + ParseJsonAsValue(&entry, it.value()); + if (entry.Type() != NULL_TYPE) + value_object.emplace(it.key(), std::move(entry)); + } + if (value_object.size() > 0) val = Value(std::move(value_object)); + } break; + case detail::json::value_t::array: { + Value::Array value_array; + value_array.reserve(o.size()); + for (auto it = o.begin(); it != o.end(); it++) { + Value entry; + ParseJsonAsValue(&entry, it.value()); + if (entry.Type() != NULL_TYPE) + value_array.emplace_back(std::move(entry)); + } + if (value_array.size() > 0) val = Value(std::move(value_array)); + } break; + case detail::json::value_t::string: + val = Value(o.get()); + break; + case detail::json::value_t::boolean: + val = Value(o.get()); + break; + case detail::json::value_t::number_integer: + case detail::json::value_t::number_unsigned: + val = Value(static_cast(o.get())); + break; + case detail::json::value_t::number_float: + val = Value(o.get()); + break; + case detail::json::value_t::null: + case detail::json::value_t::discarded: + case detail::json::value_t::binary: + // default: + break; + } +#endif + const bool isNotNull = val.Type() != NULL_TYPE; + + if (ret) *ret = std::move(val); + + return isNotNull; +} + +static bool ParseExtrasProperty(Value *ret, const detail::json &o) { + detail::json_const_iterator it; + if (!detail::FindMember(o, "extras", it)) { + return false; + } + + return ParseJsonAsValue(ret, detail::GetValue(it)); +} + +static bool ParseBooleanProperty(bool *ret, std::string *err, const detail::json &o, + const std::string &property, + const bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + auto &value = detail::GetValue(it); + + bool isBoolean; + bool boolValue = false; +#ifdef TINYGLTF_USE_RAPIDJSON + isBoolean = value.IsBool(); + if (isBoolean) { + boolValue = value.GetBool(); + } +#else + isBoolean = value.is_boolean(); + if (isBoolean) { + boolValue = value.get(); + } +#endif + if (!isBoolean) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a bool type.\n"; + } + } + return false; + } + + if (ret) { + (*ret) = boolValue; + } + + return true; +} + +static bool ParseIntegerProperty(int *ret, std::string *err, const detail::json &o, + const std::string &property, + const bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + int intValue; + bool isInt = detail::GetInt(detail::GetValue(it), intValue); + if (!isInt) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not an integer type.\n"; + } + } + return false; + } + + if (ret) { + (*ret) = intValue; + } + + return true; +} + +static bool ParseUnsignedProperty(size_t *ret, std::string *err, const detail::json &o, + const std::string &property, + const bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + auto &value = detail::GetValue(it); + + size_t uValue = 0; + bool isUValue; +#ifdef TINYGLTF_USE_RAPIDJSON + isUValue = false; + if (value.IsUint()) { + uValue = value.GetUint(); + isUValue = true; + } else if (value.IsUint64()) { + uValue = value.GetUint64(); + isUValue = true; + } +#else + isUValue = value.is_number_unsigned(); + if (isUValue) { + uValue = value.get(); + } +#endif + if (!isUValue) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a positive integer.\n"; + } + } + return false; + } + + if (ret) { + (*ret) = uValue; + } + + return true; +} + +static bool ParseNumberProperty(double *ret, std::string *err, const detail::json &o, + const std::string &property, + const bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + double numberValue; + bool isNumber = detail::GetNumber(detail::GetValue(it), numberValue); + + if (!isNumber) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a number type.\n"; + } + } + return false; + } + + if (ret) { + (*ret) = numberValue; + } + + return true; +} + +static bool ParseNumberArrayProperty(std::vector *ret, std::string *err, + const detail::json &o, const std::string &property, + bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + if (!detail::IsArray(detail::GetValue(it))) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not an array"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + ret->clear(); + auto end = detail::ArrayEnd(detail::GetValue(it)); + for (auto i = detail::ArrayBegin(detail::GetValue(it)); i != end; ++i) { + double numberValue; + const bool isNumber = detail::GetNumber(*i, numberValue); + if (!isNumber) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a number.\n"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + ret->push_back(numberValue); + } + + return true; +} + +static bool ParseIntegerArrayProperty(std::vector *ret, std::string *err, + const detail::json &o, + const std::string &property, + bool required, + const std::string &parent_node = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + if (!detail::IsArray(detail::GetValue(it))) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not an array"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + + ret->clear(); + auto end = detail::ArrayEnd(detail::GetValue(it)); + for (auto i = detail::ArrayBegin(detail::GetValue(it)); i != end; ++i) { + int numberValue; + bool isNumber = detail::GetInt(*i, numberValue); + if (!isNumber) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not an integer type.\n"; + if (!parent_node.empty()) { + (*err) += " in " + parent_node; + } + (*err) += ".\n"; + } + } + return false; + } + ret->push_back(numberValue); + } + + return true; +} + +static bool ParseStringProperty( + std::string *ret, std::string *err, const detail::json &o, + const std::string &property, bool required, + const std::string &parent_node = std::string()) { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing"; + if (parent_node.empty()) { + (*err) += ".\n"; + } else { + (*err) += " in `" + parent_node + "'.\n"; + } + } + } + return false; + } + + std::string strValue; + if (!detail::GetString(detail::GetValue(it), strValue)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a string type.\n"; + } + } + return false; + } + + if (ret) { + (*ret) = std::move(strValue); + } + + return true; +} + +static bool ParseStringIntegerProperty(std::map *ret, + std::string *err, const detail::json &o, + const std::string &property, + bool required, + const std::string &parent = "") { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + if (!parent.empty()) { + (*err) += + "'" + property + "' property is missing in " + parent + ".\n"; + } else { + (*err) += "'" + property + "' property is missing.\n"; + } + } + } + return false; + } + + const detail::json &dict = detail::GetValue(it); + + // Make sure we are dealing with an object / dictionary. + if (!detail::IsObject(dict)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not an object.\n"; + } + } + return false; + } + + ret->clear(); + + detail::json_const_iterator dictIt(detail::ObjectBegin(dict)); + detail::json_const_iterator dictItEnd(detail::ObjectEnd(dict)); + + for (; dictIt != dictItEnd; ++dictIt) { + int intVal; + if (!detail::GetInt(detail::GetValue(dictIt), intVal)) { + if (required) { + if (err) { + (*err) += "'" + property + "' value is not an integer type.\n"; + } + } + return false; + } + + // Insert into the list. + (*ret)[detail::GetKey(dictIt)] = intVal; + } + return true; +} + +static bool ParseJSONProperty(std::map *ret, + std::string *err, const detail::json &o, + const std::string &property, bool required) { + detail::json_const_iterator it; + if (!detail::FindMember(o, property.c_str(), it)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is missing. \n'"; + } + } + return false; + } + + const detail::json &obj = detail::GetValue(it); + + if (!detail::IsObject(obj)) { + if (required) { + if (err) { + (*err) += "'" + property + "' property is not a JSON object.\n"; + } + } + return false; + } + + ret->clear(); + + detail::json_const_iterator it2(detail::ObjectBegin(obj)); + detail::json_const_iterator itEnd(detail::ObjectEnd(obj)); + for (; it2 != itEnd; ++it2) { + double numVal; + if (detail::GetNumber(detail::GetValue(it2), numVal)) + ret->emplace(std::string(detail::GetKey(it2)), numVal); + } + + return true; +} + +static bool ParseParameterProperty(Parameter *param, std::string *err, + const detail::json &o, const std::string &prop, + bool required) { + // A parameter value can either be a string or an array of either a boolean or + // a number. Booleans of any kind aren't supported here. Granted, it + // complicates the Parameter structure and breaks it semantically in the sense + // that the client probably works off the assumption that if the string is + // empty the vector is used, etc. Would a tagged union work? + if (ParseStringProperty(¶m->string_value, err, o, prop, false)) { + // Found string property. + return true; + } else if (ParseNumberArrayProperty(¶m->number_array, err, o, prop, + false)) { + // Found a number array. + return true; + } else if (ParseNumberProperty(¶m->number_value, err, o, prop, false)) { + param->has_number_value = true; + return true; + } else if (ParseJSONProperty(¶m->json_double_value, err, o, prop, + false)) { + return true; + } else if (ParseBooleanProperty(¶m->bool_value, err, o, prop, false)) { + return true; + } else { + if (required) { + if (err) { + (*err) += "parameter must be a string or number / number array.\n"; + } + } + return false; + } +} + +static bool ParseExtensionsProperty(ExtensionMap *ret, std::string *err, + const detail::json &o) { + (void)err; + + detail::json_const_iterator it; + if (!detail::FindMember(o, "extensions", it)) { + return false; + } + + auto &obj = detail::GetValue(it); + if (!detail::IsObject(obj)) { + return false; + } + ExtensionMap extensions; + detail::json_const_iterator extIt = detail::ObjectBegin(obj); // it.value().begin(); + detail::json_const_iterator extEnd = detail::ObjectEnd(obj); + for (; extIt != extEnd; ++extIt) { + auto &itObj = detail::GetValue(extIt); + if (!detail::IsObject(itObj)) continue; + std::string key(detail::GetKey(extIt)); + if (!ParseJsonAsValue(&extensions[key], itObj)) { + if (!key.empty()) { + // create empty object so that an extension object is still of type + // object + extensions[key] = Value{Value::Object{}}; + } + } + } + if (ret) { + (*ret) = std::move(extensions); + } + return true; +} + +static bool ParseAsset(Asset *asset, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&asset->version, err, o, "version", true, "Asset"); + ParseStringProperty(&asset->generator, err, o, "generator", false, "Asset"); + ParseStringProperty(&asset->minVersion, err, o, "minVersion", false, "Asset"); + ParseStringProperty(&asset->copyright, err, o, "copyright", false, "Asset"); + + ParseExtensionsProperty(&asset->extensions, err, o); + + // Unity exporter version is added as extra here + ParseExtrasProperty(&(asset->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + asset->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + asset->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseImage(Image *image, const int image_idx, std::string *err, + std::string *warn, const detail::json &o, + bool store_original_json_for_extras_and_extensions, + const std::string &basedir, const size_t max_file_size, FsCallbacks *fs, + const URICallbacks *uri_cb, + LoadImageDataFunction *LoadImageData = nullptr, + void *load_image_user_data = nullptr) { + // A glTF image must either reference a bufferView or an image uri + + // schema says oneOf [`bufferView`, `uri`] + // TODO(syoyo): Check the type of each parameters. + detail::json_const_iterator it; + bool hasBufferView = detail::FindMember(o, "bufferView", it); + bool hasURI = detail::FindMember(o, "uri", it); + + ParseStringProperty(&image->name, err, o, "name", false); + + if (hasBufferView && hasURI) { + // Should not both defined. + if (err) { + (*err) += + "Only one of `bufferView` or `uri` should be defined, but both are " + "defined for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + "\"\n"; + } + return false; + } + + if (!hasBufferView && !hasURI) { + if (err) { + (*err) += "Neither required `bufferView` nor `uri` defined for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + return false; + } + + ParseExtensionsProperty(&image->extensions, err, o); + ParseExtrasProperty(&image->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extensions", eit)) { + image->extensions_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extras", eit)) { + image->extras_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + } + + if (hasBufferView) { + int bufferView = -1; + if (!ParseIntegerProperty(&bufferView, err, o, "bufferView", true)) { + if (err) { + (*err) += "Failed to parse `bufferView` for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + return false; + } + + std::string mime_type; + ParseStringProperty(&mime_type, err, o, "mimeType", false); + + int width = 0; + ParseIntegerProperty(&width, err, o, "width", false); + + int height = 0; + ParseIntegerProperty(&height, err, o, "height", false); + + // Just only save some information here. Loading actual image data from + // bufferView is done after this `ParseImage` function. + image->bufferView = bufferView; + image->mimeType = mime_type; + image->width = width; + image->height = height; + + return true; + } + + // Parse URI & Load image data. + + std::string uri; + std::string tmp_err; + if (!ParseStringProperty(&uri, &tmp_err, o, "uri", true)) { + if (err) { + (*err) += "Failed to parse `uri` for image[" + std::to_string(image_idx) + + "] name = \"" + image->name + "\".\n"; + } + return false; + } + + std::vector img; + + if (IsDataURI(uri)) { + if (!DecodeDataURI(&img, image->mimeType, uri, 0, false)) { + if (err) { + (*err) += "Failed to decode 'uri' for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + return false; + } + } else { + // Assume external file + // Keep texture path (for textures that cannot be decoded) + image->uri = uri; +#ifdef TINYGLTF_NO_EXTERNAL_IMAGE + return true; +#else + std::string decoded_uri; + if (!uri_cb->decode(uri, &decoded_uri, uri_cb->user_data)) { + if (warn) { + (*warn) += "Failed to decode 'uri' for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\"\n"; + } + + // Image loading failure is not critical to overall gltf loading. + return true; + } + + if (!LoadExternalFile(&img, err, warn, decoded_uri, basedir, + /* required */ false, /* required bytes */ 0, + /* checksize */ false, /* max file size */ max_file_size, fs)) { + if (warn) { + (*warn) += "Failed to load external 'uri' for image[" + + std::to_string(image_idx) + "] name = \"" + decoded_uri + + "\"\n"; + } + // If the image cannot be loaded, keep uri as image->uri. + return true; + } + + if (img.empty()) { + if (warn) { + (*warn) += "Image data is empty for image[" + + std::to_string(image_idx) + "] name = \"" + image->name + + "\" \n"; + } + return false; + } +#endif + } + + if (*LoadImageData == nullptr) { + if (err) { + (*err) += "No LoadImageData callback specified.\n"; + } + return false; + } + return (*LoadImageData)(image, image_idx, err, warn, 0, 0, &img.at(0), + static_cast(img.size()), load_image_user_data); +} + +static bool ParseTexture(Texture *texture, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions, + const std::string &basedir) { + (void)basedir; + int sampler = -1; + int source = -1; + ParseIntegerProperty(&sampler, err, o, "sampler", false); + + ParseIntegerProperty(&source, err, o, "source", false); + + texture->sampler = sampler; + texture->source = source; + + ParseExtensionsProperty(&texture->extensions, err, o); + ParseExtrasProperty(&texture->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + texture->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + texture->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + ParseStringProperty(&texture->name, err, o, "name", false); + + return true; +} + +static bool ParseTextureInfo( + TextureInfo *texinfo, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (texinfo == nullptr) { + return false; + } + + if (!ParseIntegerProperty(&texinfo->index, err, o, "index", + /* required */ true, "TextureInfo")) { + return false; + } + + ParseIntegerProperty(&texinfo->texCoord, err, o, "texCoord", false); + + ParseExtensionsProperty(&texinfo->extensions, err, o); + ParseExtrasProperty(&texinfo->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + texinfo->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + texinfo->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseNormalTextureInfo( + NormalTextureInfo *texinfo, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (texinfo == nullptr) { + return false; + } + + if (!ParseIntegerProperty(&texinfo->index, err, o, "index", + /* required */ true, "NormalTextureInfo")) { + return false; + } + + ParseIntegerProperty(&texinfo->texCoord, err, o, "texCoord", false); + ParseNumberProperty(&texinfo->scale, err, o, "scale", false); + + ParseExtensionsProperty(&texinfo->extensions, err, o); + ParseExtrasProperty(&texinfo->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + texinfo->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + texinfo->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseOcclusionTextureInfo( + OcclusionTextureInfo *texinfo, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (texinfo == nullptr) { + return false; + } + + if (!ParseIntegerProperty(&texinfo->index, err, o, "index", + /* required */ true, "NormalTextureInfo")) { + return false; + } + + ParseIntegerProperty(&texinfo->texCoord, err, o, "texCoord", false); + ParseNumberProperty(&texinfo->strength, err, o, "strength", false); + + ParseExtensionsProperty(&texinfo->extensions, err, o); + ParseExtrasProperty(&texinfo->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + texinfo->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + texinfo->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseBuffer(Buffer *buffer, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions, + FsCallbacks *fs, const URICallbacks *uri_cb, + const std::string &basedir, const size_t max_buffer_size, bool is_binary = false, + const unsigned char *bin_data = nullptr, + size_t bin_size = 0) { + size_t byteLength; + if (!ParseUnsignedProperty(&byteLength, err, o, "byteLength", true, + "Buffer")) { + return false; + } + + // In glTF 2.0, uri is not mandatory anymore + buffer->uri.clear(); + ParseStringProperty(&buffer->uri, err, o, "uri", false, "Buffer"); + + // having an empty uri for a non embedded image should not be valid + if (!is_binary && buffer->uri.empty()) { + if (err) { + (*err) += "'uri' is missing from non binary glTF file buffer.\n"; + } + } + + detail::json_const_iterator type; + if (detail::FindMember(o, "type", type)) { + std::string typeStr; + if (detail::GetString(detail::GetValue(type), typeStr)) { + if (typeStr.compare("arraybuffer") == 0) { + // buffer.type = "arraybuffer"; + } + } + } + + if (is_binary) { + // Still binary glTF accepts external dataURI. + if (!buffer->uri.empty()) { + // First try embedded data URI. + if (IsDataURI(buffer->uri)) { + std::string mime_type; + if (!DecodeDataURI(&buffer->data, mime_type, buffer->uri, byteLength, + true)) { + if (err) { + (*err) += + "Failed to decode 'uri' : " + buffer->uri + " in Buffer\n"; + } + return false; + } + } else { + // External .bin file. + std::string decoded_uri; + if (!uri_cb->decode(buffer->uri, &decoded_uri, uri_cb->user_data)) { + return false; + } + if (!LoadExternalFile(&buffer->data, err, /* warn */ nullptr, + decoded_uri, basedir, /* required */ true, + byteLength, /* checkSize */ true, /* max_file_size */max_buffer_size, fs)) { + return false; + } + } + } else { + // load data from (embedded) binary data + + if ((bin_size == 0) || (bin_data == nullptr)) { + if (err) { + (*err) += "Invalid binary data in `Buffer', or GLB with empty BIN chunk.\n"; + } + return false; + } + + if (byteLength > bin_size) { + if (err) { + std::stringstream ss; + ss << "Invalid `byteLength'. Must be equal or less than binary size: " + "`byteLength' = " + << byteLength << ", binary size = " << bin_size << std::endl; + (*err) += ss.str(); + } + return false; + } + + // Read buffer data + buffer->data.resize(static_cast(byteLength)); + memcpy(&(buffer->data.at(0)), bin_data, static_cast(byteLength)); + } + + } else { + if (IsDataURI(buffer->uri)) { + std::string mime_type; + if (!DecodeDataURI(&buffer->data, mime_type, buffer->uri, byteLength, + true)) { + if (err) { + (*err) += "Failed to decode 'uri' : " + buffer->uri + " in Buffer\n"; + } + return false; + } + } else { + // Assume external .bin file. + std::string decoded_uri; + if (!uri_cb->decode(buffer->uri, &decoded_uri, uri_cb->user_data)) { + return false; + } + if (!LoadExternalFile(&buffer->data, err, /* warn */ nullptr, decoded_uri, + basedir, /* required */ true, byteLength, + /* checkSize */ true, /* max file size */max_buffer_size, fs)) { + return false; + } + } + } + + ParseStringProperty(&buffer->name, err, o, "name", false); + + ParseExtensionsProperty(&buffer->extensions, err, o); + ParseExtrasProperty(&buffer->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + buffer->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + buffer->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseBufferView( + BufferView *bufferView, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + int buffer = -1; + if (!ParseIntegerProperty(&buffer, err, o, "buffer", true, "BufferView")) { + return false; + } + + size_t byteOffset = 0; + ParseUnsignedProperty(&byteOffset, err, o, "byteOffset", false); + + size_t byteLength = 1; + if (!ParseUnsignedProperty(&byteLength, err, o, "byteLength", true, + "BufferView")) { + return false; + } + + size_t byteStride = 0; + if (!ParseUnsignedProperty(&byteStride, err, o, "byteStride", false)) { + // Spec says: When byteStride of referenced bufferView is not defined, it + // means that accessor elements are tightly packed, i.e., effective stride + // equals the size of the element. + // We cannot determine the actual byteStride until Accessor are parsed, thus + // set 0(= tightly packed) here(as done in OpenGL's VertexAttribPoiner) + byteStride = 0; + } + + if ((byteStride > 252) || ((byteStride % 4) != 0)) { + if (err) { + std::stringstream ss; + ss << "Invalid `byteStride' value. `byteStride' must be the multiple of " + "4 : " + << byteStride << std::endl; + + (*err) += ss.str(); + } + return false; + } + + int target = 0; + ParseIntegerProperty(&target, err, o, "target", false); + if ((target == TINYGLTF_TARGET_ARRAY_BUFFER) || + (target == TINYGLTF_TARGET_ELEMENT_ARRAY_BUFFER)) { + // OK + } else { + target = 0; + } + bufferView->target = target; + + ParseStringProperty(&bufferView->name, err, o, "name", false); + + ParseExtensionsProperty(&bufferView->extensions, err, o); + ParseExtrasProperty(&bufferView->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + bufferView->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + bufferView->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + bufferView->buffer = buffer; + bufferView->byteOffset = byteOffset; + bufferView->byteLength = byteLength; + bufferView->byteStride = byteStride; + return true; +} + +static bool ParseSparseAccessor(Accessor *accessor, std::string *err, + const detail::json &o) { + accessor->sparse.isSparse = true; + + int count = 0; + if (!ParseIntegerProperty(&count, err, o, "count", true, "SparseAccessor")) { + return false; + } + + detail::json_const_iterator indices_iterator; + detail::json_const_iterator values_iterator; + if (!detail::FindMember(o, "indices", indices_iterator)) { + (*err) = "the sparse object of this accessor doesn't have indices"; + return false; + } + + if (!detail::FindMember(o, "values", values_iterator)) { + (*err) = "the sparse object of this accessor doesn't have values"; + return false; + } + + const detail::json &indices_obj = detail::GetValue(indices_iterator); + const detail::json &values_obj = detail::GetValue(values_iterator); + + int indices_buffer_view = 0, indices_byte_offset = 0, component_type = 0; + if (!ParseIntegerProperty(&indices_buffer_view, err, indices_obj, + "bufferView", true, "SparseAccessor")) { + return false; + } + ParseIntegerProperty(&indices_byte_offset, err, indices_obj, "byteOffset", + false); + if (!ParseIntegerProperty(&component_type, err, indices_obj, "componentType", + true, "SparseAccessor")) { + return false; + } + + int values_buffer_view = 0, values_byte_offset = 0; + if (!ParseIntegerProperty(&values_buffer_view, err, values_obj, "bufferView", + true, "SparseAccessor")) { + return false; + } + ParseIntegerProperty(&values_byte_offset, err, values_obj, "byteOffset", + false); + + accessor->sparse.count = count; + accessor->sparse.indices.bufferView = indices_buffer_view; + accessor->sparse.indices.byteOffset = indices_byte_offset; + accessor->sparse.indices.componentType = component_type; + accessor->sparse.values.bufferView = values_buffer_view; + accessor->sparse.values.byteOffset = values_byte_offset; + + return true; +} + +static bool ParseAccessor(Accessor *accessor, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + int bufferView = -1; + ParseIntegerProperty(&bufferView, err, o, "bufferView", false, "Accessor"); + + size_t byteOffset = 0; + ParseUnsignedProperty(&byteOffset, err, o, "byteOffset", false, "Accessor"); + + bool normalized = false; + ParseBooleanProperty(&normalized, err, o, "normalized", false, "Accessor"); + + size_t componentType = 0; + if (!ParseUnsignedProperty(&componentType, err, o, "componentType", true, + "Accessor")) { + return false; + } + + size_t count = 0; + if (!ParseUnsignedProperty(&count, err, o, "count", true, "Accessor")) { + return false; + } + + std::string type; + if (!ParseStringProperty(&type, err, o, "type", true, "Accessor")) { + return false; + } + + if (type.compare("SCALAR") == 0) { + accessor->type = TINYGLTF_TYPE_SCALAR; + } else if (type.compare("VEC2") == 0) { + accessor->type = TINYGLTF_TYPE_VEC2; + } else if (type.compare("VEC3") == 0) { + accessor->type = TINYGLTF_TYPE_VEC3; + } else if (type.compare("VEC4") == 0) { + accessor->type = TINYGLTF_TYPE_VEC4; + } else if (type.compare("MAT2") == 0) { + accessor->type = TINYGLTF_TYPE_MAT2; + } else if (type.compare("MAT3") == 0) { + accessor->type = TINYGLTF_TYPE_MAT3; + } else if (type.compare("MAT4") == 0) { + accessor->type = TINYGLTF_TYPE_MAT4; + } else { + std::stringstream ss; + ss << "Unsupported `type` for accessor object. Got \"" << type << "\"\n"; + if (err) { + (*err) += ss.str(); + } + return false; + } + + ParseStringProperty(&accessor->name, err, o, "name", false); + + accessor->minValues.clear(); + accessor->maxValues.clear(); + ParseNumberArrayProperty(&accessor->minValues, err, o, "min", false, + "Accessor"); + + ParseNumberArrayProperty(&accessor->maxValues, err, o, "max", false, + "Accessor"); + + accessor->count = count; + accessor->bufferView = bufferView; + accessor->byteOffset = byteOffset; + accessor->normalized = normalized; + { + if (componentType >= TINYGLTF_COMPONENT_TYPE_BYTE && + componentType <= TINYGLTF_COMPONENT_TYPE_DOUBLE) { + // OK + accessor->componentType = int(componentType); + } else { + std::stringstream ss; + ss << "Invalid `componentType` in accessor. Got " << componentType + << "\n"; + if (err) { + (*err) += ss.str(); + } + return false; + } + } + + ParseExtensionsProperty(&(accessor->extensions), err, o); + ParseExtrasProperty(&(accessor->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + accessor->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + accessor->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + // check if accessor has a "sparse" object: + detail::json_const_iterator iterator; + if (detail::FindMember(o, "sparse", iterator)) { + // here this accessor has a "sparse" subobject + return ParseSparseAccessor(accessor, err, detail::GetValue(iterator)); + } + + return true; +} + +#ifdef TINYGLTF_ENABLE_DRACO + +static void DecodeIndexBuffer(draco::Mesh *mesh, size_t componentSize, + std::vector &outBuffer) { + if (componentSize == 4) { + assert(sizeof(mesh->face(draco::FaceIndex(0))[0]) == componentSize); + memcpy(outBuffer.data(), &mesh->face(draco::FaceIndex(0))[0], + outBuffer.size()); + } else { + size_t faceStride = componentSize * 3; + for (draco::FaceIndex f(0); f < mesh->num_faces(); ++f) { + const draco::Mesh::Face &face = mesh->face(f); + if (componentSize == 2) { + uint16_t indices[3] = {(uint16_t)face[0].value(), + (uint16_t)face[1].value(), + (uint16_t)face[2].value()}; + memcpy(outBuffer.data() + f.value() * faceStride, &indices[0], + faceStride); + } else { + uint8_t indices[3] = {(uint8_t)face[0].value(), + (uint8_t)face[1].value(), + (uint8_t)face[2].value()}; + memcpy(outBuffer.data() + f.value() * faceStride, &indices[0], + faceStride); + } + } + } +} + +template +static bool GetAttributeForAllPoints(draco::Mesh *mesh, + const draco::PointAttribute *pAttribute, + std::vector &outBuffer) { + size_t byteOffset = 0; + T values[4] = {0, 0, 0, 0}; + for (draco::PointIndex i(0); i < mesh->num_points(); ++i) { + const draco::AttributeValueIndex val_index = pAttribute->mapped_index(i); + if (!pAttribute->ConvertValue(val_index, pAttribute->num_components(), + values)) + return false; + + memcpy(outBuffer.data() + byteOffset, &values[0], + sizeof(T) * pAttribute->num_components()); + byteOffset += sizeof(T) * pAttribute->num_components(); + } + + return true; +} + +static bool GetAttributeForAllPoints(uint32_t componentType, draco::Mesh *mesh, + const draco::PointAttribute *pAttribute, + std::vector &outBuffer) { + bool decodeResult = false; + switch (componentType) { + case TINYGLTF_COMPONENT_TYPE_UNSIGNED_BYTE: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_BYTE: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_UNSIGNED_SHORT: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_SHORT: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_INT: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_UNSIGNED_INT: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_FLOAT: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + case TINYGLTF_COMPONENT_TYPE_DOUBLE: + decodeResult = + GetAttributeForAllPoints(mesh, pAttribute, outBuffer); + break; + default: + return false; + } + + return decodeResult; +} + +static bool ParseDracoExtension(Primitive *primitive, Model *model, + std::string *err, + const Value &dracoExtensionValue) { + (void)err; + auto bufferViewValue = dracoExtensionValue.Get("bufferView"); + if (!bufferViewValue.IsInt()) return false; + auto attributesValue = dracoExtensionValue.Get("attributes"); + if (!attributesValue.IsObject()) return false; + + auto attributesObject = attributesValue.Get(); + int bufferView = bufferViewValue.Get(); + + BufferView &view = model->bufferViews[bufferView]; + Buffer &buffer = model->buffers[view.buffer]; + // BufferView has already been decoded + if (view.dracoDecoded) return true; + view.dracoDecoded = true; + + const char *bufferViewData = + reinterpret_cast(buffer.data.data() + view.byteOffset); + size_t bufferViewSize = view.byteLength; + + // decode draco + draco::DecoderBuffer decoderBuffer; + decoderBuffer.Init(bufferViewData, bufferViewSize); + draco::Decoder decoder; + auto decodeResult = decoder.DecodeMeshFromBuffer(&decoderBuffer); + if (!decodeResult.ok()) { + return false; + } + const std::unique_ptr &mesh = decodeResult.value(); + + // create new bufferView for indices + if (primitive->indices >= 0) { + int32_t componentSize = GetComponentSizeInBytes( + model->accessors[primitive->indices].componentType); + Buffer decodedIndexBuffer; + decodedIndexBuffer.data.resize(mesh->num_faces() * 3 * componentSize); + + DecodeIndexBuffer(mesh.get(), componentSize, decodedIndexBuffer.data); + + model->buffers.emplace_back(std::move(decodedIndexBuffer)); + + BufferView decodedIndexBufferView; + decodedIndexBufferView.buffer = int(model->buffers.size() - 1); + decodedIndexBufferView.byteLength = + int(mesh->num_faces() * 3 * componentSize); + decodedIndexBufferView.byteOffset = 0; + decodedIndexBufferView.byteStride = 0; + decodedIndexBufferView.target = TINYGLTF_TARGET_ARRAY_BUFFER; + model->bufferViews.emplace_back(std::move(decodedIndexBufferView)); + + model->accessors[primitive->indices].bufferView = + int(model->bufferViews.size() - 1); + model->accessors[primitive->indices].count = int(mesh->num_faces() * 3); + } + + for (const auto &attribute : attributesObject) { + if (!attribute.second.IsInt()) return false; + auto primitiveAttribute = primitive->attributes.find(attribute.first); + if (primitiveAttribute == primitive->attributes.end()) return false; + + int dracoAttributeIndex = attribute.second.Get(); + const auto pAttribute = mesh->GetAttributeByUniqueId(dracoAttributeIndex); + const auto componentType = + model->accessors[primitiveAttribute->second].componentType; + + // Create a new buffer for this decoded buffer + Buffer decodedBuffer; + size_t bufferSize = mesh->num_points() * pAttribute->num_components() * + GetComponentSizeInBytes(componentType); + decodedBuffer.data.resize(bufferSize); + + if (!GetAttributeForAllPoints(componentType, mesh.get(), pAttribute, + decodedBuffer.data)) + return false; + + model->buffers.emplace_back(std::move(decodedBuffer)); + + BufferView decodedBufferView; + decodedBufferView.buffer = int(model->buffers.size() - 1); + decodedBufferView.byteLength = bufferSize; + decodedBufferView.byteOffset = pAttribute->byte_offset(); + decodedBufferView.byteStride = pAttribute->byte_stride(); + decodedBufferView.target = primitive->indices >= 0 + ? TINYGLTF_TARGET_ELEMENT_ARRAY_BUFFER + : TINYGLTF_TARGET_ARRAY_BUFFER; + model->bufferViews.emplace_back(std::move(decodedBufferView)); + + model->accessors[primitiveAttribute->second].bufferView = + int(model->bufferViews.size() - 1); + model->accessors[primitiveAttribute->second].count = + int(mesh->num_points()); + } + + return true; +} +#endif + +static bool ParsePrimitive(Primitive *primitive, Model *model, std::string *err, + const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + int material = -1; + ParseIntegerProperty(&material, err, o, "material", false); + primitive->material = material; + + int mode = TINYGLTF_MODE_TRIANGLES; + ParseIntegerProperty(&mode, err, o, "mode", false); + primitive->mode = mode; // Why only triangles were supported ? + + int indices = -1; + ParseIntegerProperty(&indices, err, o, "indices", false); + primitive->indices = indices; + if (!ParseStringIntegerProperty(&primitive->attributes, err, o, "attributes", + true, "Primitive")) { + return false; + } + + // Look for morph targets + detail::json_const_iterator targetsObject; + if (detail::FindMember(o, "targets", targetsObject) && + detail::IsArray(detail::GetValue(targetsObject))) { + auto targetsObjectEnd = detail::ArrayEnd(detail::GetValue(targetsObject)); + for (detail::json_const_array_iterator i = detail::ArrayBegin(detail::GetValue(targetsObject)); + i != targetsObjectEnd; ++i) { + std::map targetAttribues; + + const detail::json &dict = *i; + if (detail::IsObject(dict)) { + detail::json_const_iterator dictIt(detail::ObjectBegin(dict)); + detail::json_const_iterator dictItEnd(detail::ObjectEnd(dict)); + + for (; dictIt != dictItEnd; ++dictIt) { + int iVal; + if (detail::GetInt(detail::GetValue(dictIt), iVal)) + targetAttribues[detail::GetKey(dictIt)] = iVal; + } + primitive->targets.emplace_back(std::move(targetAttribues)); + } + } + } + + ParseExtrasProperty(&(primitive->extras), o); + ParseExtensionsProperty(&primitive->extensions, err, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + primitive->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + primitive->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + +#ifdef TINYGLTF_ENABLE_DRACO + auto dracoExtension = + primitive->extensions.find("KHR_draco_mesh_compression"); + if (dracoExtension != primitive->extensions.end()) { + ParseDracoExtension(primitive, model, err, dracoExtension->second); + } +#else + (void)model; +#endif + + return true; +} + +static bool ParseMesh(Mesh *mesh, Model *model, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&mesh->name, err, o, "name", false); + + mesh->primitives.clear(); + detail::json_const_iterator primObject; + if (detail::FindMember(o, "primitives", primObject) && + detail::IsArray(detail::GetValue(primObject))) { + detail::json_const_array_iterator primEnd = detail::ArrayEnd(detail::GetValue(primObject)); + for (detail::json_const_array_iterator i = detail::ArrayBegin(detail::GetValue(primObject)); + i != primEnd; ++i) { + Primitive primitive; + if (ParsePrimitive(&primitive, model, err, *i, + store_original_json_for_extras_and_extensions)) { + // Only add the primitive if the parsing succeeds. + mesh->primitives.emplace_back(std::move(primitive)); + } + } + } + + // Should probably check if has targets and if dimensions fit + ParseNumberArrayProperty(&mesh->weights, err, o, "weights", false); + + ParseExtensionsProperty(&mesh->extensions, err, o); + ParseExtrasProperty(&(mesh->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + mesh->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + mesh->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseNode(Node *node, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&node->name, err, o, "name", false); + + int skin = -1; + ParseIntegerProperty(&skin, err, o, "skin", false); + node->skin = skin; + + // Matrix and T/R/S are exclusive + if (!ParseNumberArrayProperty(&node->matrix, err, o, "matrix", false)) { + ParseNumberArrayProperty(&node->rotation, err, o, "rotation", false); + ParseNumberArrayProperty(&node->scale, err, o, "scale", false); + ParseNumberArrayProperty(&node->translation, err, o, "translation", false); + } + + int camera = -1; + ParseIntegerProperty(&camera, err, o, "camera", false); + node->camera = camera; + + int mesh = -1; + ParseIntegerProperty(&mesh, err, o, "mesh", false); + node->mesh = mesh; + + node->children.clear(); + ParseIntegerArrayProperty(&node->children, err, o, "children", false); + + ParseNumberArrayProperty(&node->weights, err, o, "weights", false); + + ParseExtensionsProperty(&node->extensions, err, o); + ParseExtrasProperty(&(node->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + node->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + node->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParsePbrMetallicRoughness( + PbrMetallicRoughness *pbr, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (pbr == nullptr) { + return false; + } + + std::vector baseColorFactor; + if (ParseNumberArrayProperty(&baseColorFactor, err, o, "baseColorFactor", + /* required */ false)) { + if (baseColorFactor.size() != 4) { + if (err) { + (*err) += + "Array length of `baseColorFactor` parameter in " + "pbrMetallicRoughness must be 4, but got " + + std::to_string(baseColorFactor.size()) + "\n"; + } + return false; + } + pbr->baseColorFactor = baseColorFactor; + } + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "baseColorTexture", it)) { + ParseTextureInfo(&pbr->baseColorTexture, err, detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "metallicRoughnessTexture", it)) { + ParseTextureInfo(&pbr->metallicRoughnessTexture, err, detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + ParseNumberProperty(&pbr->metallicFactor, err, o, "metallicFactor", false); + ParseNumberProperty(&pbr->roughnessFactor, err, o, "roughnessFactor", false); + + ParseExtensionsProperty(&pbr->extensions, err, o); + ParseExtrasProperty(&pbr->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + pbr->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + pbr->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseMaterial(Material *material, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&material->name, err, o, "name", /* required */ false); + + if (ParseNumberArrayProperty(&material->emissiveFactor, err, o, + "emissiveFactor", + /* required */ false)) { + if (material->emissiveFactor.size() != 3) { + if (err) { + (*err) += + "Array length of `emissiveFactor` parameter in " + "material must be 3, but got " + + std::to_string(material->emissiveFactor.size()) + "\n"; + } + return false; + } + } else { + // fill with default values + material->emissiveFactor = {0.0, 0.0, 0.0}; + } + + ParseStringProperty(&material->alphaMode, err, o, "alphaMode", + /* required */ false); + ParseNumberProperty(&material->alphaCutoff, err, o, "alphaCutoff", + /* required */ false); + ParseBooleanProperty(&material->doubleSided, err, o, "doubleSided", + /* required */ false); + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "pbrMetallicRoughness", it)) { + ParsePbrMetallicRoughness(&material->pbrMetallicRoughness, err, + detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "normalTexture", it)) { + ParseNormalTextureInfo(&material->normalTexture, err, detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "occlusionTexture", it)) { + ParseOcclusionTextureInfo(&material->occlusionTexture, err, detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "emissiveTexture", it)) { + ParseTextureInfo(&material->emissiveTexture, err, detail::GetValue(it), + store_original_json_for_extras_and_extensions); + } + } + + // Old code path. For backward compatibility, we still store material values + // as Parameter. This will create duplicated information for + // example(pbrMetallicRoughness), but should be negligible in terms of memory + // consumption. + // TODO(syoyo): Remove in the next major release. + material->values.clear(); + material->additionalValues.clear(); + + detail::json_const_iterator it(detail::ObjectBegin(o)); + detail::json_const_iterator itEnd(detail::ObjectEnd(o)); + + for (; it != itEnd; ++it) { + std::string key(detail::GetKey(it)); + if (key == "pbrMetallicRoughness") { + if (detail::IsObject(detail::GetValue(it))) { + const detail::json &values_object = detail::GetValue(it); + + detail::json_const_iterator itVal(detail::ObjectBegin(values_object)); + detail::json_const_iterator itValEnd(detail::ObjectEnd(values_object)); + + for (; itVal != itValEnd; ++itVal) { + Parameter param; + if (ParseParameterProperty(¶m, err, values_object, detail::GetKey(itVal), + false)) { + material->values.emplace(detail::GetKey(itVal), std::move(param)); + } + } + } + } else if (key == "extensions" || key == "extras") { + // done later, skip, otherwise poorly parsed contents will be saved in the + // parametermap and serialized again later + } else { + Parameter param; + if (ParseParameterProperty(¶m, err, o, key, false)) { + // names of materials have already been parsed. Putting it in this map + // doesn't correctly reflect the glTF specification + if (key != "name") + material->additionalValues.emplace(std::move(key), std::move(param)); + } + } + } + + material->extensions.clear(); + ParseExtensionsProperty(&material->extensions, err, o); + ParseExtrasProperty(&(material->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extensions", eit)) { + material->extensions_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extras", eit)) { + material->extras_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + } + + return true; +} + +static bool ParseAnimationChannel( + AnimationChannel *channel, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + int samplerIndex = -1; + int targetIndex = -1; + if (!ParseIntegerProperty(&samplerIndex, err, o, "sampler", true, + "AnimationChannel")) { + if (err) { + (*err) += "`sampler` field is missing in animation channels\n"; + } + return false; + } + + detail::json_const_iterator targetIt; + if (detail::FindMember(o, "target", targetIt) && detail::IsObject(detail::GetValue(targetIt))) { + const detail::json &target_object = detail::GetValue(targetIt); + + ParseIntegerProperty(&targetIndex, err, target_object, "node", false); + + if (!ParseStringProperty(&channel->target_path, err, target_object, "path", + true)) { + if (err) { + (*err) += "`path` field is missing in animation.channels.target\n"; + } + return false; + } + ParseExtensionsProperty(&channel->target_extensions, err, target_object); + if (store_original_json_for_extras_and_extensions) { + detail::json_const_iterator it; + if (detail::FindMember(target_object, "extensions", it)) { + channel->target_extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + channel->sampler = samplerIndex; + channel->target_node = targetIndex; + + ParseExtensionsProperty(&channel->extensions, err, o); + ParseExtrasProperty(&(channel->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + channel->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + channel->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseAnimation(Animation *animation, std::string *err, + const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator channelsIt; + if (detail::FindMember(o, "channels", channelsIt) && + detail::IsArray(detail::GetValue(channelsIt))) { + detail::json_const_array_iterator channelEnd = detail::ArrayEnd(detail::GetValue(channelsIt)); + for (detail::json_const_array_iterator i = detail::ArrayBegin(detail::GetValue(channelsIt)); + i != channelEnd; ++i) { + AnimationChannel channel; + if (ParseAnimationChannel( + &channel, err, *i, + store_original_json_for_extras_and_extensions)) { + // Only add the channel if the parsing succeeds. + animation->channels.emplace_back(std::move(channel)); + } + } + } + } + + { + detail::json_const_iterator samplerIt; + if (detail::FindMember(o, "samplers", samplerIt) && detail::IsArray(detail::GetValue(samplerIt))) { + const detail::json &sampler_array = detail::GetValue(samplerIt); + + detail::json_const_array_iterator it = detail::ArrayBegin(sampler_array); + detail::json_const_array_iterator itEnd = detail::ArrayEnd(sampler_array); + + for (; it != itEnd; ++it) { + const detail::json &s = *it; + + AnimationSampler sampler; + int inputIndex = -1; + int outputIndex = -1; + if (!ParseIntegerProperty(&inputIndex, err, s, "input", true)) { + if (err) { + (*err) += "`input` field is missing in animation.sampler\n"; + } + return false; + } + ParseStringProperty(&sampler.interpolation, err, s, "interpolation", + false); + if (!ParseIntegerProperty(&outputIndex, err, s, "output", true)) { + if (err) { + (*err) += "`output` field is missing in animation.sampler\n"; + } + return false; + } + sampler.input = inputIndex; + sampler.output = outputIndex; + ParseExtensionsProperty(&(sampler.extensions), err, o); + ParseExtrasProperty(&(sampler.extras), s); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extensions", eit)) { + sampler.extensions_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + { + detail::json_const_iterator eit; + if (detail::FindMember(o, "extras", eit)) { + sampler.extras_json_string = detail::JsonToString(detail::GetValue(eit)); + } + } + } + + animation->samplers.emplace_back(std::move(sampler)); + } + } + } + + ParseStringProperty(&animation->name, err, o, "name", false); + + ParseExtensionsProperty(&animation->extensions, err, o); + ParseExtrasProperty(&(animation->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + animation->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + animation->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseSampler(Sampler *sampler, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&sampler->name, err, o, "name", false); + + int minFilter = -1; + int magFilter = -1; + int wrapS = TINYGLTF_TEXTURE_WRAP_REPEAT; + int wrapT = TINYGLTF_TEXTURE_WRAP_REPEAT; + // int wrapR = TINYGLTF_TEXTURE_WRAP_REPEAT; + ParseIntegerProperty(&minFilter, err, o, "minFilter", false); + ParseIntegerProperty(&magFilter, err, o, "magFilter", false); + ParseIntegerProperty(&wrapS, err, o, "wrapS", false); + ParseIntegerProperty(&wrapT, err, o, "wrapT", false); + // ParseIntegerProperty(&wrapR, err, o, "wrapR", false); // tinygltf + // extension + + // TODO(syoyo): Check the value is allowed one. + // (e.g. we allow 9728(NEAREST), but don't allow 9727) + + sampler->minFilter = minFilter; + sampler->magFilter = magFilter; + sampler->wrapS = wrapS; + sampler->wrapT = wrapT; + // sampler->wrapR = wrapR; + + ParseExtensionsProperty(&(sampler->extensions), err, o); + ParseExtrasProperty(&(sampler->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + sampler->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + sampler->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseSkin(Skin *skin, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseStringProperty(&skin->name, err, o, "name", false, "Skin"); + + std::vector joints; + if (!ParseIntegerArrayProperty(&joints, err, o, "joints", false, "Skin")) { + return false; + } + skin->joints = std::move(joints); + + int skeleton = -1; + ParseIntegerProperty(&skeleton, err, o, "skeleton", false, "Skin"); + skin->skeleton = skeleton; + + int invBind = -1; + ParseIntegerProperty(&invBind, err, o, "inverseBindMatrices", true, "Skin"); + skin->inverseBindMatrices = invBind; + + ParseExtensionsProperty(&(skin->extensions), err, o); + ParseExtrasProperty(&(skin->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + skin->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + skin->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParsePerspectiveCamera( + PerspectiveCamera *camera, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + double yfov = 0.0; + if (!ParseNumberProperty(&yfov, err, o, "yfov", true, "OrthographicCamera")) { + return false; + } + + double znear = 0.0; + if (!ParseNumberProperty(&znear, err, o, "znear", true, + "PerspectiveCamera")) { + return false; + } + + double aspectRatio = 0.0; // = invalid + ParseNumberProperty(&aspectRatio, err, o, "aspectRatio", false, + "PerspectiveCamera"); + + double zfar = 0.0; // = invalid + ParseNumberProperty(&zfar, err, o, "zfar", false, "PerspectiveCamera"); + + camera->aspectRatio = aspectRatio; + camera->zfar = zfar; + camera->yfov = yfov; + camera->znear = znear; + + ParseExtensionsProperty(&camera->extensions, err, o); + ParseExtrasProperty(&(camera->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + camera->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + camera->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + // TODO(syoyo): Validate parameter values. + + return true; +} + +static bool ParseSpotLight(SpotLight *light, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + ParseNumberProperty(&light->innerConeAngle, err, o, "innerConeAngle", false); + ParseNumberProperty(&light->outerConeAngle, err, o, "outerConeAngle", false); + + ParseExtensionsProperty(&light->extensions, err, o); + ParseExtrasProperty(&light->extras, o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + light->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + light->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + // TODO(syoyo): Validate parameter values. + + return true; +} + +static bool ParseOrthographicCamera( + OrthographicCamera *camera, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + double xmag = 0.0; + if (!ParseNumberProperty(&xmag, err, o, "xmag", true, "OrthographicCamera")) { + return false; + } + + double ymag = 0.0; + if (!ParseNumberProperty(&ymag, err, o, "ymag", true, "OrthographicCamera")) { + return false; + } + + double zfar = 0.0; + if (!ParseNumberProperty(&zfar, err, o, "zfar", true, "OrthographicCamera")) { + return false; + } + + double znear = 0.0; + if (!ParseNumberProperty(&znear, err, o, "znear", true, + "OrthographicCamera")) { + return false; + } + + ParseExtensionsProperty(&camera->extensions, err, o); + ParseExtrasProperty(&(camera->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + camera->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + camera->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + camera->xmag = xmag; + camera->ymag = ymag; + camera->zfar = zfar; + camera->znear = znear; + + // TODO(syoyo): Validate parameter values. + + return true; +} + +static bool ParseCamera(Camera *camera, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (!ParseStringProperty(&camera->type, err, o, "type", true, "Camera")) { + return false; + } + + if (camera->type.compare("orthographic") == 0) { + detail::json_const_iterator orthoIt; + if (!detail::FindMember(o, "orthographic", orthoIt)) { + if (err) { + std::stringstream ss; + ss << "Orthographic camera description not found." << std::endl; + (*err) += ss.str(); + } + return false; + } + + const detail::json &v = detail::GetValue(orthoIt); + if (!detail::IsObject(v)) { + if (err) { + std::stringstream ss; + ss << "\"orthographic\" is not a JSON object." << std::endl; + (*err) += ss.str(); + } + return false; + } + + if (!ParseOrthographicCamera( + &camera->orthographic, err, v, + store_original_json_for_extras_and_extensions)) { + return false; + } + } else if (camera->type.compare("perspective") == 0) { + detail::json_const_iterator perspIt; + if (!detail::FindMember(o, "perspective", perspIt)) { + if (err) { + std::stringstream ss; + ss << "Perspective camera description not found." << std::endl; + (*err) += ss.str(); + } + return false; + } + + const detail::json &v = detail::GetValue(perspIt); + if (!detail::IsObject(v)) { + if (err) { + std::stringstream ss; + ss << "\"perspective\" is not a JSON object." << std::endl; + (*err) += ss.str(); + } + return false; + } + + if (!ParsePerspectiveCamera( + &camera->perspective, err, v, + store_original_json_for_extras_and_extensions)) { + return false; + } + } else { + if (err) { + std::stringstream ss; + ss << "Invalid camera type: \"" << camera->type + << "\". Must be \"perspective\" or \"orthographic\"" << std::endl; + (*err) += ss.str(); + } + return false; + } + + ParseStringProperty(&camera->name, err, o, "name", false); + + ParseExtensionsProperty(&camera->extensions, err, o); + ParseExtrasProperty(&(camera->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + camera->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + camera->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +static bool ParseLight(Light *light, std::string *err, const detail::json &o, + bool store_original_json_for_extras_and_extensions) { + if (!ParseStringProperty(&light->type, err, o, "type", true)) { + return false; + } + + if (light->type == "spot") { + detail::json_const_iterator spotIt; + if (!detail::FindMember(o, "spot", spotIt)) { + if (err) { + std::stringstream ss; + ss << "Spot light description not found." << std::endl; + (*err) += ss.str(); + } + return false; + } + + const detail::json &v = detail::GetValue(spotIt); + if (!detail::IsObject(v)) { + if (err) { + std::stringstream ss; + ss << "\"spot\" is not a JSON object." << std::endl; + (*err) += ss.str(); + } + return false; + } + + if (!ParseSpotLight(&light->spot, err, v, + store_original_json_for_extras_and_extensions)) { + return false; + } + } + + ParseStringProperty(&light->name, err, o, "name", false); + ParseNumberArrayProperty(&light->color, err, o, "color", false); + ParseNumberProperty(&light->range, err, o, "range", false); + ParseNumberProperty(&light->intensity, err, o, "intensity", false); + ParseExtensionsProperty(&light->extensions, err, o); + ParseExtrasProperty(&(light->extras), o); + + if (store_original_json_for_extras_and_extensions) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + light->extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + light->extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + return true; +} + +bool TinyGLTF::LoadFromString(Model *model, std::string *err, std::string *warn, + const char *json_str, + unsigned int json_str_length, + const std::string &base_dir, + unsigned int check_sections) { + if (json_str_length < 4) { + if (err) { + (*err) = "JSON string too short.\n"; + } + return false; + } + + detail::JsonDocument v; + +#if (defined(__cpp_exceptions) || defined(__EXCEPTIONS) || \ + defined(_CPPUNWIND)) && \ + !defined(TINYGLTF_NOEXCEPTION) + try { + detail::JsonParse(v, json_str, json_str_length, true); + + } catch (const std::exception &e) { + if (err) { + (*err) = e.what(); + } + return false; + } +#else + { + detail::JsonParse(v, json_str, json_str_length); + + if (!detail::IsObject(v)) { + // Assume parsing was failed. + if (err) { + (*err) = "Failed to parse JSON object\n"; + } + return false; + } + } +#endif + + if (!detail::IsObject(v)) { + // root is not an object. + if (err) { + (*err) = "Root element is not a JSON object\n"; + } + return false; + } + + { + bool version_found = false; + detail::json_const_iterator it; + if (detail::FindMember(v, "asset", it) && detail::IsObject(detail::GetValue(it))) { + auto &itObj = detail::GetValue(it); + detail::json_const_iterator version_it; + std::string versionStr; + if (detail::FindMember(itObj, "version", version_it) && + detail::GetString(detail::GetValue(version_it), versionStr)) { + version_found = true; + } + } + if (version_found) { + // OK + } else if (check_sections & REQUIRE_VERSION) { + if (err) { + (*err) += "\"asset\" object not found in .gltf or not an object type\n"; + } + return false; + } + } + + // scene is not mandatory. + // FIXME Maybe a better way to handle it than removing the code + + auto IsArrayMemberPresent = [](const detail::json &_v, const char *name) -> bool { + detail::json_const_iterator it; + return detail::FindMember(_v, name, it) && detail::IsArray(detail::GetValue(it)); + }; + + { + if ((check_sections & REQUIRE_SCENES) && + !IsArrayMemberPresent(v, "scenes")) { + if (err) { + (*err) += "\"scenes\" object not found in .gltf or not an array type\n"; + } + return false; + } + } + + { + if ((check_sections & REQUIRE_NODES) && !IsArrayMemberPresent(v, "nodes")) { + if (err) { + (*err) += "\"nodes\" object not found in .gltf\n"; + } + return false; + } + } + + { + if ((check_sections & REQUIRE_ACCESSORS) && + !IsArrayMemberPresent(v, "accessors")) { + if (err) { + (*err) += "\"accessors\" object not found in .gltf\n"; + } + return false; + } + } + + { + if ((check_sections & REQUIRE_BUFFERS) && + !IsArrayMemberPresent(v, "buffers")) { + if (err) { + (*err) += "\"buffers\" object not found in .gltf\n"; + } + return false; + } + } + + { + if ((check_sections & REQUIRE_BUFFER_VIEWS) && + !IsArrayMemberPresent(v, "bufferViews")) { + if (err) { + (*err) += "\"bufferViews\" object not found in .gltf\n"; + } + return false; + } + } + + model->buffers.clear(); + model->bufferViews.clear(); + model->accessors.clear(); + model->meshes.clear(); + model->cameras.clear(); + model->nodes.clear(); + model->extensionsUsed.clear(); + model->extensionsRequired.clear(); + model->extensions.clear(); + model->defaultScene = -1; + + // 1. Parse Asset + { + detail::json_const_iterator it; + if (detail::FindMember(v, "asset", it) && detail::IsObject(detail::GetValue(it))) { + const detail::json &root = detail::GetValue(it); + + ParseAsset(&model->asset, err, root, + store_original_json_for_extras_and_extensions_); + } + } + +#ifdef TINYGLTF_USE_CPP14 + auto ForEachInArray = [](const detail::json &_v, const char *member, + const auto &cb) -> bool +#else + // The std::function<> implementation can be less efficient because it will + // allocate heap when the size of the captured lambda is above 16 bytes with + // clang and gcc, but it does not require C++14. + auto ForEachInArray = [](const detail::json &_v, const char *member, + const std::function &cb) -> bool +#endif + { + detail::json_const_iterator itm; + if (detail::FindMember(_v, member, itm) && detail::IsArray(detail::GetValue(itm))) { + const detail::json &root = detail::GetValue(itm); + auto it = detail::ArrayBegin(root); + auto end = detail::ArrayEnd(root); + for (; it != end; ++it) { + if (!cb(*it)) return false; + } + } + return true; + }; + + // 2. Parse extensionUsed + { + ForEachInArray(v, "extensionsUsed", [&](const detail::json &o) { + std::string str; + detail::GetString(o, str); + model->extensionsUsed.emplace_back(std::move(str)); + return true; + }); + } + + { + ForEachInArray(v, "extensionsRequired", [&](const detail::json &o) { + std::string str; + detail::GetString(o, str); + model->extensionsRequired.emplace_back(std::move(str)); + return true; + }); + } + + // 3. Parse Buffer + { + bool success = ForEachInArray(v, "buffers", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`buffers' does not contain an JSON object."; + } + return false; + } + Buffer buffer; + if (!ParseBuffer(&buffer, err, o, + store_original_json_for_extras_and_extensions_, &fs, + &uri_cb, base_dir, max_external_file_size_, is_binary_, bin_data_, bin_size_)) { + return false; + } + + model->buffers.emplace_back(std::move(buffer)); + return true; + }); + + if (!success) { + return false; + } + } + // 4. Parse BufferView + { + bool success = ForEachInArray(v, "bufferViews", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`bufferViews' does not contain an JSON object."; + } + return false; + } + BufferView bufferView; + if (!ParseBufferView(&bufferView, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->bufferViews.emplace_back(std::move(bufferView)); + return true; + }); + + if (!success) { + return false; + } + } + + // 5. Parse Accessor + { + bool success = ForEachInArray(v, "accessors", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`accessors' does not contain an JSON object."; + } + return false; + } + Accessor accessor; + if (!ParseAccessor(&accessor, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->accessors.emplace_back(std::move(accessor)); + return true; + }); + + if (!success) { + return false; + } + } + + // 6. Parse Mesh + { + bool success = ForEachInArray(v, "meshes", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`meshes' does not contain an JSON object."; + } + return false; + } + Mesh mesh; + if (!ParseMesh(&mesh, model, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->meshes.emplace_back(std::move(mesh)); + return true; + }); + + if (!success) { + return false; + } + } + + // Assign missing bufferView target types + // - Look for missing Mesh indices + // - Look for missing Mesh attributes + for (auto &mesh : model->meshes) { + for (auto &primitive : mesh.primitives) { + if (primitive.indices > + -1) // has indices from parsing step, must be Element Array Buffer + { + if (size_t(primitive.indices) >= model->accessors.size()) { + if (err) { + (*err) += "primitive indices accessor out of bounds"; + } + return false; + } + + auto bufferView = + model->accessors[size_t(primitive.indices)].bufferView; + if (bufferView < 0 || size_t(bufferView) >= model->bufferViews.size()) { + if (err) { + (*err) += "accessor[" + std::to_string(primitive.indices) + + "] invalid bufferView"; + } + return false; + } + + model->bufferViews[size_t(bufferView)].target = + TINYGLTF_TARGET_ELEMENT_ARRAY_BUFFER; + // we could optionally check if accessors' bufferView type is Scalar, as + // it should be + } + + for (auto &attribute : primitive.attributes) { + const auto accessorsIndex = size_t(attribute.second); + if (accessorsIndex < model->accessors.size()) { + const auto bufferView = model->accessors[accessorsIndex].bufferView; + // bufferView could be null(-1) for sparse morph target + if (bufferView >= 0 && bufferView < (int)model->bufferViews.size()) { + model->bufferViews[size_t(bufferView)].target = + TINYGLTF_TARGET_ARRAY_BUFFER; + } + } + } + + for (auto &target : primitive.targets) { + for (auto &attribute : target) { + const auto accessorsIndex = size_t(attribute.second); + if (accessorsIndex < model->accessors.size()) { + const auto bufferView = model->accessors[accessorsIndex].bufferView; + // bufferView could be null(-1) for sparse morph target + if (bufferView >= 0 && bufferView < (int)model->bufferViews.size()) { + model->bufferViews[size_t(bufferView)].target = + TINYGLTF_TARGET_ARRAY_BUFFER; + } + } + } + } + } + } + + // 7. Parse Node + { + bool success = ForEachInArray(v, "nodes", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`nodes' does not contain an JSON object."; + } + return false; + } + Node node; + if (!ParseNode(&node, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->nodes.emplace_back(std::move(node)); + return true; + }); + + if (!success) { + return false; + } + } + + // 8. Parse scenes. + { + bool success = ForEachInArray(v, "scenes", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`scenes' does not contain an JSON object."; + } + return false; + } + std::vector nodes; + ParseIntegerArrayProperty(&nodes, err, o, "nodes", false); + + Scene scene; + scene.nodes = std::move(nodes); + + ParseStringProperty(&scene.name, err, o, "name", false); + + ParseExtensionsProperty(&scene.extensions, err, o); + ParseExtrasProperty(&scene.extras, o); + + if (store_original_json_for_extras_and_extensions_) { + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + scene.extensions_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extras", it)) { + scene.extras_json_string = detail::JsonToString(detail::GetValue(it)); + } + } + } + + model->scenes.emplace_back(std::move(scene)); + return true; + }); + + if (!success) { + return false; + } + } + + // 9. Parse default scenes. + { + detail::json_const_iterator rootIt; + int iVal; + if (detail::FindMember(v, "scene", rootIt) && detail::GetInt(detail::GetValue(rootIt), iVal)) { + model->defaultScene = iVal; + } + } + + // 10. Parse Material + { + bool success = ForEachInArray(v, "materials", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`materials' does not contain an JSON object."; + } + return false; + } + Material material; + ParseStringProperty(&material.name, err, o, "name", false); + + if (!ParseMaterial(&material, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->materials.emplace_back(std::move(material)); + return true; + }); + + if (!success) { + return false; + } + } + + // 11. Parse Image + void *load_image_user_data{nullptr}; + + LoadImageDataOption load_image_option; + + if (user_image_loader_) { + // Use user supplied pointer + load_image_user_data = load_image_user_data_; + } else { + load_image_option.preserve_channels = preserve_image_channels_; + load_image_user_data = reinterpret_cast(&load_image_option); + } + + { + int idx = 0; + bool success = ForEachInArray(v, "images", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "image[" + std::to_string(idx) + "] is not a JSON object."; + } + return false; + } + Image image; + if (!ParseImage(&image, idx, err, warn, o, + store_original_json_for_extras_and_extensions_, base_dir, + max_external_file_size_, &fs, &uri_cb, &this->LoadImageData, + load_image_user_data)) { + return false; + } + + if (image.bufferView != -1) { + // Load image from the buffer view. + if (size_t(image.bufferView) >= model->bufferViews.size()) { + if (err) { + std::stringstream ss; + ss << "image[" << idx << "] bufferView \"" << image.bufferView + << "\" not found in the scene." << std::endl; + (*err) += ss.str(); + } + return false; + } + + const BufferView &bufferView = + model->bufferViews[size_t(image.bufferView)]; + if (size_t(bufferView.buffer) >= model->buffers.size()) { + if (err) { + std::stringstream ss; + ss << "image[" << idx << "] buffer \"" << bufferView.buffer + << "\" not found in the scene." << std::endl; + (*err) += ss.str(); + } + return false; + } + const Buffer &buffer = model->buffers[size_t(bufferView.buffer)]; + + if (*LoadImageData == nullptr) { + if (err) { + (*err) += "No LoadImageData callback specified.\n"; + } + return false; + } + bool ret = LoadImageData( + &image, idx, err, warn, image.width, image.height, + &buffer.data[bufferView.byteOffset], + static_cast(bufferView.byteLength), load_image_user_data); + if (!ret) { + return false; + } + } + + model->images.emplace_back(std::move(image)); + ++idx; + return true; + }); + + if (!success) { + return false; + } + } + + // 12. Parse Texture + { + bool success = ForEachInArray(v, "textures", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`textures' does not contain an JSON object."; + } + return false; + } + Texture texture; + if (!ParseTexture(&texture, err, o, + store_original_json_for_extras_and_extensions_, + base_dir)) { + return false; + } + + model->textures.emplace_back(std::move(texture)); + return true; + }); + + if (!success) { + return false; + } + } + + // 13. Parse Animation + { + bool success = ForEachInArray(v, "animations", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`animations' does not contain an JSON object."; + } + return false; + } + Animation animation; + if (!ParseAnimation(&animation, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->animations.emplace_back(std::move(animation)); + return true; + }); + + if (!success) { + return false; + } + } + + // 14. Parse Skin + { + bool success = ForEachInArray(v, "skins", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`skins' does not contain an JSON object."; + } + return false; + } + Skin skin; + if (!ParseSkin(&skin, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->skins.emplace_back(std::move(skin)); + return true; + }); + + if (!success) { + return false; + } + } + + // 15. Parse Sampler + { + bool success = ForEachInArray(v, "samplers", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`samplers' does not contain an JSON object."; + } + return false; + } + Sampler sampler; + if (!ParseSampler(&sampler, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->samplers.emplace_back(std::move(sampler)); + return true; + }); + + if (!success) { + return false; + } + } + + // 16. Parse Camera + { + bool success = ForEachInArray(v, "cameras", [&](const detail::json &o) { + if (!detail::IsObject(o)) { + if (err) { + (*err) += "`cameras' does not contain an JSON object."; + } + return false; + } + Camera camera; + if (!ParseCamera(&camera, err, o, + store_original_json_for_extras_and_extensions_)) { + return false; + } + + model->cameras.emplace_back(std::move(camera)); + return true; + }); + + if (!success) { + return false; + } + } + + // 17. Parse Extensions + ParseExtensionsProperty(&model->extensions, err, v); + + // 18. Specific extension implementations + { + detail::json_const_iterator rootIt; + if (detail::FindMember(v, "extensions", rootIt) && detail::IsObject(detail::GetValue(rootIt))) { + const detail::json &root = detail::GetValue(rootIt); + + detail::json_const_iterator it(detail::ObjectBegin(root)); + detail::json_const_iterator itEnd(detail::ObjectEnd(root)); + for (; it != itEnd; ++it) { + // parse KHR_lights_punctual extension + std::string key(detail::GetKey(it)); + if ((key == "KHR_lights_punctual") && detail::IsObject(detail::GetValue(it))) { + const detail::json &object = detail::GetValue(it); + detail::json_const_iterator itLight; + if (detail::FindMember(object, "lights", itLight)) { + const detail::json &lights = detail::GetValue(itLight); + if (!detail::IsArray(lights)) { + continue; + } + + auto arrayIt(detail::ArrayBegin(lights)); + auto arrayItEnd(detail::ArrayEnd(lights)); + for (; arrayIt != arrayItEnd; ++arrayIt) { + Light light; + if (!ParseLight(&light, err, *arrayIt, + store_original_json_for_extras_and_extensions_)) { + return false; + } + model->lights.emplace_back(std::move(light)); + } + } + } + } + } + } + + // 19. Parse Extras + ParseExtrasProperty(&model->extras, v); + + if (store_original_json_for_extras_and_extensions_) { + model->extras_json_string = detail::JsonToString(v["extras"]); + model->extensions_json_string = detail::JsonToString(v["extensions"]); + } + + return true; +} + +bool TinyGLTF::LoadASCIIFromString(Model *model, std::string *err, + std::string *warn, const char *str, + unsigned int length, + const std::string &base_dir, + unsigned int check_sections) { + is_binary_ = false; + bin_data_ = nullptr; + bin_size_ = 0; + + return LoadFromString(model, err, warn, str, length, base_dir, + check_sections); +} + +bool TinyGLTF::LoadASCIIFromFile(Model *model, std::string *err, + std::string *warn, const std::string &filename, + unsigned int check_sections) { + std::stringstream ss; + + if (fs.ReadWholeFile == nullptr) { + // Programmer error, assert() ? + ss << "Failed to read file: " << filename + << ": one or more FS callback not set" << std::endl; + if (err) { + (*err) = ss.str(); + } + return false; + } + + std::vector data; + std::string fileerr; + bool fileread = fs.ReadWholeFile(&data, &fileerr, filename, fs.user_data); + if (!fileread) { + ss << "Failed to read file: " << filename << ": " << fileerr << std::endl; + if (err) { + (*err) = ss.str(); + } + return false; + } + + size_t sz = data.size(); + if (sz == 0) { + if (err) { + (*err) = "Empty file."; + } + return false; + } + + std::string basedir = GetBaseDir(filename); + + bool ret = LoadASCIIFromString( + model, err, warn, reinterpret_cast(&data.at(0)), + static_cast(data.size()), basedir, check_sections); + + return ret; +} + +bool TinyGLTF::LoadBinaryFromMemory(Model *model, std::string *err, + std::string *warn, + const unsigned char *bytes, + unsigned int size, + const std::string &base_dir, + unsigned int check_sections) { + if (size < 20) { + if (err) { + (*err) = "Too short data size for glTF Binary."; + } + return false; + } + + if (bytes[0] == 'g' && bytes[1] == 'l' && bytes[2] == 'T' && + bytes[3] == 'F') { + // ok + } else { + if (err) { + (*err) = "Invalid magic."; + } + return false; + } + + unsigned int version; // 4 bytes + unsigned int length; // 4 bytes + unsigned int chunk0_length; // 4 bytes + unsigned int chunk0_format; // 4 bytes; + + memcpy(&version, bytes + 4, 4); + swap4(&version); + memcpy(&length, bytes + 8, 4); + swap4(&length); + memcpy(&chunk0_length, bytes + 12, 4); // JSON data length + swap4(&chunk0_length); + memcpy(&chunk0_format, bytes + 16, 4); + swap4(&chunk0_format); + + // https://registry.khronos.org/glTF/specs/2.0/glTF-2.0.html#binary-gltf-layout + // + // In case the Bin buffer is not present, the size is exactly 20 + size of + // JSON contents, + // so use "greater than" operator. + // + // https://github.com/syoyo/tinygltf/issues/372 + // Use 64bit uint to avoid integer overflow. + uint64_t header_and_json_size = 20ull + uint64_t(chunk0_length); + + if (header_and_json_size > std::numeric_limits::max()) { + // Do not allow 4GB or more GLB data. + (*err) = "Invalid glTF binary. GLB data exceeds 4GB."; + } + + if ((header_and_json_size > uint64_t(size)) || (chunk0_length < 1) || (length > size) || + (header_and_json_size > uint64_t(length)) || + (chunk0_format != 0x4E4F534A)) { // 0x4E4F534A = JSON format. + if (err) { + (*err) = "Invalid glTF binary."; + } + return false; + } + + // Padding check + // The start and the end of each chunk must be aligned to a 4-byte boundary. + // No padding check for chunk0 start since its 4byte-boundary is ensured. + if ((header_and_json_size % 4) != 0) { + if (err) { + (*err) = "JSON Chunk end does not aligned to a 4-byte boundary."; + } + } + + //std::cout << "header_and_json_size = " << header_and_json_size << "\n"; + //std::cout << "length = " << length << "\n"; + + // Chunk1(BIN) data + // The spec says: When the binary buffer is empty or when it is stored by other means, this chunk SHOULD be omitted. + // So when header + JSON data == binary size, Chunk1 is omitted. + if (header_and_json_size == uint64_t(length)) { + + bin_data_ = nullptr; + bin_size_ = 0; + } else { + // Read Chunk1 info(BIN data) + // At least Chunk1 should have 12 bytes(8 bytes(header) + 4 bytes(bin payload could be 1~3 bytes, but need to be aligned to 4 bytes) + if ((header_and_json_size + 12ull) > uint64_t(length)) { + if (err) { + (*err) = "Insufficient storage space for Chunk1(BIN data). At least Chunk1 Must have 4 bytes or more bytes, but got " + std::to_string((header_and_json_size + 12ull) - uint64_t(length)) + ".\n"; + } + return false; + } + + unsigned int chunk1_length; // 4 bytes + unsigned int chunk1_format; // 4 bytes; + memcpy(&chunk1_length, bytes + header_and_json_size, 4); // JSON data length + swap4(&chunk1_length); + memcpy(&chunk1_format, bytes + header_and_json_size + 4, 4); + swap4(&chunk1_format); + + //std::cout << "chunk1_length = " << chunk1_length << "\n"; + + if (chunk1_length < 4) { + if (err) { + (*err) = "Insufficient Chunk1(BIN) data size."; + } + return false; + } + + if ((chunk1_length % 4) != 0) { + if (err) { + (*err) = "BIN Chunk end does not aligned to a 4-byte boundary."; + } + return false; + } + + if (uint64_t(chunk1_length) + header_and_json_size > uint64_t(length)) { + if (err) { + (*err) = "BIN Chunk data length exceeds the GLB size."; + } + return false; + } + + if (chunk1_format != 0x004e4942) { + if (err) { + (*err) = "Invalid type for chunk1 data."; + } + return false; + } + + //std::cout << "chunk1_length = " << chunk1_length << "\n"; + + bin_data_ = bytes + header_and_json_size + + 8; // 4 bytes (bin_buffer_length) + 4 bytes(bin_buffer_format) + + bin_size_ = size_t(chunk1_length); + } + + // Extract JSON string. + std::string jsonString(reinterpret_cast(&bytes[20]), + chunk0_length); + + is_binary_ = true; + + bool ret = LoadFromString(model, err, warn, + reinterpret_cast(&bytes[20]), + chunk0_length, base_dir, check_sections); + if (!ret) { + return ret; + } + + return true; +} + +bool TinyGLTF::LoadBinaryFromFile(Model *model, std::string *err, + std::string *warn, + const std::string &filename, + unsigned int check_sections) { + std::stringstream ss; + + if (fs.ReadWholeFile == nullptr) { + // Programmer error, assert() ? + ss << "Failed to read file: " << filename + << ": one or more FS callback not set" << std::endl; + if (err) { + (*err) = ss.str(); + } + return false; + } + + std::vector data; + std::string fileerr; + bool fileread = fs.ReadWholeFile(&data, &fileerr, filename, fs.user_data); + if (!fileread) { + ss << "Failed to read file: " << filename << ": " << fileerr << std::endl; + if (err) { + (*err) = ss.str(); + } + return false; + } + + std::string basedir = GetBaseDir(filename); + + bool ret = LoadBinaryFromMemory(model, err, warn, &data.at(0), + static_cast(data.size()), + basedir, check_sections); + + return ret; +} + +/////////////////////// +// GLTF Serialization +/////////////////////// +namespace detail { +detail::json JsonFromString(const char *s) { +#ifdef TINYGLTF_USE_RAPIDJSON + return detail::json(s, detail::GetAllocator()); +#else + return detail::json(s); +#endif +} + +void JsonAssign(detail::json &dest, const detail::json &src) { +#ifdef TINYGLTF_USE_RAPIDJSON + dest.CopyFrom(src, detail::GetAllocator()); +#else + dest = src; +#endif +} + +void JsonAddMember(detail::json &o, const char *key, detail::json &&value) { +#ifdef TINYGLTF_USE_RAPIDJSON + if (!o.IsObject()) { + o.SetObject(); + } + o.AddMember(detail::json(key, detail::GetAllocator()), std::move(value), detail::GetAllocator()); +#else + o[key] = std::move(value); +#endif +} + +void JsonPushBack(detail::json &o, detail::json &&value) { +#ifdef TINYGLTF_USE_RAPIDJSON + o.PushBack(std::move(value), detail::GetAllocator()); +#else + o.push_back(std::move(value)); +#endif +} + +bool JsonIsNull(const detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + return o.IsNull(); +#else + return o.is_null(); +#endif +} + +void JsonSetObject(detail::json &o) { +#ifdef TINYGLTF_USE_RAPIDJSON + o.SetObject(); +#else + o = o.object({}); +#endif +} + +void JsonReserveArray(detail::json &o, size_t s) { +#ifdef TINYGLTF_USE_RAPIDJSON + o.SetArray(); + o.Reserve(static_cast(s), detail::GetAllocator()); +#endif + (void)(o); + (void)(s); +} +} // namespace + +// typedef std::pair json_object_pair; + +template +static void SerializeNumberProperty(const std::string &key, T number, + detail::json &obj) { + // obj.insert( + // json_object_pair(key, detail::json(static_cast(number)))); + // obj[key] = static_cast(number); + detail::JsonAddMember(obj, key.c_str(), detail::json(number)); +} + +#ifdef TINYGLTF_USE_RAPIDJSON +template <> +void SerializeNumberProperty(const std::string &key, size_t number, detail::json &obj) { + detail::JsonAddMember(obj, key.c_str(), detail::json(static_cast(number))); +} +#endif + +template +static void SerializeNumberArrayProperty(const std::string &key, + const std::vector &value, + detail::json &obj) { + if (value.empty()) return; + + detail::json ary; + detail::JsonReserveArray(ary, value.size()); + for (const auto &s : value) { + detail::JsonPushBack(ary, detail::json(s)); + } + detail::JsonAddMember(obj, key.c_str(), std::move(ary)); +} + +static void SerializeStringProperty(const std::string &key, + const std::string &value, detail::json &obj) { + detail::JsonAddMember(obj, key.c_str(), detail::JsonFromString(value.c_str())); +} + +static void SerializeStringArrayProperty(const std::string &key, + const std::vector &value, + detail::json &obj) { + detail::json ary; + detail::JsonReserveArray(ary, value.size()); + for (auto &s : value) { + detail::JsonPushBack(ary, detail::JsonFromString(s.c_str())); + } + detail::JsonAddMember(obj, key.c_str(), std::move(ary)); +} + +static bool ValueToJson(const Value &value, detail::json *ret) { + detail::json obj; +#ifdef TINYGLTF_USE_RAPIDJSON + switch (value.Type()) { + case REAL_TYPE: + obj.SetDouble(value.Get()); + break; + case INT_TYPE: + obj.SetInt(value.Get()); + break; + case BOOL_TYPE: + obj.SetBool(value.Get()); + break; + case STRING_TYPE: + obj.SetString(value.Get().c_str(), detail::GetAllocator()); + break; + case ARRAY_TYPE: { + obj.SetArray(); + obj.Reserve(static_cast(value.ArrayLen()), + detail::GetAllocator()); + for (unsigned int i = 0; i < value.ArrayLen(); ++i) { + Value elementValue = value.Get(int(i)); + detail::json elementJson; + if (ValueToJson(value.Get(int(i)), &elementJson)) + obj.PushBack(std::move(elementJson), detail::GetAllocator()); + } + break; + } + case BINARY_TYPE: + // TODO + // obj = detail::json(value.Get>()); + return false; + break; + case OBJECT_TYPE: { + obj.SetObject(); + Value::Object objMap = value.Get(); + for (auto &it : objMap) { + detail::json elementJson; + if (ValueToJson(it.second, &elementJson)) { + obj.AddMember(detail::json(it.first.c_str(), detail::GetAllocator()), + std::move(elementJson), detail::GetAllocator()); + } + } + break; + } + case NULL_TYPE: + default: + return false; + } +#else + switch (value.Type()) { + case REAL_TYPE: + obj = detail::json(value.Get()); + break; + case INT_TYPE: + obj = detail::json(value.Get()); + break; + case BOOL_TYPE: + obj = detail::json(value.Get()); + break; + case STRING_TYPE: + obj = detail::json(value.Get()); + break; + case ARRAY_TYPE: { + for (unsigned int i = 0; i < value.ArrayLen(); ++i) { + Value elementValue = value.Get(int(i)); + detail::json elementJson; + if (ValueToJson(value.Get(int(i)), &elementJson)) + obj.push_back(elementJson); + } + break; + } + case BINARY_TYPE: + // TODO + // obj = json(value.Get>()); + return false; + break; + case OBJECT_TYPE: { + Value::Object objMap = value.Get(); + for (auto &it : objMap) { + detail::json elementJson; + if (ValueToJson(it.second, &elementJson)) obj[it.first] = elementJson; + } + break; + } + case NULL_TYPE: + default: + return false; + } +#endif + if (ret) *ret = std::move(obj); + return true; +} + +static void SerializeValue(const std::string &key, const Value &value, + detail::json &obj) { + detail::json ret; + if (ValueToJson(value, &ret)) { + detail::JsonAddMember(obj, key.c_str(), std::move(ret)); + } +} + +static void SerializeGltfBufferData(const std::vector &data, + detail::json &o) { + std::string header = "data:application/octet-stream;base64,"; + if (data.size() > 0) { + std::string encodedData = + base64_encode(&data[0], static_cast(data.size())); + SerializeStringProperty("uri", header + encodedData, o); + } else { + // Issue #229 + // size 0 is allowed. Just emit mime header. + SerializeStringProperty("uri", header, o); + } +} + +static bool SerializeGltfBufferData(const std::vector &data, + const std::string &binFilename) { +#ifdef _WIN32 +#if defined(__GLIBCXX__) // mingw + int file_descriptor = _wopen(UTF8ToWchar(binFilename).c_str(), + _O_CREAT | _O_WRONLY | _O_TRUNC | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf( + file_descriptor, std::ios_base::out | std::ios_base::binary); + std::ostream output(&wfile_buf); + if (!wfile_buf.is_open()) return false; +#elif defined(_MSC_VER) + std::ofstream output(UTF8ToWchar(binFilename).c_str(), std::ofstream::binary); + if (!output.is_open()) return false; +#else + std::ofstream output(binFilename.c_str(), std::ofstream::binary); + if (!output.is_open()) return false; +#endif +#else + std::ofstream output(binFilename.c_str(), std::ofstream::binary); + if (!output.is_open()) return false; +#endif + if (data.size() > 0) { + output.write(reinterpret_cast(&data[0]), + std::streamsize(data.size())); + } else { + // Issue #229 + // size 0 will be still valid buffer data. + // write empty file. + } + return true; +} + +#if 0 // FIXME(syoyo): not used. will be removed in the future release. +static void SerializeParameterMap(ParameterMap ¶m, detail::json &o) { + for (ParameterMap::iterator paramIt = param.begin(); paramIt != param.end(); + ++paramIt) { + if (paramIt->second.number_array.size()) { + SerializeNumberArrayProperty(paramIt->first, + paramIt->second.number_array, o); + } else if (paramIt->second.json_double_value.size()) { + detail::json json_double_value; + for (std::map::iterator it = + paramIt->second.json_double_value.begin(); + it != paramIt->second.json_double_value.end(); ++it) { + if (it->first == "index") { + json_double_value[it->first] = paramIt->second.TextureIndex(); + } else { + json_double_value[it->first] = it->second; + } + } + + o[paramIt->first] = json_double_value; + } else if (!paramIt->second.string_value.empty()) { + SerializeStringProperty(paramIt->first, paramIt->second.string_value, o); + } else if (paramIt->second.has_number_value) { + o[paramIt->first] = paramIt->second.number_value; + } else { + o[paramIt->first] = paramIt->second.bool_value; + } + } +} +#endif + +static void SerializeExtensionMap(const ExtensionMap &extensions, detail::json &o) { + if (!extensions.size()) return; + + detail::json extMap; + for (ExtensionMap::const_iterator extIt = extensions.begin(); + extIt != extensions.end(); ++extIt) { + // Allow an empty object for extension(#97) + detail::json ret; + bool isNull = true; + if (ValueToJson(extIt->second, &ret)) { + isNull = detail::JsonIsNull(ret); + detail::JsonAddMember(extMap, extIt->first.c_str(), std::move(ret)); + } + if (isNull) { + if (!(extIt->first.empty())) { // name should not be empty, but for sure + // create empty object so that an extension name is still included in + // json. + detail::json empty; + detail::JsonSetObject(empty); + detail::JsonAddMember(extMap, extIt->first.c_str(), std::move(empty)); + } + } + } + detail::JsonAddMember(o, "extensions", std::move(extMap)); +} + +static void SerializeGltfAccessor(const Accessor &accessor, detail::json &o) { + if (accessor.bufferView >= 0) + SerializeNumberProperty("bufferView", accessor.bufferView, o); + + if (accessor.byteOffset != 0) + SerializeNumberProperty("byteOffset", int(accessor.byteOffset), o); + + SerializeNumberProperty("componentType", accessor.componentType, o); + SerializeNumberProperty("count", accessor.count, o); + + if ((accessor.componentType == TINYGLTF_COMPONENT_TYPE_FLOAT) || + (accessor.componentType == TINYGLTF_COMPONENT_TYPE_DOUBLE)) { + SerializeNumberArrayProperty("min", accessor.minValues, o); + SerializeNumberArrayProperty("max", accessor.maxValues, o); + } else { + // Issue #301. Serialize as integer. + // Assume int value is within [-2**31-1, 2**31-1] + { + std::vector values; + std::transform(accessor.minValues.begin(), accessor.minValues.end(), + std::back_inserter(values), + [](double v) { return static_cast(v); }); + + SerializeNumberArrayProperty("min", values, o); + } + + { + std::vector values; + std::transform(accessor.maxValues.begin(), accessor.maxValues.end(), + std::back_inserter(values), + [](double v) { return static_cast(v); }); + + SerializeNumberArrayProperty("max", values, o); + } + } + + if (accessor.normalized) + SerializeValue("normalized", Value(accessor.normalized), o); + std::string type; + switch (accessor.type) { + case TINYGLTF_TYPE_SCALAR: + type = "SCALAR"; + break; + case TINYGLTF_TYPE_VEC2: + type = "VEC2"; + break; + case TINYGLTF_TYPE_VEC3: + type = "VEC3"; + break; + case TINYGLTF_TYPE_VEC4: + type = "VEC4"; + break; + case TINYGLTF_TYPE_MAT2: + type = "MAT2"; + break; + case TINYGLTF_TYPE_MAT3: + type = "MAT3"; + break; + case TINYGLTF_TYPE_MAT4: + type = "MAT4"; + break; + } + + SerializeStringProperty("type", type, o); + if (!accessor.name.empty()) SerializeStringProperty("name", accessor.name, o); + + if (accessor.extras.Type() != NULL_TYPE) { + SerializeValue("extras", accessor.extras, o); + } + + // sparse + if (accessor.sparse.isSparse) + { + detail::json sparse; + SerializeNumberProperty("count", accessor.sparse.count, sparse); + { + detail::json indices; + SerializeNumberProperty("bufferView", accessor.sparse.indices.bufferView, indices); + SerializeNumberProperty("byteOffset", accessor.sparse.indices.byteOffset, indices); + SerializeNumberProperty("componentType", accessor.sparse.indices.componentType, indices); + detail::JsonAddMember(sparse, "indices", std::move(indices)); + } + { + detail::json values; + SerializeNumberProperty("bufferView", accessor.sparse.values.bufferView, values); + SerializeNumberProperty("byteOffset", accessor.sparse.values.byteOffset, values); + detail::JsonAddMember(sparse, "values", std::move(values)); + } + detail::JsonAddMember(o, "sparse", std::move(sparse)); + } +} + +static void SerializeGltfAnimationChannel(const AnimationChannel &channel, + detail::json &o) { + SerializeNumberProperty("sampler", channel.sampler, o); + { + detail::json target; + + if (channel.target_node >= 0) { + SerializeNumberProperty("node", channel.target_node, target); + } + + SerializeStringProperty("path", channel.target_path, target); + + SerializeExtensionMap(channel.target_extensions, target); + + detail::JsonAddMember(o, "target", std::move(target)); + } + + if (channel.extras.Type() != NULL_TYPE) { + SerializeValue("extras", channel.extras, o); + } + + SerializeExtensionMap(channel.extensions, o); +} + +static void SerializeGltfAnimationSampler(const AnimationSampler &sampler, + detail::json &o) { + SerializeNumberProperty("input", sampler.input, o); + SerializeNumberProperty("output", sampler.output, o); + SerializeStringProperty("interpolation", sampler.interpolation, o); + + if (sampler.extras.Type() != NULL_TYPE) { + SerializeValue("extras", sampler.extras, o); + } +} + +static void SerializeGltfAnimation(const Animation &animation, detail::json &o) { + if (!animation.name.empty()) + SerializeStringProperty("name", animation.name, o); + + { + detail::json channels; + detail::JsonReserveArray(channels, animation.channels.size()); + for (unsigned int i = 0; i < animation.channels.size(); ++i) { + detail::json channel; + AnimationChannel gltfChannel = animation.channels[i]; + SerializeGltfAnimationChannel(gltfChannel, channel); + detail::JsonPushBack(channels, std::move(channel)); + } + + detail::JsonAddMember(o, "channels", std::move(channels)); + } + + { + detail::json samplers; + detail::JsonReserveArray(samplers, animation.samplers.size()); + for (unsigned int i = 0; i < animation.samplers.size(); ++i) { + detail::json sampler; + AnimationSampler gltfSampler = animation.samplers[i]; + SerializeGltfAnimationSampler(gltfSampler, sampler); + detail::JsonPushBack(samplers, std::move(sampler)); + } + detail::JsonAddMember(o, "samplers", std::move(samplers)); + } + + if (animation.extras.Type() != NULL_TYPE) { + SerializeValue("extras", animation.extras, o); + } + + SerializeExtensionMap(animation.extensions, o); +} + +static void SerializeGltfAsset(const Asset &asset, detail::json &o) { + if (!asset.generator.empty()) { + SerializeStringProperty("generator", asset.generator, o); + } + + if (!asset.copyright.empty()) { + SerializeStringProperty("copyright", asset.copyright, o); + } + + auto version = asset.version; + if (version.empty()) { + // Just in case + // `version` must be defined + version = "2.0"; + } + + // TODO(syoyo): Do we need to check if `version` is greater or equal to 2.0? + SerializeStringProperty("version", version, o); + + if (asset.extras.Keys().size()) { + SerializeValue("extras", asset.extras, o); + } + + SerializeExtensionMap(asset.extensions, o); +} + +static void SerializeGltfBufferBin(const Buffer &buffer, detail::json &o, + std::vector &binBuffer) { + SerializeNumberProperty("byteLength", buffer.data.size(), o); + binBuffer = buffer.data; + + if (buffer.name.size()) SerializeStringProperty("name", buffer.name, o); + + if (buffer.extras.Type() != NULL_TYPE) { + SerializeValue("extras", buffer.extras, o); + } +} + +static void SerializeGltfBuffer(const Buffer &buffer, detail::json &o) { + SerializeNumberProperty("byteLength", buffer.data.size(), o); + SerializeGltfBufferData(buffer.data, o); + + if (buffer.name.size()) SerializeStringProperty("name", buffer.name, o); + + if (buffer.extras.Type() != NULL_TYPE) { + SerializeValue("extras", buffer.extras, o); + } +} + +static bool SerializeGltfBuffer(const Buffer &buffer, detail::json &o, + const std::string &binFilename, + const std::string &binUri) { + if (!SerializeGltfBufferData(buffer.data, binFilename)) return false; + SerializeNumberProperty("byteLength", buffer.data.size(), o); + SerializeStringProperty("uri", binUri, o); + + if (buffer.name.size()) SerializeStringProperty("name", buffer.name, o); + + if (buffer.extras.Type() != NULL_TYPE) { + SerializeValue("extras", buffer.extras, o); + } + return true; +} + +static void SerializeGltfBufferView(const BufferView &bufferView, detail::json &o) { + SerializeNumberProperty("buffer", bufferView.buffer, o); + SerializeNumberProperty("byteLength", bufferView.byteLength, o); + + // byteStride is optional, minimum allowed is 4 + if (bufferView.byteStride >= 4) { + SerializeNumberProperty("byteStride", bufferView.byteStride, o); + } + // byteOffset is optional, default is 0 + if (bufferView.byteOffset > 0) { + SerializeNumberProperty("byteOffset", bufferView.byteOffset, o); + } + // Target is optional, check if it contains a valid value + if (bufferView.target == TINYGLTF_TARGET_ARRAY_BUFFER || + bufferView.target == TINYGLTF_TARGET_ELEMENT_ARRAY_BUFFER) { + SerializeNumberProperty("target", bufferView.target, o); + } + if (bufferView.name.size()) { + SerializeStringProperty("name", bufferView.name, o); + } + + if (bufferView.extras.Type() != NULL_TYPE) { + SerializeValue("extras", bufferView.extras, o); + } +} + +static void SerializeGltfImage(const Image &image, const std::string &uri, + detail::json &o) { + // From 2.7.0, we look for `uri` parameter, not `Image.uri` + // if uri is empty, the mimeType and bufferview should be set + if (uri.empty()) { + SerializeStringProperty("mimeType", image.mimeType, o); + SerializeNumberProperty("bufferView", image.bufferView, o); + } else { + SerializeStringProperty("uri", uri, o); + } + + if (image.name.size()) { + SerializeStringProperty("name", image.name, o); + } + + if (image.extras.Type() != NULL_TYPE) { + SerializeValue("extras", image.extras, o); + } + + SerializeExtensionMap(image.extensions, o); +} + +static void SerializeGltfTextureInfo(const TextureInfo &texinfo, detail::json &o) { + SerializeNumberProperty("index", texinfo.index, o); + + if (texinfo.texCoord != 0) { + SerializeNumberProperty("texCoord", texinfo.texCoord, o); + } + + if (texinfo.extras.Type() != NULL_TYPE) { + SerializeValue("extras", texinfo.extras, o); + } + + SerializeExtensionMap(texinfo.extensions, o); +} + +static void SerializeGltfNormalTextureInfo(const NormalTextureInfo &texinfo, + detail::json &o) { + SerializeNumberProperty("index", texinfo.index, o); + + if (texinfo.texCoord != 0) { + SerializeNumberProperty("texCoord", texinfo.texCoord, o); + } + + if (!TINYGLTF_DOUBLE_EQUAL(texinfo.scale, 1.0)) { + SerializeNumberProperty("scale", texinfo.scale, o); + } + + if (texinfo.extras.Type() != NULL_TYPE) { + SerializeValue("extras", texinfo.extras, o); + } + + SerializeExtensionMap(texinfo.extensions, o); +} + +static void SerializeGltfOcclusionTextureInfo( + const OcclusionTextureInfo &texinfo, detail::json &o) { + SerializeNumberProperty("index", texinfo.index, o); + + if (texinfo.texCoord != 0) { + SerializeNumberProperty("texCoord", texinfo.texCoord, o); + } + + if (!TINYGLTF_DOUBLE_EQUAL(texinfo.strength, 1.0)) { + SerializeNumberProperty("strength", texinfo.strength, o); + } + + if (texinfo.extras.Type() != NULL_TYPE) { + SerializeValue("extras", texinfo.extras, o); + } + + SerializeExtensionMap(texinfo.extensions, o); +} + +static void SerializeGltfPbrMetallicRoughness(const PbrMetallicRoughness &pbr, + detail::json &o) { + std::vector default_baseColorFactor = {1.0, 1.0, 1.0, 1.0}; + if (!Equals(pbr.baseColorFactor, default_baseColorFactor)) { + SerializeNumberArrayProperty("baseColorFactor", pbr.baseColorFactor, + o); + } + + if (!TINYGLTF_DOUBLE_EQUAL(pbr.metallicFactor, 1.0)) { + SerializeNumberProperty("metallicFactor", pbr.metallicFactor, o); + } + + if (!TINYGLTF_DOUBLE_EQUAL(pbr.roughnessFactor, 1.0)) { + SerializeNumberProperty("roughnessFactor", pbr.roughnessFactor, o); + } + + if (pbr.baseColorTexture.index > -1) { + detail::json texinfo; + SerializeGltfTextureInfo(pbr.baseColorTexture, texinfo); + detail::JsonAddMember(o, "baseColorTexture", std::move(texinfo)); + } + + if (pbr.metallicRoughnessTexture.index > -1) { + detail::json texinfo; + SerializeGltfTextureInfo(pbr.metallicRoughnessTexture, texinfo); + detail::JsonAddMember(o, "metallicRoughnessTexture", std::move(texinfo)); + } + + SerializeExtensionMap(pbr.extensions, o); + + if (pbr.extras.Type() != NULL_TYPE) { + SerializeValue("extras", pbr.extras, o); + } +} + +static void SerializeGltfMaterial(const Material &material, detail::json &o) { + if (material.name.size()) { + SerializeStringProperty("name", material.name, o); + } + + // QUESTION(syoyo): Write material parameters regardless of its default value? + + if (!TINYGLTF_DOUBLE_EQUAL(material.alphaCutoff, 0.5)) { + SerializeNumberProperty("alphaCutoff", material.alphaCutoff, o); + } + + if (material.alphaMode.compare("OPAQUE") != 0) { + SerializeStringProperty("alphaMode", material.alphaMode, o); + } + + if (material.doubleSided != false) + detail::JsonAddMember(o, "doubleSided", detail::json(material.doubleSided)); + + if (material.normalTexture.index > -1) { + detail::json texinfo; + SerializeGltfNormalTextureInfo(material.normalTexture, texinfo); + detail::JsonAddMember(o, "normalTexture", std::move(texinfo)); + } + + if (material.occlusionTexture.index > -1) { + detail::json texinfo; + SerializeGltfOcclusionTextureInfo(material.occlusionTexture, texinfo); + detail::JsonAddMember(o, "occlusionTexture", std::move(texinfo)); + } + + if (material.emissiveTexture.index > -1) { + detail::json texinfo; + SerializeGltfTextureInfo(material.emissiveTexture, texinfo); + detail::JsonAddMember(o, "emissiveTexture", std::move(texinfo)); + } + + std::vector default_emissiveFactor = {0.0, 0.0, 0.0}; + if (!Equals(material.emissiveFactor, default_emissiveFactor)) { + SerializeNumberArrayProperty("emissiveFactor", + material.emissiveFactor, o); + } + + { + detail::json pbrMetallicRoughness; + SerializeGltfPbrMetallicRoughness(material.pbrMetallicRoughness, + pbrMetallicRoughness); + // Issue 204 + // Do not serialize `pbrMetallicRoughness` if pbrMetallicRoughness has all + // default values(json is null). Otherwise it will serialize to + // `pbrMetallicRoughness : null`, which cannot be read by other glTF + // importers (and validators). + // + if (!detail::JsonIsNull(pbrMetallicRoughness)) { + detail::JsonAddMember(o, "pbrMetallicRoughness", std::move(pbrMetallicRoughness)); + } + } + +#if 0 // legacy way. just for the record. + if (material.values.size()) { + detail::json pbrMetallicRoughness; + SerializeParameterMap(material.values, pbrMetallicRoughness); + detail::JsonAddMember(o, "pbrMetallicRoughness", std::move(pbrMetallicRoughness)); + } + + SerializeParameterMap(material.additionalValues, o); +#else + +#endif + + SerializeExtensionMap(material.extensions, o); + + if (material.extras.Type() != NULL_TYPE) { + SerializeValue("extras", material.extras, o); + } +} + +static void SerializeGltfMesh(const Mesh &mesh, detail::json &o) { + detail::json primitives; + detail::JsonReserveArray(primitives, mesh.primitives.size()); + for (unsigned int i = 0; i < mesh.primitives.size(); ++i) { + detail::json primitive; + const Primitive &gltfPrimitive = mesh.primitives[i]; // don't make a copy + { + detail::json attributes; + for (auto attrIt = gltfPrimitive.attributes.begin(); + attrIt != gltfPrimitive.attributes.end(); ++attrIt) { + SerializeNumberProperty(attrIt->first, attrIt->second, attributes); + } + + detail::JsonAddMember(primitive, "attributes", std::move(attributes)); + } + + // Indices is optional + if (gltfPrimitive.indices > -1) { + SerializeNumberProperty("indices", gltfPrimitive.indices, primitive); + } + // Material is optional + if (gltfPrimitive.material > -1) { + SerializeNumberProperty("material", gltfPrimitive.material, + primitive); + } + SerializeNumberProperty("mode", gltfPrimitive.mode, primitive); + + // Morph targets + if (gltfPrimitive.targets.size()) { + detail::json targets; + detail::JsonReserveArray(targets, gltfPrimitive.targets.size()); + for (unsigned int k = 0; k < gltfPrimitive.targets.size(); ++k) { + detail::json targetAttributes; + std::map targetData = gltfPrimitive.targets[k]; + for (std::map::iterator attrIt = targetData.begin(); + attrIt != targetData.end(); ++attrIt) { + SerializeNumberProperty(attrIt->first, attrIt->second, + targetAttributes); + } + detail::JsonPushBack(targets, std::move(targetAttributes)); + } + detail::JsonAddMember(primitive, "targets", std::move(targets)); + } + + SerializeExtensionMap(gltfPrimitive.extensions, primitive); + + if (gltfPrimitive.extras.Type() != NULL_TYPE) { + SerializeValue("extras", gltfPrimitive.extras, primitive); + } + + detail::JsonPushBack(primitives, std::move(primitive)); + } + + detail::JsonAddMember(o, "primitives", std::move(primitives)); + + if (mesh.weights.size()) { + SerializeNumberArrayProperty("weights", mesh.weights, o); + } + + if (mesh.name.size()) { + SerializeStringProperty("name", mesh.name, o); + } + + SerializeExtensionMap(mesh.extensions, o); + if (mesh.extras.Type() != NULL_TYPE) { + SerializeValue("extras", mesh.extras, o); + } +} + +static void SerializeSpotLight(const SpotLight &spot, detail::json &o) { + SerializeNumberProperty("innerConeAngle", spot.innerConeAngle, o); + SerializeNumberProperty("outerConeAngle", spot.outerConeAngle, o); + SerializeExtensionMap(spot.extensions, o); + if (spot.extras.Type() != NULL_TYPE) { + SerializeValue("extras", spot.extras, o); + } +} + +static void SerializeGltfLight(const Light &light, detail::json &o) { + if (!light.name.empty()) SerializeStringProperty("name", light.name, o); + SerializeNumberProperty("intensity", light.intensity, o); + if (light.range > 0.0) { + SerializeNumberProperty("range", light.range, o); + } + SerializeNumberArrayProperty("color", light.color, o); + SerializeStringProperty("type", light.type, o); + if (light.type == "spot") { + detail::json spot; + SerializeSpotLight(light.spot, spot); + detail::JsonAddMember(o, "spot", std::move(spot)); + } + SerializeExtensionMap(light.extensions, o); + if (light.extras.Type() != NULL_TYPE) { + SerializeValue("extras", light.extras, o); + } +} + +static void SerializeGltfNode(const Node &node, detail::json &o) { + if (node.translation.size() > 0) { + SerializeNumberArrayProperty("translation", node.translation, o); + } + if (node.rotation.size() > 0) { + SerializeNumberArrayProperty("rotation", node.rotation, o); + } + if (node.scale.size() > 0) { + SerializeNumberArrayProperty("scale", node.scale, o); + } + if (node.matrix.size() > 0) { + SerializeNumberArrayProperty("matrix", node.matrix, o); + } + if (node.mesh != -1) { + SerializeNumberProperty("mesh", node.mesh, o); + } + + if (node.skin != -1) { + SerializeNumberProperty("skin", node.skin, o); + } + + if (node.camera != -1) { + SerializeNumberProperty("camera", node.camera, o); + } + + if (node.weights.size() > 0) { + SerializeNumberArrayProperty("weights", node.weights, o); + } + + if (node.extras.Type() != NULL_TYPE) { + SerializeValue("extras", node.extras, o); + } + + SerializeExtensionMap(node.extensions, o); + if (!node.name.empty()) SerializeStringProperty("name", node.name, o); + SerializeNumberArrayProperty("children", node.children, o); +} + +static void SerializeGltfSampler(const Sampler &sampler, detail::json &o) { + if (!sampler.name.empty()) { + SerializeStringProperty("name", sampler.name, o); + } + if (sampler.magFilter != -1) { + SerializeNumberProperty("magFilter", sampler.magFilter, o); + } + if (sampler.minFilter != -1) { + SerializeNumberProperty("minFilter", sampler.minFilter, o); + } + // SerializeNumberProperty("wrapR", sampler.wrapR, o); + SerializeNumberProperty("wrapS", sampler.wrapS, o); + SerializeNumberProperty("wrapT", sampler.wrapT, o); + + if (sampler.extras.Type() != NULL_TYPE) { + SerializeValue("extras", sampler.extras, o); + } +} + +static void SerializeGltfOrthographicCamera(const OrthographicCamera &camera, + detail::json &o) { + SerializeNumberProperty("zfar", camera.zfar, o); + SerializeNumberProperty("znear", camera.znear, o); + SerializeNumberProperty("xmag", camera.xmag, o); + SerializeNumberProperty("ymag", camera.ymag, o); + + if (camera.extras.Type() != NULL_TYPE) { + SerializeValue("extras", camera.extras, o); + } +} + +static void SerializeGltfPerspectiveCamera(const PerspectiveCamera &camera, + detail::json &o) { + SerializeNumberProperty("zfar", camera.zfar, o); + SerializeNumberProperty("znear", camera.znear, o); + if (camera.aspectRatio > 0) { + SerializeNumberProperty("aspectRatio", camera.aspectRatio, o); + } + + if (camera.yfov > 0) { + SerializeNumberProperty("yfov", camera.yfov, o); + } + + if (camera.extras.Type() != NULL_TYPE) { + SerializeValue("extras", camera.extras, o); + } +} + +static void SerializeGltfCamera(const Camera &camera, detail::json &o) { + SerializeStringProperty("type", camera.type, o); + if (!camera.name.empty()) { + SerializeStringProperty("name", camera.name, o); + } + + if (camera.type.compare("orthographic") == 0) { + detail::json orthographic; + SerializeGltfOrthographicCamera(camera.orthographic, orthographic); + detail::JsonAddMember(o, "orthographic", std::move(orthographic)); + } else if (camera.type.compare("perspective") == 0) { + detail::json perspective; + SerializeGltfPerspectiveCamera(camera.perspective, perspective); + detail::JsonAddMember(o, "perspective", std::move(perspective)); + } else { + // ??? + } + + if (camera.extras.Type() != NULL_TYPE) { + SerializeValue("extras", camera.extras, o); + } + SerializeExtensionMap(camera.extensions, o); +} + +static void SerializeGltfScene(const Scene &scene, detail::json &o) { + SerializeNumberArrayProperty("nodes", scene.nodes, o); + + if (scene.name.size()) { + SerializeStringProperty("name", scene.name, o); + } + if (scene.extras.Type() != NULL_TYPE) { + SerializeValue("extras", scene.extras, o); + } + SerializeExtensionMap(scene.extensions, o); +} + +static void SerializeGltfSkin(const Skin &skin, detail::json &o) { + // required + SerializeNumberArrayProperty("joints", skin.joints, o); + + if (skin.inverseBindMatrices >= 0) { + SerializeNumberProperty("inverseBindMatrices", skin.inverseBindMatrices, o); + } + + if (skin.skeleton >= 0) { + SerializeNumberProperty("skeleton", skin.skeleton, o); + } + + if (skin.name.size()) { + SerializeStringProperty("name", skin.name, o); + } +} + +static void SerializeGltfTexture(const Texture &texture, detail::json &o) { + if (texture.sampler > -1) { + SerializeNumberProperty("sampler", texture.sampler, o); + } + if (texture.source > -1) { + SerializeNumberProperty("source", texture.source, o); + } + if (texture.name.size()) { + SerializeStringProperty("name", texture.name, o); + } + if (texture.extras.Type() != NULL_TYPE) { + SerializeValue("extras", texture.extras, o); + } + SerializeExtensionMap(texture.extensions, o); +} + +/// +/// Serialize all properties except buffers and images. +/// +static void SerializeGltfModel(const Model *model, detail::json &o) { + // ACCESSORS + if (model->accessors.size()) { + detail::json accessors; + detail::JsonReserveArray(accessors, model->accessors.size()); + for (unsigned int i = 0; i < model->accessors.size(); ++i) { + detail::json accessor; + SerializeGltfAccessor(model->accessors[i], accessor); + detail::JsonPushBack(accessors, std::move(accessor)); + } + detail::JsonAddMember(o, "accessors", std::move(accessors)); + } + + // ANIMATIONS + if (model->animations.size()) { + detail::json animations; + detail::JsonReserveArray(animations, model->animations.size()); + for (unsigned int i = 0; i < model->animations.size(); ++i) { + if (model->animations[i].channels.size()) { + detail::json animation; + SerializeGltfAnimation(model->animations[i], animation); + detail::JsonPushBack(animations, std::move(animation)); + } + } + + detail::JsonAddMember(o, "animations", std::move(animations)); + } + + // ASSET + detail::json asset; + SerializeGltfAsset(model->asset, asset); + detail::JsonAddMember(o, "asset", std::move(asset)); + + // BUFFERVIEWS + if (model->bufferViews.size()) { + detail::json bufferViews; + detail::JsonReserveArray(bufferViews, model->bufferViews.size()); + for (unsigned int i = 0; i < model->bufferViews.size(); ++i) { + detail::json bufferView; + SerializeGltfBufferView(model->bufferViews[i], bufferView); + detail::JsonPushBack(bufferViews, std::move(bufferView)); + } + detail::JsonAddMember(o, "bufferViews", std::move(bufferViews)); + } + + // Extensions required + if (model->extensionsRequired.size()) { + SerializeStringArrayProperty("extensionsRequired", + model->extensionsRequired, o); + } + + // MATERIALS + if (model->materials.size()) { + detail::json materials; + detail::JsonReserveArray(materials, model->materials.size()); + for (unsigned int i = 0; i < model->materials.size(); ++i) { + detail::json material; + SerializeGltfMaterial(model->materials[i], material); + + if (detail::JsonIsNull(material)) { + // Issue 294. + // `material` does not have any required parameters + // so the result may be null(unmodified) when all material parameters + // have default value. + // + // null is not allowed thus we create an empty JSON object. + detail::JsonSetObject(material); + } + detail::JsonPushBack(materials, std::move(material)); + } + detail::JsonAddMember(o, "materials", std::move(materials)); + } + + // MESHES + if (model->meshes.size()) { + detail::json meshes; + detail::JsonReserveArray(meshes, model->meshes.size()); + for (unsigned int i = 0; i < model->meshes.size(); ++i) { + detail::json mesh; + SerializeGltfMesh(model->meshes[i], mesh); + detail::JsonPushBack(meshes, std::move(mesh)); + } + detail::JsonAddMember(o, "meshes", std::move(meshes)); + } + + // NODES + if (model->nodes.size()) { + detail::json nodes; + detail::JsonReserveArray(nodes, model->nodes.size()); + for (unsigned int i = 0; i < model->nodes.size(); ++i) { + detail::json node; + SerializeGltfNode(model->nodes[i], node); + detail::JsonPushBack(nodes, std::move(node)); + } + detail::JsonAddMember(o, "nodes", std::move(nodes)); + } + + // SCENE + if (model->defaultScene > -1) { + SerializeNumberProperty("scene", model->defaultScene, o); + } + + // SCENES + if (model->scenes.size()) { + detail::json scenes; + detail::JsonReserveArray(scenes, model->scenes.size()); + for (unsigned int i = 0; i < model->scenes.size(); ++i) { + detail::json currentScene; + SerializeGltfScene(model->scenes[i], currentScene); + detail::JsonPushBack(scenes, std::move(currentScene)); + } + detail::JsonAddMember(o, "scenes", std::move(scenes)); + } + + // SKINS + if (model->skins.size()) { + detail::json skins; + detail::JsonReserveArray(skins, model->skins.size()); + for (unsigned int i = 0; i < model->skins.size(); ++i) { + detail::json skin; + SerializeGltfSkin(model->skins[i], skin); + detail::JsonPushBack(skins, std::move(skin)); + } + detail::JsonAddMember(o, "skins", std::move(skins)); + } + + // TEXTURES + if (model->textures.size()) { + detail::json textures; + detail::JsonReserveArray(textures, model->textures.size()); + for (unsigned int i = 0; i < model->textures.size(); ++i) { + detail::json texture; + SerializeGltfTexture(model->textures[i], texture); + detail::JsonPushBack(textures, std::move(texture)); + } + detail::JsonAddMember(o, "textures", std::move(textures)); + } + + // SAMPLERS + if (model->samplers.size()) { + detail::json samplers; + detail::JsonReserveArray(samplers, model->samplers.size()); + for (unsigned int i = 0; i < model->samplers.size(); ++i) { + detail::json sampler; + SerializeGltfSampler(model->samplers[i], sampler); + detail::JsonPushBack(samplers, std::move(sampler)); + } + detail::JsonAddMember(o, "samplers", std::move(samplers)); + } + + // CAMERAS + if (model->cameras.size()) { + detail::json cameras; + detail::JsonReserveArray(cameras, model->cameras.size()); + for (unsigned int i = 0; i < model->cameras.size(); ++i) { + detail::json camera; + SerializeGltfCamera(model->cameras[i], camera); + detail::JsonPushBack(cameras, std::move(camera)); + } + detail::JsonAddMember(o, "cameras", std::move(cameras)); + } + + // EXTENSIONS + SerializeExtensionMap(model->extensions, o); + + auto extensionsUsed = model->extensionsUsed; + + // LIGHTS as KHR_lights_punctual + if (model->lights.size()) { + detail::json lights; + detail::JsonReserveArray(lights, model->lights.size()); + for (unsigned int i = 0; i < model->lights.size(); ++i) { + detail::json light; + SerializeGltfLight(model->lights[i], light); + detail::JsonPushBack(lights, std::move(light)); + } + detail::json khr_lights_cmn; + detail::JsonAddMember(khr_lights_cmn, "lights", std::move(lights)); + detail::json ext_j; + + { + detail::json_const_iterator it; + if (detail::FindMember(o, "extensions", it)) { + detail::JsonAssign(ext_j, detail::GetValue(it)); + } + } + + detail::JsonAddMember(ext_j, "KHR_lights_punctual", std::move(khr_lights_cmn)); + + detail::JsonAddMember(o, "extensions", std::move(ext_j)); + + // Also add "KHR_lights_punctual" to `extensionsUsed` + { + auto has_khr_lights_punctual = + std::find_if(extensionsUsed.begin(), extensionsUsed.end(), + [](const std::string &s) { + return (s.compare("KHR_lights_punctual") == 0); + }); + + if (has_khr_lights_punctual == extensionsUsed.end()) { + extensionsUsed.push_back("KHR_lights_punctual"); + } + } + } + + // Extensions used + if (extensionsUsed.size()) { + SerializeStringArrayProperty("extensionsUsed", extensionsUsed, o); + } + + // EXTRAS + if (model->extras.Type() != NULL_TYPE) { + SerializeValue("extras", model->extras, o); + } +} + +static bool WriteGltfStream(std::ostream &stream, const std::string &content) { + stream << content << std::endl; + return stream.good(); +} + +static bool WriteGltfFile(const std::string &output, + const std::string &content) { +#ifdef _WIN32 +#if defined(_MSC_VER) + std::ofstream gltfFile(UTF8ToWchar(output).c_str()); +#elif defined(__GLIBCXX__) + int file_descriptor = _wopen(UTF8ToWchar(output).c_str(), + _O_CREAT | _O_WRONLY | _O_TRUNC | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf( + file_descriptor, std::ios_base::out | std::ios_base::binary); + std::ostream gltfFile(&wfile_buf); + if (!wfile_buf.is_open()) return false; +#else + std::ofstream gltfFile(output.c_str()); + if (!gltfFile.is_open()) return false; +#endif +#else + std::ofstream gltfFile(output.c_str()); + if (!gltfFile.is_open()) return false; +#endif + return WriteGltfStream(gltfFile, content); +} + +static bool WriteBinaryGltfStream(std::ostream &stream, + const std::string &content, + const std::vector &binBuffer) { + const std::string header = "glTF"; + const int version = 2; + + const uint32_t content_size = uint32_t(content.size()); + const uint32_t binBuffer_size = uint32_t(binBuffer.size()); + // determine number of padding bytes required to ensure 4 byte alignment + const uint32_t content_padding_size = + content_size % 4 == 0 ? 0 : 4 - content_size % 4; + const uint32_t bin_padding_size = + binBuffer_size % 4 == 0 ? 0 : 4 - binBuffer_size % 4; + + // 12 bytes for header, JSON content length, 8 bytes for JSON chunk info. + // Chunk data must be located at 4-byte boundary, which may require padding + const uint32_t length = + 12 + 8 + content_size + content_padding_size + + (binBuffer_size ? (8 + binBuffer_size + bin_padding_size) : 0); + + stream.write(header.c_str(), std::streamsize(header.size())); + stream.write(reinterpret_cast(&version), sizeof(version)); + stream.write(reinterpret_cast(&length), sizeof(length)); + + // JSON chunk info, then JSON data + const uint32_t model_length = uint32_t(content.size()) + content_padding_size; + const uint32_t model_format = 0x4E4F534A; + stream.write(reinterpret_cast(&model_length), + sizeof(model_length)); + stream.write(reinterpret_cast(&model_format), + sizeof(model_format)); + stream.write(content.c_str(), std::streamsize(content.size())); + + // Chunk must be multiplies of 4, so pad with spaces + if (content_padding_size > 0) { + const std::string padding = std::string(size_t(content_padding_size), ' '); + stream.write(padding.c_str(), std::streamsize(padding.size())); + } + if (binBuffer.size() > 0) { + // BIN chunk info, then BIN data + const uint32_t bin_length = uint32_t(binBuffer.size()) + bin_padding_size; + const uint32_t bin_format = 0x004e4942; + stream.write(reinterpret_cast(&bin_length), + sizeof(bin_length)); + stream.write(reinterpret_cast(&bin_format), + sizeof(bin_format)); + stream.write(reinterpret_cast(binBuffer.data()), + std::streamsize(binBuffer.size())); + // Chunksize must be multiplies of 4, so pad with zeroes + if (bin_padding_size > 0) { + const std::vector padding = + std::vector(size_t(bin_padding_size), 0); + stream.write(reinterpret_cast(padding.data()), + std::streamsize(padding.size())); + } + } + + stream.flush(); + return stream.good(); +} + +static bool WriteBinaryGltfFile(const std::string &output, + const std::string &content, + const std::vector &binBuffer) { +#ifdef _WIN32 +#if defined(_MSC_VER) + std::ofstream gltfFile(UTF8ToWchar(output).c_str(), std::ios::binary); +#elif defined(__GLIBCXX__) + int file_descriptor = _wopen(UTF8ToWchar(output).c_str(), + _O_CREAT | _O_WRONLY | _O_TRUNC | _O_BINARY); + __gnu_cxx::stdio_filebuf wfile_buf( + file_descriptor, std::ios_base::out | std::ios_base::binary); + std::ostream gltfFile(&wfile_buf); +#else + std::ofstream gltfFile(output.c_str(), std::ios::binary); +#endif +#else + std::ofstream gltfFile(output.c_str(), std::ios::binary); +#endif + return WriteBinaryGltfStream(gltfFile, content, binBuffer); +} + +bool TinyGLTF::WriteGltfSceneToStream(const Model *model, std::ostream &stream, + bool prettyPrint = true, + bool writeBinary = false) { + detail::JsonDocument output; + + /// Serialize all properties except buffers and images. + SerializeGltfModel(model, output); + + // BUFFERS + std::vector binBuffer; + if (model->buffers.size()) { + detail::json buffers; + detail::JsonReserveArray(buffers, model->buffers.size()); + for (unsigned int i = 0; i < model->buffers.size(); ++i) { + detail::json buffer; + if (writeBinary && i == 0 && model->buffers[i].uri.empty()) { + SerializeGltfBufferBin(model->buffers[i], buffer, binBuffer); + } else { + SerializeGltfBuffer(model->buffers[i], buffer); + } + detail::JsonPushBack(buffers, std::move(buffer)); + } + detail::JsonAddMember(output, "buffers", std::move(buffers)); + } + + // IMAGES + if (model->images.size()) { + detail::json images; + detail::JsonReserveArray(images, model->images.size()); + for (unsigned int i = 0; i < model->images.size(); ++i) { + detail::json image; + + std::string dummystring = ""; + // UpdateImageObject need baseDir but only uses it if embeddedImages is + // enabled, since we won't write separate images when writing to a stream + // we + std::string uri; + if (!UpdateImageObject(model->images[i], dummystring, int(i), true, + &uri_cb, &this->WriteImageData, + this->write_image_user_data_, &uri)) { + return false; + } + SerializeGltfImage(model->images[i], uri, image); + detail::JsonPushBack(images, std::move(image)); + } + detail::JsonAddMember(output, "images", std::move(images)); + } + + if (writeBinary) { + return WriteBinaryGltfStream(stream, detail::JsonToString(output), binBuffer); + } else { + return WriteGltfStream(stream, detail::JsonToString(output, prettyPrint ? 2 : -1)); + } +} + +bool TinyGLTF::WriteGltfSceneToFile(const Model *model, + const std::string &filename, + bool embedImages = false, + bool embedBuffers = false, + bool prettyPrint = true, + bool writeBinary = false) { + detail::JsonDocument output; + std::string defaultBinFilename = GetBaseFilename(filename); + std::string defaultBinFileExt = ".bin"; + std::string::size_type pos = + defaultBinFilename.rfind('.', defaultBinFilename.length()); + + if (pos != std::string::npos) { + defaultBinFilename = defaultBinFilename.substr(0, pos); + } + std::string baseDir = GetBaseDir(filename); + if (baseDir.empty()) { + baseDir = "./"; + } + /// Serialize all properties except buffers and images. + SerializeGltfModel(model, output); + + // BUFFERS + std::vector usedFilenames; + std::vector binBuffer; + if (model->buffers.size()) { + detail::json buffers; + detail::JsonReserveArray(buffers, model->buffers.size()); + for (unsigned int i = 0; i < model->buffers.size(); ++i) { + detail::json buffer; + if (writeBinary && i == 0 && model->buffers[i].uri.empty()) { + SerializeGltfBufferBin(model->buffers[i], buffer, binBuffer); + } else if (embedBuffers) { + SerializeGltfBuffer(model->buffers[i], buffer); + } else { + std::string binSavePath; + std::string binFilename; + std::string binUri; + if (!model->buffers[i].uri.empty() && + !IsDataURI(model->buffers[i].uri)) { + binUri = model->buffers[i].uri; + if (!uri_cb.decode(binUri, &binFilename, uri_cb.user_data)) { + return false; + } + } else { + binFilename = defaultBinFilename + defaultBinFileExt; + bool inUse = true; + int numUsed = 0; + while (inUse) { + inUse = false; + for (const std::string &usedName : usedFilenames) { + if (binFilename.compare(usedName) != 0) continue; + inUse = true; + binFilename = defaultBinFilename + std::to_string(numUsed++) + + defaultBinFileExt; + break; + } + } + + if (uri_cb.encode) { + if (!uri_cb.encode(binFilename, "buffer", &binUri, + uri_cb.user_data)) { + return false; + } + } else { + binUri = binFilename; + } + } + usedFilenames.push_back(binFilename); + binSavePath = JoinPath(baseDir, binFilename); + if (!SerializeGltfBuffer(model->buffers[i], buffer, binSavePath, + binUri)) { + return false; + } + } + detail::JsonPushBack(buffers, std::move(buffer)); + } + detail::JsonAddMember(output, "buffers", std::move(buffers)); + } + + // IMAGES + if (model->images.size()) { + detail::json images; + detail::JsonReserveArray(images, model->images.size()); + for (unsigned int i = 0; i < model->images.size(); ++i) { + detail::json image; + + std::string uri; + if (!UpdateImageObject(model->images[i], baseDir, int(i), embedImages, + &uri_cb, &this->WriteImageData, + this->write_image_user_data_, &uri)) { + return false; + } + SerializeGltfImage(model->images[i], uri, image); + detail::JsonPushBack(images, std::move(image)); + } + detail::JsonAddMember(output, "images", std::move(images)); + } + + if (writeBinary) { + return WriteBinaryGltfFile(filename, detail::JsonToString(output), binBuffer); + } else { + return WriteGltfFile(filename, detail::JsonToString(output, (prettyPrint ? 2 : -1))); + } +} + +} // namespace tinygltf + +#ifdef __clang__ +#pragma clang diagnostic pop +#endif + +#endif // TINYGLTF_IMPLEMENTATION diff --git a/src/loader/AndroidManifest.xml b/src/loader/AndroidManifest.xml index 291b1d30..c5f71935 100644 --- a/src/loader/AndroidManifest.xml +++ b/src/loader/AndroidManifest.xml @@ -9,7 +9,14 @@ + + + + + + + diff --git a/src/loader/AndroidManifest.xml.in b/src/loader/AndroidManifest.xml.in index 4315f57d..ac2a0693 100644 --- a/src/loader/AndroidManifest.xml.in +++ b/src/loader/AndroidManifest.xml.in @@ -1,18 +1,24 @@ - - + + + + + + + + diff --git a/src/loader/CMakeLists.txt b/src/loader/CMakeLists.txt index 8c89171d..9225bba2 100644 --- a/src/loader/CMakeLists.txt +++ b/src/loader/CMakeLists.txt @@ -1,4 +1,4 @@ -# Copyright (c) 2017 The Khronos Group Inc. +# Copyright (c) 2017-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 # @@ -14,8 +14,6 @@ # See the License for the specific language governing permissions and # limitations under the License. # -# Author: -# if(WIN32) # Pass version fields as preprocessor for .rc resource version on Windows. @@ -275,7 +273,11 @@ if(CMAKE_COMPILER_IS_GNUCC OR CMAKE_C_COMPILER_ID MATCHES "Clang") # Make build depend on the version script/export map target_sources(openxr_loader PRIVATE ${CMAKE_CURRENT_SOURCE_DIR}/openxr-loader.map) # Add the linker flag. - set_target_properties(openxr_loader PROPERTIES LINK_FLAGS "-Wl,--version-script=${CMAKE_CURRENT_SOURCE_DIR}/openxr-loader.map") + if(APPLE) + set_target_properties(openxr_loader PROPERTIES LINK_FLAGS "-Wl,-exported_symbols_list,${CMAKE_CURRENT_SOURCE_DIR}/openxr-loader.expsym") + else() + set_target_properties(openxr_loader PROPERTIES LINK_FLAGS "-Wl,--version-script=${CMAKE_CURRENT_SOURCE_DIR}/openxr-loader.map") + endif() # For GCC version 7.1 or greater, we need to disable the implicit fallthrough warning since # there's no consistent way to satisfy all compilers until they all accept the C++17 standard if(CMAKE_COMPILER_IS_GNUCC AND NOT (CMAKE_CXX_COMPILER_VERSION LESS 7.1)) @@ -284,7 +286,10 @@ if(CMAKE_COMPILER_IS_GNUCC OR CMAKE_C_COMPILER_ID MATCHES "Clang") endif() add_library(headers INTERFACE) -target_include_directories(headers INTERFACE $) +target_include_directories(headers INTERFACE + $ + $ + $) # We will never actually install here, but it's easier to set something than block all install code. if(ANDROID AND NOT INSTALL_TO_ARCHITECTURE_PREFIXES) @@ -305,6 +310,10 @@ export( NAMESPACE OpenXR:: ) +# Create aliases so that it can be used the same whether vendored as source or found with CMake. +add_library(OpenXR::openxr_loader ALIAS openxr_loader) +add_library(OpenXR::headers ALIAS headers) + if(WIN32 AND NOT INSTALL_TO_ARCHITECTURE_PREFIXES) set(TARGET_DESTINATION cmake) else() diff --git a/src/loader/android_utilities.cpp b/src/loader/android_utilities.cpp index 11c112dc..9a3ad76c 100644 --- a/src/loader/android_utilities.cpp +++ b/src/loader/android_utilities.cpp @@ -19,10 +19,10 @@ #include #include -#define ALOGE(...) __android_log_print(ANDROID_LOG_ERROR, "openxr_loader", __VA_ARGS__) -#define ALOGW(...) __android_log_print(ANDROID_LOG_WARN, "openxr_loader", __VA_ARGS__) -#define ALOGV(...) __android_log_print(ANDROID_LOG_VERBOSE, "openxr_loader", __VA_ARGS__) -#define ALOGI(...) __android_log_print(ANDROID_LOG_INFO, "openxr_loader", __VA_ARGS__) +#define ALOGE(...) __android_log_print(ANDROID_LOG_ERROR, "OpenXR-Loader", __VA_ARGS__) +#define ALOGW(...) __android_log_print(ANDROID_LOG_WARN, "OpenXR-Loader", __VA_ARGS__) +#define ALOGV(...) __android_log_print(ANDROID_LOG_VERBOSE, "OpenXR-Loader", __VA_ARGS__) +#define ALOGI(...) __android_log_print(ANDROID_LOG_INFO, "OpenXR-Loader", __VA_ARGS__) namespace openxr_android { using wrap::android::content::ContentUris; diff --git a/src/loader/api_layer_interface.cpp b/src/loader/api_layer_interface.cpp index 93a225f3..5560c31a 100644 --- a/src/loader/api_layer_interface.cpp +++ b/src/loader/api_layer_interface.cpp @@ -82,6 +82,12 @@ XrResult ApiLayerInterface::GetApiLayerProperties(const std::string& openxr_comm return result; } + // check for potential overflow before static_cast + if (manifest_files.size() >= UINT32_MAX) { + LoaderLogger::LogErrorMessage(openxr_command, "ApiLayerInterface::GetApiLayerProperties - too many API layers found"); + return XR_ERROR_RUNTIME_FAILURE; + } + manifest_count = static_cast(manifest_files.size()); if (nullptr == outgoing_count) { LoaderLogger::LogErrorMessage("xrEnumerateInstanceExtensionProperties", @@ -131,8 +137,8 @@ XrResult ApiLayerInterface::GetInstanceExtensionProperties(const std::string& op } bool found = false; - auto num_files = static_cast(manifest_files.size()); - for (uint32_t man_file = 0; man_file < num_files; ++man_file) { + size_t num_files = manifest_files.size(); + for (size_t man_file = 0; man_file < num_files; ++man_file) { // If a layer with the provided name exists, get it's instance extension information. if (manifest_files[man_file]->LayerName() == layer_name) { manifest_files[man_file]->GetInstanceExtensionProperties(extension_properties); @@ -172,8 +178,8 @@ XrResult ApiLayerInterface::GetInstanceExtensionProperties(const std::string& op } // Grab the layer instance extensions information - auto num_files = static_cast(manifest_files.size()); - for (uint32_t man_file = 0; man_file < num_files; ++man_file) { + size_t num_files = manifest_files.size(); + for (size_t man_file = 0; man_file < num_files; ++man_file) { manifest_files[man_file]->GetInstanceExtensionProperties(extension_properties); } } diff --git a/src/loader/build.gradle b/src/loader/build.gradle deleted file mode 100644 index acf7971f..00000000 --- a/src/loader/build.gradle +++ /dev/null @@ -1,204 +0,0 @@ -// Copyright (c) 2020-2023, The Khronos Group Inc. -// -// SPDX-License-Identifier: Apache-2.0 - -// Open this directory in Android Studio, or build with Gradle, -// as one way to create an AAR file of the loader. -// The other way is with maintainer-scripts/build-aar.sh which is simpler. -buildscript { - repositories { - google() - mavenCentral() - } - dependencies { - classpath "com.android.tools.build:gradle:4.1.3" - } -} - -plugins { - id "ru.vyarus.use-python" version "2.3.0" - id "maven-publish" -} - -apply plugin: "com.android.library" - -// These next few lines are just to make the version match the OpenXR release. -def repoRoot = file("../..") -project.ext.repoRoot = repoRoot -apply from: new File(repoRoot, "src/version.gradle") - -repositories { - google() - mavenCentral() -} - -def baseIncludeDir = new File(buildDir, "intermediates/includes") -def includeDir = layout.buildDirectory.file("intermediates/includes/openxr") -def scriptDir = "${project.repoRoot}/specification/scripts" -def registry = "${project.repoRoot}/specification/registry/xr.xml" - -ext.stl = "c++_static" - -// Python is used to generate header files -python { - pip "jinja2:2.10.3" - pip "MarkupSafe:2.0.1" - minPythonVersion = "3.4" - - environment = ["PYTHONPATH": scriptDir] -} - -def ensureDirectory = tasks.register("ensureDirectory") { - doLast { - mkdir(includeDir) - } -} - -// Copy or generate the OpenXR headers for packaging in the prefab AAR. -// Cannot just use ones from the build here, unfortunately. -tasks.addRule("Pattern: stageHeader_") { String taskName -> - if (!taskName.startsWith("stageHeader_")) { - return - } - def headerName = taskName - "stageHeader_" - - def pregenerated = new File(repoRoot, "include/openxr/${headerName}.h") - - if (pregenerated.exists()) { - // if the file exists, our task just copies it. - task(taskName, type: Copy) { - from pregenerated - into includeDir - dependsOn ensureDirectory - } - - } else { - - // if the file does not exist, our task generates it. - task(taskName, type: PythonTask) { - command = ["$scriptDir/genxr.py", - "-registry", - registry, - "-quiet", - "-o", - includeDir.get(), - "${headerName}.h"] - dependsOn ensureDirectory - } - } -} - - -preBuild.dependsOn stageHeader_openxr, - stageHeader_openxr_platform, - stageHeader_openxr_reflection, - stageHeader_openxr_platform_defines - -// Used for publishing/deploying -version = project.versionOpenXR - -android { - compileSdkVersion 29 - buildToolsVersion "30.0.3" - ndkVersion "21.3.6528147" - defaultConfig { - // for Vulkan, need at least 24 - minSdkVersion 24 - - versionName = project.versionOpenXR.toString() + project.versionQualifier - versionCode = project.versionOpenXR.getVersionCode() - externalNativeBuild { - cmake { - arguments "-DANDROID_STL=${stl}", - "-DBUILD_API_LAYERS=OFF", - "-DBUILD_TESTS=OFF", - "-DBUILD_LOADER=ON", - "-DBUILD_CONFORMANCE_TESTS=OFF", - "-DBUILD_ALL_EXTENSIONS=ON" - targets "openxr_loader" - } - } - } - sourceSets { - main { - manifest.srcFile "AndroidManifest.xml" - } - } - - buildTypes { - release { - minifyEnabled false - } - } - - externalNativeBuild { - cmake { - path "${project.repoRoot}/CMakeLists.txt" - } - } - - buildFeatures { - prefabPublishing true - } - prefab { - openxr_loader { - headers "${baseIncludeDir}" - } - } -} - -// Note: While publishing is available in this build file, -// what seems to be a bug in the Android gradle plugin duplicates -// the files in the .aar file, so the alternate approach is recommended for making/publishing -// artifacts for now. - -def siteUrl = "https://github.com/KhronosGroup/OpenXR-SDK-Source" -def gitUrl = "scm:git:https://github.com/KhronosGroup/OpenXR-SDK-Source.git" -publishing { - publications { - // Creates a Maven publication called "maven". - maven(MavenPublication) { - - artifactId "openxr_loader_for_android" - version project.versionOpenXR.toString() + project.versionQualifier - - pom { - name = "OpenXR Loader for Android" - description = "The AAR for the OpenXR Loader as used on Android." - url = siteUrl - - // Set your license - licenses { - license { - name = "The Apache Software License, Version 2.0" - url = "http://www.apache.org/licenses/LICENSE-2.0.txt" - } - // or MIT, but not easy to express clearly in POM. - } - developers { - developer { - name = "The Khronos Group, Inc." - email = "openxr-speceditor AT khronos DOT org" - } - } - scm { - connection = gitUrl - developerConnection = gitUrl - url = siteUrl - } - } - repositories { - maven { - url = uri(layout.buildDirectory.dir("repo")) - } - } - } - - } -} - -// Applies the component for the release build variant. -afterEvaluate { - publishing.publications["maven"].from(components.release) -} - diff --git a/src/loader/loader_instance.cpp b/src/loader/loader_instance.cpp index 21fc4632..badd3919 100644 --- a/src/loader/loader_instance.cpp +++ b/src/loader/loader_instance.cpp @@ -200,8 +200,8 @@ XrResult LoaderInstance::CreateInstance(PFN_xrGetInstanceProcAddr get_instance_p if (!api_layer_interfaces.empty()) { // Initialize an array of ApiLayerNextInfo structs std::unique_ptr next_info_list(new XrApiLayerNextInfo[api_layer_interfaces.size()]); - auto ni_index = static_cast(api_layer_interfaces.size() - 1); - for (uint32_t i = 0; i <= ni_index; i++) { + size_t ni_index = api_layer_interfaces.size() - 1; + for (size_t i = 0; i <= ni_index; i++) { next_info_list[i].structType = XR_LOADER_INTERFACE_STRUCT_API_LAYER_NEXT_INFO; next_info_list[i].structVersion = XR_API_LAYER_NEXT_INFO_STRUCT_VERSION; next_info_list[i].structSize = sizeof(XrApiLayerNextInfo); diff --git a/src/loader/loader_logger_recorders.cpp b/src/loader/loader_logger_recorders.cpp index 1b3892b8..32e4687b 100644 --- a/src/loader/loader_logger_recorders.cpp +++ b/src/loader/loader_logger_recorders.cpp @@ -160,12 +160,15 @@ bool DebugUtilsLogRecorder::LogMessage(XrLoaderLogMessageSeverityFlagBits messag XrDebugUtilsMessageTypeFlagsEXT utils_type = LoaderLogMessageTypesToDebugUtilsMessageTypes(message_type); // Convert the loader log message into the debug utils log message information - XrDebugUtilsMessengerCallbackDataEXT utils_callback_data = {XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + XrDebugUtilsMessengerCallbackDataEXT utils_callback_data{}; + utils_callback_data.type = XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT; utils_callback_data.messageId = callback_data->message_id; utils_callback_data.functionName = callback_data->command_name; utils_callback_data.message = callback_data->message; - std::vector utils_objects(callback_data->object_count, - {XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT}); + + XrDebugUtilsObjectNameInfoEXT example_utils_info{}; + example_utils_info.type = XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT; + std::vector utils_objects(callback_data->object_count, example_utils_info); for (uint8_t object = 0; object < callback_data->object_count; ++object) { utils_objects[object].objectHandle = callback_data->objects[object].handle; utils_objects[object].objectType = callback_data->objects[object].type; diff --git a/src/loader/manifest_file.cpp b/src/loader/manifest_file.cpp index bba6b8ca..99f4e841 100644 --- a/src/loader/manifest_file.cpp +++ b/src/loader/manifest_file.cpp @@ -233,6 +233,12 @@ static void ReadDataFilesInSearchPaths(const std::string &override_env_var, cons relative_home_path += relative_path; CopyIncludedPaths(true, home, relative_home_path, search_path); } +#elif defined(XR_OS_ANDROID) + CopyIncludedPaths(true, "/product/etc", relative_path, search_path); + CopyIncludedPaths(true, "/odm/etc", relative_path, search_path); + CopyIncludedPaths(true, "/oem/etc", relative_path, search_path); + CopyIncludedPaths(true, "/vendor/etc", relative_path, search_path); + CopyIncludedPaths(true, "/system/etc", relative_path, search_path); #else (void)relative_path; #endif @@ -447,7 +453,8 @@ static void GetExtensionProperties(const std::vector &extensio if (it != props.end()) { it->extensionVersion = std::max(it->extensionVersion, ext.extension_version); } else { - XrExtensionProperties prop = {XR_TYPE_EXTENSION_PROPERTIES}; + XrExtensionProperties prop{}; + prop.type = XR_TYPE_EXTENSION_PROPERTIES; strncpy(prop.extensionName, ext.name.c_str(), XR_MAX_EXTENSION_NAME_SIZE - 1); prop.extensionName[XR_MAX_EXTENSION_NAME_SIZE - 1] = '\0'; prop.extensionVersion = ext.extension_version; diff --git a/src/loader/manifest_file.hpp b/src/loader/manifest_file.hpp index de0aab65..46b842c6 100644 --- a/src/loader/manifest_file.hpp +++ b/src/loader/manifest_file.hpp @@ -1,4 +1,4 @@ -// Copyright (c) 2017 The Khronos Group Inc. +// Copyright (c) 2017-2023, The Khronos Group Inc. // Copyright (c) 2017 Valve Corporation // Copyright (c) 2017 LunarG, Inc. // diff --git a/src/loader/openxr-loader.expsym b/src/loader/openxr-loader.expsym new file mode 100644 index 00000000..98a14921 --- /dev/null +++ b/src/loader/openxr-loader.expsym @@ -0,0 +1,59 @@ +# Copyright (c) 2019-2023, The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 OR MIT + +_xrCreateInstance +_xrDestroyInstance +_xrEnumerateApiLayerProperties +_xrEnumerateInstanceExtensionProperties +_xrGetInstanceProcAddr +_xrGetInstanceProperties +_xrPollEvent +_xrResultToString +_xrStructureTypeToString +_xrGetSystem +_xrGetSystemProperties +_xrEnumerateEnvironmentBlendModes +_xrCreateSession +_xrDestroySession +_xrEnumerateReferenceSpaces +_xrCreateReferenceSpace +_xrGetReferenceSpaceBoundsRect +_xrCreateActionSpace +_xrLocateSpace +_xrDestroySpace +_xrEnumerateViewConfigurations +_xrGetViewConfigurationProperties +_xrEnumerateViewConfigurationViews +_xrEnumerateSwapchainFormats +_xrCreateSwapchain +_xrDestroySwapchain +_xrEnumerateSwapchainImages +_xrAcquireSwapchainImage +_xrWaitSwapchainImage +_xrReleaseSwapchainImage +_xrBeginSession +_xrEndSession +_xrRequestExitSession +_xrWaitFrame +_xrBeginFrame +_xrEndFrame +_xrLocateViews +_xrStringToPath +_xrPathToString +_xrCreateActionSet +_xrDestroyActionSet +_xrCreateAction +_xrDestroyAction +_xrSuggestInteractionProfileBindings +_xrAttachSessionActionSets +_xrGetCurrentInteractionProfile +_xrGetActionStateBoolean +_xrGetActionStateFloat +_xrGetActionStateVector2f +_xrGetActionStatePose +_xrSyncActions +_xrEnumerateBoundSourcesForAction +_xrGetInputSourceLocalizedName +_xrApplyHapticFeedback +_xrStopHapticFeedback diff --git a/src/loader/runtime_interface.cpp b/src/loader/runtime_interface.cpp index cb1a98af..d9ab86bb 100644 --- a/src/loader/runtime_interface.cpp +++ b/src/loader/runtime_interface.cpp @@ -430,7 +430,9 @@ void RuntimeInterface::GetInstanceExtensionProperties(std::vector 0) { - runtime_extension_properties.resize(count_output, {XR_TYPE_EXTENSION_PROPERTIES}); + XrExtensionProperties example_properties{}; + example_properties.type = XR_TYPE_EXTENSION_PROPERTIES; + runtime_extension_properties.resize(count_output, example_properties); count = count_output; rt_xrEnumerateInstanceExtensionProperties(nullptr, count, &count_output, runtime_extension_properties.data()); } diff --git a/src/scripts/utility_source_generator.py b/src/scripts/utility_source_generator.py index 1db955b5..c5728697 100755 --- a/src/scripts/utility_source_generator.py +++ b/src/scripts/utility_source_generator.py @@ -50,6 +50,7 @@ def beginFile(self, genOpts): if self.genOpts.filename == 'xr_generated_dispatch_table.h': preamble += '#pragma once\n' elif self.genOpts.filename == 'xr_generated_dispatch_table.c': + preamble += '#include \n' preamble += '#include "xr_generated_dispatch_table.h"\n' preamble += '#include "xr_dependencies.h"\n' diff --git a/src/scripts/validation_layer_generator.py b/src/scripts/validation_layer_generator.py index ac5698d1..c928838d 100644 --- a/src/scripts/validation_layer_generator.py +++ b/src/scripts/validation_layer_generator.py @@ -1437,9 +1437,10 @@ def outputParamMemberContents(self, is_command, struct_command_name, param_membe indent) if (param_member.is_handle or self.isEnumType(param_member.type) or (self.isStruct(param_member.type) and not self.isStructAlwaysValid(param_member.type))): - loop_string += self.writeIndent(indent) - loop_string += 'if (%s) {\n' % (prefixed_param_member_name) - indent = indent + 1 + if not param_member.is_static_array: + loop_string += self.writeIndent(indent) + loop_string += 'if (%s) {\n' % (prefixed_param_member_name) + indent = indent + 1 loop_string += self.writeIndent(indent) loop_string += 'for (uint32_t %s = 0; %s < %s; ++%s) {\n' % (loop_param_name, loop_param_name, @@ -1841,12 +1842,14 @@ def outputParamMemberContents(self, is_command, struct_command_name, param_membe param_member_contents += self.writeIndent(indent) param_member_contents += '}\n' if is_loop: - indent = indent - 2 if wrote_loop: - param_member_contents += self.writeIndent(indent + 1) - param_member_contents += '}\n' + indent = indent - 1 param_member_contents += self.writeIndent(indent) param_member_contents += '}\n' + if not param_member.is_static_array: + indent = indent - 1 + param_member_contents += self.writeIndent(indent) + param_member_contents += '}\n' return param_member_contents From ad7859d6796dbf1045911bb00d9fab8934870dc0 Mon Sep 17 00:00:00 2001 From: Ryan Pavlik Date: Thu, 27 Jul 2023 14:17:13 -0500 Subject: [PATCH 2/4] Changes from hotfix OpenXR 1.0.28.2 --- .reuse/dep5 | 1 + HOTFIX | 1 + changes/conformance/mr.2840.gl.md | 4 ++ .../platform_specific/AndroidManifest.xml | 6 ++- src/loader/AndroidManifest.xml | 6 ++- src/loader/AndroidManifest.xml.in | 4 +- src/loader/CMakeLists.txt | 8 ++-- src/loader/openxr_loader_for_android.pom | 11 +++-- src/version.cmake | 47 +++++++++++++------ src/version.gradle | 26 +++++++--- 10 files changed, 81 insertions(+), 33 deletions(-) create mode 100644 HOTFIX create mode 100644 changes/conformance/mr.2840.gl.md diff --git a/.reuse/dep5 b/.reuse/dep5 index 9091bf0b..cefd0765 100644 --- a/.reuse/dep5 +++ b/.reuse/dep5 @@ -4,6 +4,7 @@ Upstream-Contact: Ryan Pavlik Source: https://khronos.org/registry/OpenXR/ Files: changes/* + HOTFIX Copyright: 2019-2023, The Khronos Group Inc. License: CC-BY-4.0 diff --git a/HOTFIX b/HOTFIX new file mode 100644 index 00000000..0cfbf088 --- /dev/null +++ b/HOTFIX @@ -0,0 +1 @@ +2 diff --git a/changes/conformance/mr.2840.gl.md b/changes/conformance/mr.2840.gl.md new file mode 100644 index 00000000..9860ffb9 --- /dev/null +++ b/changes/conformance/mr.2840.gl.md @@ -0,0 +1,4 @@ +--- +- issue.2053.gl +--- +- Fix: Fix `` element contents in Android manifest. diff --git a/src/conformance/platform_specific/AndroidManifest.xml b/src/conformance/platform_specific/AndroidManifest.xml index 32f17c06..66737109 100644 --- a/src/conformance/platform_specific/AndroidManifest.xml +++ b/src/conformance/platform_specific/AndroidManifest.xml @@ -21,8 +21,10 @@ - + + + @@ -33,6 +35,8 @@ + + diff --git a/src/loader/AndroidManifest.xml b/src/loader/AndroidManifest.xml index c5f71935..2817b34a 100644 --- a/src/loader/AndroidManifest.xml +++ b/src/loader/AndroidManifest.xml @@ -14,8 +14,10 @@ - - + + + + diff --git a/src/loader/AndroidManifest.xml.in b/src/loader/AndroidManifest.xml.in index ac2a0693..9bd8a460 100644 --- a/src/loader/AndroidManifest.xml.in +++ b/src/loader/AndroidManifest.xml.in @@ -1,7 +1,7 @@ + android:versionName="${OPENXR_FULL_VERSION}${OPENXR_ANDROID_VERSION_SUFFIX}"> + + diff --git a/src/loader/CMakeLists.txt b/src/loader/CMakeLists.txt index 9225bba2..22047fbc 100644 --- a/src/loader/CMakeLists.txt +++ b/src/loader/CMakeLists.txt @@ -166,11 +166,11 @@ if(CMAKE_SYSTEM_NAME STREQUAL "Linux") target_compile_definitions(openxr_loader PRIVATE EXTRASYSCONFDIR="/etc") endif() - set_target_properties(openxr_loader PROPERTIES SOVERSION "${MAJOR}" VERSION "${MAJOR}.${MINOR}.${PATCH}") + set_target_properties(openxr_loader PROPERTIES SOVERSION "${MAJOR}" VERSION "${OPENXR_FULL_VERSION}") add_custom_target( libopenxr_loader.so.${MAJOR}.${MINOR} ALL - COMMAND ${CMAKE_COMMAND} -E create_symlink libopenxr_loader.so.${MAJOR}.${MINOR}.${PATCH} + COMMAND ${CMAKE_COMMAND} -E create_symlink libopenxr_loader.so.${OPENXR_FULL_VERSION} libopenxr_loader.so.${MAJOR}.${MINOR} ) elseif(WIN32) @@ -338,7 +338,7 @@ configure_package_config_file( ) write_basic_package_version_file( ${CMAKE_CURRENT_BINARY_DIR}/OpenXRConfigVersion.cmake - VERSION "${MAJOR}.${MINOR}.${PATCH}" + VERSION "${OPENXR_FULL_VERSION}" COMPATIBILITY SameMajorVersion ) install( @@ -401,5 +401,5 @@ elseif(ANDROID AND INSTALL_TO_ARCHITECTURE_PREFIXES) # This gets used directly by build-aar.sh configure_file(${CMAKE_CURRENT_SOURCE_DIR}/openxr_loader_for_android.pom - ${CMAKE_CURRENT_BINARY_DIR}/openxr_loader_for_android-${MAJOR}.${MINOR}.${PATCH}${OPENXR_ANDROID_VERSION_SUFFIX}.pom) + ${CMAKE_CURRENT_BINARY_DIR}/openxr_loader_for_android-${OPENXR_FULL_VERSION}${OPENXR_ANDROID_VERSION_SUFFIX}.pom) endif() diff --git a/src/loader/openxr_loader_for_android.pom b/src/loader/openxr_loader_for_android.pom index 130efb86..1b696ecc 100644 --- a/src/loader/openxr_loader_for_android.pom +++ b/src/loader/openxr_loader_for_android.pom @@ -1,13 +1,14 @@ - + org.khronos.openxr openxr_loader_for_android - @MAJOR@.@MINOR@.@PATCH@@OPENXR_ANDROID_VERSION_SUFFIX@ + @MAJOR@.@MINOR@.@PATCH@@OPENXR_SDK_HOTFIX_VERSION_SUFFIX@@OPENXR_ANDROID_VERSION_SUFFIX@ aar OpenXR Loader for Android The AAR for the OpenXR Loader as used on Android. diff --git a/src/version.cmake b/src/version.cmake index d863918d..5e64c70d 100644 --- a/src/version.cmake +++ b/src/version.cmake @@ -18,26 +18,45 @@ set(MAJOR "0") set(MINOR "0") set(PATCH "0") +set(OPENXR_SDK_HOTFIX_VERSION) if(EXISTS "${PROJECT_SOURCE_DIR}/specification/registry/xr.xml") - file(STRINGS ${PROJECT_SOURCE_DIR}/specification/registry/xr.xml lines REGEX "#define XR_CURRENT_API_VERSION") + file(STRINGS ${PROJECT_SOURCE_DIR}/specification/registry/xr.xml lines REGEX "#define XR_CURRENT_API_VERSION") else() - file(STRINGS ${PROJECT_SOURCE_DIR}/include/openxr/openxr.h lines REGEX "#define XR_CURRENT_API_VERSION") + file(STRINGS ${PROJECT_SOURCE_DIR}/include/openxr/openxr.h lines REGEX "#define XR_CURRENT_API_VERSION") endif() + list(LENGTH lines len) if(${len} EQUAL 1) list(GET lines 0 cur_line) - # Erase everything up to the open parentheses - string(REGEX REPLACE "^[^\(]+" "" VERSION_BEFORE_ERASED ${cur_line}) - # Erase everything after the close parentheses - string(REGEX REPLACE "[^\)]+$" "" VERSION_AFTER_ERASED ${VERSION_BEFORE_ERASED}) - # Erase the parentheses - string(REPLACE "(" "" VERSION_AFTER_ERASED2 ${VERSION_AFTER_ERASED}) - string(REPLACE ")" "" VERSION_AFTER_ERASED3 ${VERSION_AFTER_ERASED2}) - string(REPLACE " " "" VERSION_AFTER_ERASED4 ${VERSION_AFTER_ERASED3}) - string(REGEX REPLACE "^([0-9]+)\\,[0-9]+\\,[0-9]+" "\\1" MAJOR "${VERSION_AFTER_ERASED4}") - string(REGEX REPLACE "^[0-9]+\\,([0-9]+)\\,[0-9]+" "\\1" MINOR "${VERSION_AFTER_ERASED4}") - string(REGEX REPLACE "^[0-9]+\\,[0-9]+\\,([0-9]+)" "\\1" PATCH "${VERSION_AFTER_ERASED4}") + + # Grab just the stuff in the parentheses of XR_MAKE_VERSION( ), + # by replacing the whole line with the stuff in the parentheses + string(REGEX REPLACE "^.+\\(([^\)]+)\\).+$" "\\1" VERSION_WITH_WHITESPACE ${cur_line}) + + # Remove whitespace + string(REPLACE " " "" VERSION_NO_WHITESPACE ${VERSION_WITH_WHITESPACE}) + + # Grab components + string(REGEX REPLACE "^([0-9]+)\\,[0-9]+\\,[0-9]+" "\\1" MAJOR "${VERSION_NO_WHITESPACE}") + string(REGEX REPLACE "^[0-9]+\\,([0-9]+)\\,[0-9]+" "\\1" MINOR "${VERSION_NO_WHITESPACE}") + string(REGEX REPLACE "^[0-9]+\\,[0-9]+\\,([0-9]+)" "\\1" PATCH "${VERSION_NO_WHITESPACE}") else() - message(FATAL_ERROR "Unable to fetch major/minor version from registry or header") + message(FATAL_ERROR "Unable to fetch major/minor/patch version from registry or header") +endif() + +# Check for an SDK hotfix version indicator file. +if(EXISTS "${PROJECT_SOURCE_DIR}/HOTFIX") + file(STRINGS "${PROJECT_SOURCE_DIR}/HOTFIX" OPENXR_SDK_HOTFIX_VERSION) +endif() +if(OPENXR_SDK_HOTFIX_VERSION) + set(OPENXR_SDK_HOTFIX_VERSION_SUFFIX ".${OPENXR_SDK_HOTFIX_VERSION}") +endif() + +set(OPENXR_VERSION "${MAJOR}.${MINOR}.${PATCH}") +set(OPENXR_FULL_VERSION "${OPENXR_VERSION}${OPENXR_SDK_HOTFIX_VERSION_SUFFIX}") + +if(PROJECT_NAME STREQUAL "OPENXR") + # Only show the message if we aren't a subproject + message(STATUS "OpenXR ${OPENXR_FULL_VERSION}") endif() diff --git a/src/version.gradle b/src/version.gradle index 32cf7bf3..bd2aa658 100644 --- a/src/version.gradle +++ b/src/version.gradle @@ -6,6 +6,8 @@ class OpenXRVersion { public int major; public int minor; public int patch; + public boolean hasHotfix; + public int hotfix; /// Returns a version code for Android usage. int getVersionCode() { @@ -17,17 +19,22 @@ class OpenXRVersion { @Override String toString() { - "${major}.${minor}.${patch}" + if (hasHotfix) { + "${major}.${minor}.${patch}.${hotfix}" + } else { + "${major}.${minor}.${patch}" + } } /** * Parses the registry (as a text file) or the openxr.h header to get the version. * @param project Gradle project - * @param fn registry or header filename + * @param registryOrHeaderFile registry or header File + * @param hotfixFile HOTFIX File * @return version parsed */ - static def parseOpenXRVersionFile(Project project, def fn) { - def matches = project.file(fn).readLines().find { + static def parseOpenXRVersionFile(def registryOrHeaderFile, def hotfixFile) { + def matches = registryOrHeaderFile.readLines().find { it.contains('XR_CURRENT_API_VERSION') } =~ ~/\(([^\)]+)\)/ def components = matches[0][1].split(',').collect { it.replace(' ', '').trim() } @@ -35,6 +42,11 @@ class OpenXRVersion { version.major = Integer.parseInt(components[0]) version.minor = Integer.parseInt(components[1]) version.patch = Integer.parseInt(components[2]) + + if (hotfixFile.exists()) { + version.hasHotfix = true + version.hotfix = Integer.parseInt(hotfixFile.text.trim()) + } version } } @@ -42,8 +54,10 @@ class OpenXRVersion { class OpenXRVersionPlugin implements Plugin { void apply(Project project) { - project.ext.versionOpenXR = OpenXRVersion.parseOpenXRVersionFile(project, - "${project.repoRoot}/specification/registry/xr.xml") + def registryFile = project.file("${project.repoRoot}/specification/registry/xr.xml") + def hotfixFile = project.file("${project.repoRoot}/HOTFIX") + + project.ext.versionOpenXR = OpenXRVersion.parseOpenXRVersionFile(registryFile, hotfixFile) project.group = "org.khronos.openxr" if (project.file("${project.repoRoot}/SNAPSHOT").exists()) { From d3797bd714fae6f99c41f095196d5f3608240d64 Mon Sep 17 00:00:00 2001 From: Ryan Pavlik Date: Fri, 25 Aug 2023 16:59:08 -0500 Subject: [PATCH 3/4] Changes from OpenXR 1.0.29 GitOrigin-RevId: 07acb644ae0a5a42f891120d3172004cb5605ad7 --- .../shared/generate_windows_matrix_build.py | 2 +- .azure-pipelines/shared/install_vulkan.ps1 | 2 +- .../shared/organize_windows_artifacts.py | 2 +- .../shared/print_windows_artifact_names.py | 2 +- .azure-pipelines/shared/shared.py | 2 +- .git-blame-ignore-revs | 5 +- .../scripts/generate_windows_matrix_build.py | 2 +- .github/scripts/shared.py | 2 +- .gitignore | 1 + .reuse/dep5 | 27 +- HOTFIX | 1 - changes/conformance/mr.2387.gl.md | 4 + changes/conformance/mr.2672.gl.md | 1 + changes/conformance/mr.2685.gl.md | 4 + changes/conformance/mr.2704.gl.md | 1 + changes/conformance/mr.2729.gl.md | 4 + changes/conformance/mr.2737.gl.md | 3 + changes/conformance/mr.2756.gl.md | 3 + changes/conformance/mr.2766.gl.md | 2 +- changes/conformance/mr.2775.gl.md | 1 + changes/conformance/mr.2850.gl.md | 1 + maintainer-scripts/common.sh | 2 +- specification/Makefile | 18 +- specification/registry/xr.xml | 235 ++++- specification/scripts/genRef.py | 2 + specification/scripts/genxr.py | 2 + .../scripts/spec-macros/extension.rb | 15 +- specification/scripts/validitygenerator.py | 2 + specification/scripts/xrconventions.py | 2 +- src/CMakeLists.txt | 34 +- src/common/gfxwrapper_opengl.c | 1 + src/common/gfxwrapper_opengl.h | 56 +- src/common/platform_utils.hpp | 84 +- src/common/xr_dependencies.h | 6 +- src/common/xr_linear.h | 20 +- src/conformance/conformance_cli/main.cpp | 4 +- .../conformance_layer/CMakeLists.txt | 9 +- .../D3D11GraphicsValidator.cpp | 10 - .../conformance_layer/HandleState.h | 1 + .../conformance_test/test_Swapchains.cpp | 72 +- .../test_XR_EXT_debug_utils.cpp | 889 ++++++++++++++++-- .../test_XR_EXT_hand_tracking.cpp | 110 +-- .../test_XR_EXT_palm_pose.cpp | 155 ++- .../test_XR_MSFT_controller_model.cpp | 273 ++++++ src/conformance/framework/CMakeLists.txt | 5 +- src/conformance/framework/action_utils.cpp | 1 - .../framework/graphics_plugin_d3d11.cpp | 2 +- .../framework/graphics_plugin_d3d12.cpp | 2 +- .../framework/graphics_plugin_opengl.cpp | 428 ++------- .../framework/graphics_plugin_opengles.cpp | 619 +++++++----- .../framework/graphics_plugin_vulkan.cpp | 536 +++++------ .../framework/two_call_struct_metadata.h | 40 +- .../drawable/ic_launcher_foreground.xml | 4 +- .../mipmap-anydpi-v26/ic_launcher.xml | 4 +- .../mipmap-anydpi-v26/ic_launcher_round.xml | 4 +- .../android_resources/values/colors.xml | 2 +- src/conformance/utilities/CMakeLists.txt | 1 + src/conformance/utilities/d3d_common.cpp | 320 +++---- src/conformance/utilities/d3d_common.h | 5 +- .../utilities/swapchain_format_data.cpp | 329 +++++++ .../utilities/swapchain_format_data.h | 308 ++++++ .../utilities/swapchain_parameters.h | 6 + src/conformance/utilities/vulkan_utils.h | 22 +- src/loader/.gitignore | 2 +- src/loader/AndroidManifest.xml | 2 +- src/loader/AndroidManifest.xml.in | 2 +- src/loader/CMakeLists.txt | 4 +- src/loader/abi.json.license | 5 +- src/loader/loader.rc | 9 +- src/loader/loader_core.cpp | 2 +- src/loader/loader_instance.cpp | 2 +- src/loader/manifest_file.cpp | 77 +- src/loader/module.json.license | 5 +- src/loader/openxr-loader.def | 5 - src/loader/openxr.pc.in | 4 + src/loader/openxr.pc.in.license | 3 - src/loader/prefab.json.license | 5 +- src/loader/runtime_interface.cpp | 45 +- src/scripts/generate_api_layer_manifest.py | 2 +- src/scripts/loader_source_generator.py | 4 +- src/scripts/src_genxr.py | 54 +- src/scripts/utility_source_generator.py | 54 +- src/scripts/validation_layer_generator.py | 2 +- 83 files changed, 3492 insertions(+), 1508 deletions(-) delete mode 100644 HOTFIX create mode 100644 changes/conformance/mr.2387.gl.md create mode 100644 changes/conformance/mr.2672.gl.md create mode 100644 changes/conformance/mr.2685.gl.md create mode 100644 changes/conformance/mr.2775.gl.md create mode 100644 changes/conformance/mr.2850.gl.md create mode 100644 src/conformance/conformance_test/test_XR_MSFT_controller_model.cpp create mode 100644 src/conformance/utilities/swapchain_format_data.cpp create mode 100644 src/conformance/utilities/swapchain_format_data.h delete mode 100644 src/loader/openxr.pc.in.license diff --git a/.azure-pipelines/shared/generate_windows_matrix_build.py b/.azure-pipelines/shared/generate_windows_matrix_build.py index b74409b7..0f462201 100644 --- a/.azure-pipelines/shared/generate_windows_matrix_build.py +++ b/.azure-pipelines/shared/generate_windows_matrix_build.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 from itertools import product diff --git a/.azure-pipelines/shared/install_vulkan.ps1 b/.azure-pipelines/shared/install_vulkan.ps1 index aee09e94..c1f2ab59 100644 --- a/.azure-pipelines/shared/install_vulkan.ps1 +++ b/.azure-pipelines/shared/install_vulkan.ps1 @@ -1,4 +1,4 @@ -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 if (-not $env:VULKAN_SDK_VERSION) { diff --git a/.azure-pipelines/shared/organize_windows_artifacts.py b/.azure-pipelines/shared/organize_windows_artifacts.py index fc80354d..d59c53be 100644 --- a/.azure-pipelines/shared/organize_windows_artifacts.py +++ b/.azure-pipelines/shared/organize_windows_artifacts.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 from itertools import product diff --git a/.azure-pipelines/shared/print_windows_artifact_names.py b/.azure-pipelines/shared/print_windows_artifact_names.py index f7ab0da6..d58bfdfc 100644 --- a/.azure-pipelines/shared/print_windows_artifact_names.py +++ b/.azure-pipelines/shared/print_windows_artifact_names.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 from itertools import product diff --git a/.azure-pipelines/shared/shared.py b/.azure-pipelines/shared/shared.py index aae8cb70..4276880c 100644 --- a/.azure-pipelines/shared/shared.py +++ b/.azure-pipelines/shared/shared.py @@ -1,4 +1,4 @@ -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 import json diff --git a/.git-blame-ignore-revs b/.git-blame-ignore-revs index 5519b09d..9925f530 100644 --- a/.git-blame-ignore-revs +++ b/.git-blame-ignore-revs @@ -9,8 +9,11 @@ # and do try to make sure that the bulk change is made # **separate from a release commit** on all repos. # See -# for how to use this. +# for how to use this, probably something like: +# git config blame.ignoreRevsFile .git-blame-ignore-revs +## Old changes +767537d9523253de1615b01450a8b22c8e2cc6a2 ## 1.0.17 - Fix XML indentation diff --git a/.github/scripts/generate_windows_matrix_build.py b/.github/scripts/generate_windows_matrix_build.py index ef2d153b..f8538b87 100644 --- a/.github/scripts/generate_windows_matrix_build.py +++ b/.github/scripts/generate_windows_matrix_build.py @@ -1,5 +1,5 @@ #!/usr/bin/env python3 -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 from itertools import product diff --git a/.github/scripts/shared.py b/.github/scripts/shared.py index c9303515..b9ab2e95 100644 --- a/.github/scripts/shared.py +++ b/.github/scripts/shared.py @@ -1,4 +1,4 @@ -# Copyright (c) 2019 The Khronos Group Inc. +# Copyright (c) 2019-2023, The Khronos Group Inc. # SPDX-License-Identifier: Apache-2.0 import json diff --git a/.gitignore b/.gitignore index 13a23f16..fbafc27f 100644 --- a/.gitignore +++ b/.gitignore @@ -54,6 +54,7 @@ local.properties # Don't ignore these things !.*.license +!.artifact-bot.md !.appveyor.yml !.azure-pipelines/ !.azure-pipelines/nuget/NugetTemplate/build diff --git a/.reuse/dep5 b/.reuse/dep5 index cefd0765..3df6553a 100644 --- a/.reuse/dep5 +++ b/.reuse/dep5 @@ -3,7 +3,12 @@ Upstream-Name: OpenXR Upstream-Contact: Ryan Pavlik Source: https://khronos.org/registry/OpenXR/ -Files: changes/* +Files: changes/conformance/* + changes/registry/* + changes/sdk/* + changes/specification/* + changes/README.md + changes/template.md HOTFIX Copyright: 2019-2023, The Khronos Group Inc. License: CC-BY-4.0 @@ -30,25 +35,22 @@ Comment: Based on a Material Icons asset ("emoji-people") with added text Rasterized with Android Studio. Files: src/external/catch2/* -Copyright: Copyright (c) 2022 Two Blue Cubes Ltd. +Copyright: Copyright (c) 2023 Two Blue Cubes Ltd. License: BSL-1.0 -Comment: Unmodified, vendored copy of Catch2 3.1.1 +Comment: Unmodified, vendored copy of Catch2 v3.3.2 Files: src/external/jsoncpp/* Copyright: 2007-2010 Baptiste Lepilleur and The JsonCpp Authors License: MIT OR LicenseRef-jsoncpp-public-domain Comment: Unmodified, vendored copy of jsoncpp 1.9.5 -Files: src/external/tinygltf/* -Copyright: 2017 Syoyo Fujita, Aurélien Chatelain and many contributors +Files: src/external/tinygltf/README.md + src/external/tinygltf/tiny_gltf.cc + src/external/tinygltf/tiny_gltf.h +Copyright: 2015-Present, Syoyo Fujita, Aurélien Chatelain and many contributors License: MIT Comment: Unmodified, vendored copy of a subset of the tiny-gltf repo v2.8.9 -Files: src/external/tinygltf/json.hpp -Copyright: 2013-2017 Niels Lohmann -License: MIT -Comment: Unmodified, included in tiny-gltf repo. - Files: src/external/d3dx12/* Copyright: Copyright (c) Microsoft Corporation. License: MIT @@ -82,11 +84,6 @@ Copyright: 2020, Epic Games, Inc. License: CC-BY-4.0 Comment: In-line license comments requested, https://gitlab.khronos.org/openxr/openxr/-/issues/1422 -Files: specification/sources/chapters/extensions/oculus/oculus_android_session_state_enable.adoc -Copyright: 2019, Oculus VR, LLC. -License: CC-BY-4.0 -Comment: In-line license comments requested, https://gitlab.khronos.org/openxr/openxr/-/issues/1418 - Files: specification/sources/chapters/extensions/ext/ext_performance_settings.adoc specification/sources/chapters/extensions/ext/ext_thermal_query.adoc Copyright: 2017-2023, The Khronos Group Inc. diff --git a/HOTFIX b/HOTFIX deleted file mode 100644 index 0cfbf088..00000000 --- a/HOTFIX +++ /dev/null @@ -1 +0,0 @@ -2 diff --git a/changes/conformance/mr.2387.gl.md b/changes/conformance/mr.2387.gl.md new file mode 100644 index 00000000..988c6921 --- /dev/null +++ b/changes/conformance/mr.2387.gl.md @@ -0,0 +1,4 @@ +--- +- mr.2858.gl +--- +New test: Verify two-call idiom behavior of `XR_MSFT_controller_model` as well as handling of invalid model keys. diff --git a/changes/conformance/mr.2672.gl.md b/changes/conformance/mr.2672.gl.md new file mode 100644 index 00000000..a703b406 --- /dev/null +++ b/changes/conformance/mr.2672.gl.md @@ -0,0 +1 @@ +Improvement: Add non-interactive test for XR_EXT_palm_pose vendor extension. diff --git a/changes/conformance/mr.2685.gl.md b/changes/conformance/mr.2685.gl.md new file mode 100644 index 00000000..c77f7c89 --- /dev/null +++ b/changes/conformance/mr.2685.gl.md @@ -0,0 +1,4 @@ +--- +- issue.1978.gl +--- +Improvement: Refactor and standardize creation of swapchain image format tables, fixing some Vulkan invalid usage. diff --git a/changes/conformance/mr.2704.gl.md b/changes/conformance/mr.2704.gl.md index 71b2bf72..a792439d 100644 --- a/changes/conformance/mr.2704.gl.md +++ b/changes/conformance/mr.2704.gl.md @@ -2,5 +2,6 @@ - mr.2784.gl - mr.2785.gl - mr.2808.gl +- mr.2809.gl --- Improvement: Cleanup and code quality work. diff --git a/changes/conformance/mr.2729.gl.md b/changes/conformance/mr.2729.gl.md index 95e4d03c..7e44ad1a 100644 --- a/changes/conformance/mr.2729.gl.md +++ b/changes/conformance/mr.2729.gl.md @@ -1 +1,5 @@ +--- +- mr.2795.gl +- mr.2858.gl +--- Improvement: Add joint query to non-interactive test for `XR_EXT_hand_tracking`. diff --git a/changes/conformance/mr.2737.gl.md b/changes/conformance/mr.2737.gl.md index 869cf7aa..1f289d4d 100644 --- a/changes/conformance/mr.2737.gl.md +++ b/changes/conformance/mr.2737.gl.md @@ -1 +1,4 @@ +--- +- issue.1932.gl +--- Improvement: Migrate more tests to use the `SKIP` macro when appropriate. diff --git a/changes/conformance/mr.2756.gl.md b/changes/conformance/mr.2756.gl.md index fef78530..ddfb77dc 100644 --- a/changes/conformance/mr.2756.gl.md +++ b/changes/conformance/mr.2756.gl.md @@ -1 +1,4 @@ +--- +- issue.1387.gl +--- Fix: Do not use Catch2 assertion macros in graphics plugin methods that may be called before the first test case execution begins. diff --git a/changes/conformance/mr.2766.gl.md b/changes/conformance/mr.2766.gl.md index 7a0fd1ec..944b119d 100644 --- a/changes/conformance/mr.2766.gl.md +++ b/changes/conformance/mr.2766.gl.md @@ -1 +1 @@ -Fix spelling. +Fix: spelling. diff --git a/changes/conformance/mr.2775.gl.md b/changes/conformance/mr.2775.gl.md new file mode 100644 index 00000000..c2164b35 --- /dev/null +++ b/changes/conformance/mr.2775.gl.md @@ -0,0 +1 @@ +Improvement: Add additional tests for XR_EXT_debug_utils based on the test app loader_test. diff --git a/changes/conformance/mr.2850.gl.md b/changes/conformance/mr.2850.gl.md new file mode 100644 index 00000000..85701ee8 --- /dev/null +++ b/changes/conformance/mr.2850.gl.md @@ -0,0 +1 @@ +Fix building CTS with mingw compiler. diff --git a/maintainer-scripts/common.sh b/maintainer-scripts/common.sh index 37043223..065300c2 100644 --- a/maintainer-scripts/common.sh +++ b/maintainer-scripts/common.sh @@ -34,7 +34,7 @@ makeSubset() { } -COMMON_FILES=".gitignore .gitattributes .git-blame-ignore-revs CODE_OF_CONDUCT.md LICENSES .reuse .editorconfig" +COMMON_FILES=".gitignore .gitattributes .git-blame-ignore-revs CODE_OF_CONDUCT.md LICENSES .reuse .editorconfig HOTFIX" export COMMON_FILES COMMON_EXCLUDE_PATTERN="KhronosExperimental" export COMMON_EXCLUDE_PATTERN diff --git a/specification/Makefile b/specification/Makefile index 5b69343d..fcf69338 100644 --- a/specification/Makefile +++ b/specification/Makefile @@ -32,7 +32,7 @@ ifneq (,$(strip $(VERY_STRICT))) ASCIIDOC := $(ASCIIDOC) --failure-level WARN endif -SPECREVISION = 1.0.28 +SPECREVISION = 1.0.29 REVISION_COMPONENTS = $(subst ., ,$(SPECREVISION)) MAJORMINORVER = $(word 1,$(REVISION_COMPONENTS)).$(word 2,$(REVISION_COMPONENTS)) @@ -384,26 +384,14 @@ BACKEND_ARGS=--backend html5 $(ASCIIDOCTOR_TARGETS): $(ECHO) "[asciidoctor] $(SPECSRC) -> $(call MAKE_RELATIVE,$@)" $(QUIET)$(MKDIR) "$(@D)" - $(QUIET)if [ $$(uname -s | cut -c 1-6) == "CYGWIN" ]; then \ - OUTSPEC_DOS=$$(cygpath -w $@) ;\ - SPECSRC_DOS=$$(cygpath -w $(SPECSRC)) ;\ - ATTRIBOPTS_DOS='$(ATTRIBOPTS)' ;\ - ADOCOPTS_DOS="--doctype book -a data-uri -r $$(cygpath -w $(CURDIR)/scripts/spec-macros.rb) $$ATTRIBOPTS_DOS" ;\ - BATCH_FILE=$$(cygpath -w $$(mktemp)).bat ;\ - echo $(ASCIIDOC) $$ADOCOPTS_DOS $(BACKEND_ARGS) --out-file $$OUTSPEC_DOS $$SPECSRC_DOS > $$BATCH_FILE ;\ - CMD /C $$BATCH_FILE ;\ - rm -f $$BATCH_FILE ;\ - else \ - $(ASCIIDOC) $(ADOCOPTS) $(BACKEND_ARGS) --out-file $@ $(SPECSRC) 2>&1 | tee $(LOGFILE) ;\ - if [ -s $(LOGFILE) ]; then \ + $(QUIET)$(ASCIIDOC) $(ADOCOPTS) $(BACKEND_ARGS) --out-file $@ $(SPECSRC) 2>&1 | tee $(LOGFILE) + $(QUIET)if [ -s $(LOGFILE) ]; then \ echo "Failure: $(LOGFILE) exists and is not empty!"; \ false; \ else \ rm $(LOGFILE); \ - fi; \ fi $(POSTPROCESS) -# TODO: Postprocess step(s) may have broken the Cygwin build. Not sure this matters anymore with WSL, though. ################################################ diff --git a/specification/registry/xr.xml b/specification/registry/xr.xml index 0fccc519..add5a2c7 100644 --- a/specification/registry/xr.xml +++ b/specification/registry/xr.xml @@ -62,6 +62,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + @@ -131,7 +132,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. updates them automatically by processing a line at a time. --> // OpenXR current version number. -#define XR_CURRENT_API_VERSION XR_MAKE_VERSION(1, 0, 28) +#define XR_CURRENT_API_VERSION XR_MAKE_VERSION(1, 0, 29) typedef XrFlags64 XrPerformanceMetricsCounterFlagsMETA; + + typedef XrFlags64 XrPassthroughPreferenceFlagsMETA; + typedef XrFlags64 XrFoveationDynamicFlagsHTC; @@ -544,6 +548,9 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + @@ -1465,11 +1472,14 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrBlendFactorFB dstFactorAlpha + + typedef void *(*PFN_xrEglGetProcAddressMNDX)(const char *name); + XrStructureType type const void* next - PFNEGLGETPROCADDRESSPROC getProcAddress + PFN_xrEglGetProcAddressMNDX getProcAddress EGLDisplay display EGLConfig config EGLContext context @@ -2859,6 +2869,13 @@ maintained in the default branch of the Khronos OpenXR GitHub project. float floatValue + + + XrStructureType type + const void* next + XrPassthroughPreferenceFlagsMETA flags + + XrStructureType type @@ -4104,6 +4121,11 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + @@ -5589,6 +5611,13 @@ maintained in the default branch of the Khronos OpenXR GitHub project. XrPerformanceMetricsCounterMETA* counter + + + XrResult xrGetPassthroughPreferencesMETA + XrSession session + XrPassthroughPreferencesMETA* preferences + + XrResult xrApplyFoveationHTC @@ -6063,6 +6092,10 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + @@ -7720,7 +7753,7 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + @@ -9116,10 +9149,18 @@ maintained in the default branch of the Khronos OpenXR GitHub project. - + - - + + + + + + + + + + @@ -11343,6 +11384,188 @@ maintained in the default branch of the Khronos OpenXR GitHub project. + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/specification/scripts/genRef.py b/specification/scripts/genRef.py index b5dfa189..5a72a6ac 100644 --- a/specification/scripts/genRef.py +++ b/specification/scripts/genRef.py @@ -75,10 +75,12 @@ def printCopyrightSourceComments(fp): Writes an asciidoc comment block, which copyrights the source file.""" + # REUSE-IgnoreStart print('// Copyright 2014-2023 The Khronos Group Inc.', file=fp) print('//', file=fp) # This works around constraints of the 'reuse' tool print('// SPDX' + '-License-Identifier: CC-BY-4.0', file=fp) + # REUSE-IgnoreEnd print('', file=fp) diff --git a/specification/scripts/genxr.py b/specification/scripts/genxr.py index 2ddbda1e..875de43a 100755 --- a/specification/scripts/genxr.py +++ b/specification/scripts/genxr.py @@ -105,6 +105,7 @@ def makeGenOpts(args): emitExtensionsPat = makeREstring(emitExtensions, allExtensions) featuresPat = makeREstring(features, allFeatures) + # REUSE-IgnoreStart # Copyright text prefixing all headers (list of strings). prefixStrings = [ '/*', @@ -115,6 +116,7 @@ def makeGenOpts(args): '*/', '' ] + # REUSE-IgnoreEnd # Text specific to OpenXR headers xrPrefixStrings = [ diff --git a/specification/scripts/spec-macros/extension.rb b/specification/scripts/spec-macros/extension.rb index 08f953ca..be76084c 100644 --- a/specification/scripts/spec-macros/extension.rb +++ b/specification/scripts/spec-macros/extension.rb @@ -270,13 +270,14 @@ class BasetypeInlineMacro < LinkInlineMacroBase named :basetype match /basetype:(\w+)/ - # def process parent, target, attributes - # if parent.document.attributes['cross-file-links'] - # return Inline.new(parent, :anchor, target, :type => :link, :target => (target + '.html')) - # else - # return Inline.new(parent, :anchor, '' + target + '', :type => :xref, :target => ('#' + target), :attributes => {'fragment' => target, 'refid' => target}) - # end - # end + def process parent, target, attributes + node = super parent, target, attributes + if parent.document.attributes['cross-file-links'] + node + else + create_inline parent, :quoted, %(#{node.convert}) + end + end def exists? target $apiNames.basetypes.has_key? target diff --git a/specification/scripts/validitygenerator.py b/specification/scripts/validitygenerator.py index 495ae026..b121f705 100644 --- a/specification/scripts/validitygenerator.py +++ b/specification/scripts/validitygenerator.py @@ -1024,6 +1024,8 @@ def makeStructureTypeValidity(self, structname): child_structs = self.keepOnlyRequired(self.struct_children.get(structname, []), self.registry.typedict) if child_structs: + if values: + print('The struct: {} has children, it may not have a "values" attribute itself.'.format(structname)) assert(not values) if len(child_structs) > 1: entry += 'one of the following XrStructureType values: ' diff --git a/specification/scripts/xrconventions.py b/specification/scripts/xrconventions.py index 6fe3d307..2d59c0c5 100644 --- a/specification/scripts/xrconventions.py +++ b/specification/scripts/xrconventions.py @@ -22,7 +22,7 @@ (?POpenGL(ES)?)| # OpenGL and OpenGLES as words (?P[0-9]D)| # Things like 2D are words (?P # Normal-ish words, which are.... - ([A-Z]([a-z]+([0-9](?!D))*)+)| # Capital letter followed by at least one lowercase letter, possibly ending in some digits as long as the digits aren't followed by "D" + ([A-Z]([a-z]+([0-9](?!D[A-Z]{1}))*)+)| # Capital letter followed by at least one lowercase letter, possibly ending in some digits as long as the digits aren't followed by a "D" then another word ([A-Z][A-Z0-9]+(?![a-z])) # Or, all-caps letter and digit mix starting with a letter, excluding the last capital before some lowercase )''', re.VERBOSE) diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index f8af1a0d..b952079a 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -305,12 +305,10 @@ function(compile_glsl run_target_name) endfunction() -if(WIN32) +# Not available in MinGW +if(MSVC) add_definitions(-DXR_USE_GRAPHICS_API_D3D11) - if(MSVC) - # Not available in MinGW right now - add_definitions(-DXR_USE_GRAPHICS_API_D3D12) - endif() + add_definitions(-DXR_USE_GRAPHICS_API_D3D12) endif() # Check for the existence of the secure_getenv or __secure_getenv commands @@ -408,16 +406,34 @@ set(GENERATED_OUTPUT) set(GENERATED_DEPENDS) run_xr_xml_generate(utility_source_generator.py xr_generated_dispatch_table.h) run_xr_xml_generate(utility_source_generator.py xr_generated_dispatch_table.c) -if(GENERATED_DEPENDS) - add_custom_target(xr_global_generated_files DEPENDS ${GENERATED_DEPENDS}) +set(COMMON_GENERATED_OUTPUT ${GENERATED_OUTPUT}) +set(COMMON_GENERATED_DEPENDS ${GENERATED_DEPENDS}) +unset(GENERATED_OUTPUT) +unset(GENERATED_DEPENDS) + +run_xr_xml_generate(utility_source_generator.py xr_generated_dispatch_table_core.h) +run_xr_xml_generate(utility_source_generator.py xr_generated_dispatch_table_core.c) +set(LOADER_GENERATED_OUTPUT ${GENERATED_OUTPUT}) +set(LOADER_GENERATED_DEPENDS ${GENERATED_DEPENDS}) +unset(GENERATED_OUTPUT) +unset(GENERATED_DEPENDS) + +if(LOADER_GENERATED_DEPENDS) + add_custom_target(xr_global_generated_files DEPENDS ${LOADER_GENERATED_DEPENDS}) else() add_custom_target(xr_global_generated_files) endif() set_target_properties(xr_global_generated_files PROPERTIES FOLDER ${CODEGEN_FOLDER}) -set(COMMON_GENERATED_OUTPUT ${GENERATED_OUTPUT}) -set(COMMON_GENERATED_DEPENDS ${GENERATED_DEPENDS}) +if(COMMON_GENERATED_DEPENDS) + add_custom_target(xr_common_generated_files DEPENDS ${COMMON_GENERATED_DEPENDS}) +else() + add_custom_target(xr_common_generated_files) +endif() + +set_target_properties(xr_common_generated_files PROPERTIES FOLDER ${CODEGEN_FOLDER}) + if(NOT MSVC) include(CheckCXXCompilerFlag) include(CheckCCompilerFlag) diff --git a/src/common/gfxwrapper_opengl.c b/src/common/gfxwrapper_opengl.c index 8b5c0899..413dc514 100644 --- a/src/common/gfxwrapper_opengl.c +++ b/src/common/gfxwrapper_opengl.c @@ -1,4 +1,5 @@ /* +Copyright (c) 2017-2023, The Khronos Group Inc. Copyright (c) 2016 Oculus VR, LLC. Portions of macOS, iOS, functionality copyright (c) 2016 The Brenwill Workshop Ltd. diff --git a/src/common/gfxwrapper_opengl.h b/src/common/gfxwrapper_opengl.h index a4c245d3..6d8e1518 100644 --- a/src/common/gfxwrapper_opengl.h +++ b/src/common/gfxwrapper_opengl.h @@ -1,35 +1,33 @@ +// Copyright (c) 2017-2023, The Khronos Group Inc. +// Copyright (c) 2016, Oculus VR, LLC. +// Portions of macOS, iOS, functionality copyright (c) 2016, The Brenwill Workshop Ltd. +// +// SPDX-License-Identifier: Apache-2.0 +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +/* REUSE-IgnoreStart */ +/* The following has copyright notices that duplicate the header above */ + /* ================================================================================================ -Description : Convenient wrapper for the OpenGL API. -Author : J.M.P. van Waveren -Date : 12/21/2014 -Language : C99 -Format : Real tabs with the tab size equal to 4 spaces. -Copyright : Copyright (c) 2016 Oculus VR, LLC. All Rights reserved. - : Portions copyright (c) 2016 The Brenwill Workshop Ltd. All Rights reserved. - - -LICENSE -======= - -Copyright (c) 2016 Oculus VR, LLC. -Portions of macOS, iOS, functionality copyright (c) 2016 The Brenwill Workshop Ltd. - -SPDX-License-Identifier: Apache-2.0 - -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. - +Description : Convenient wrapper for the OpenGL API. +Orig. Author : J.M.P. van Waveren +Orig. Date : 12/21/2014 +Language : C99 +Copyright : Copyright (c) 2016 Oculus VR, LLC. All Rights reserved. + : Portions copyright (c) 2016 The Brenwill Workshop Ltd. All Rights reserved. IMPLEMENTATION ============== diff --git a/src/common/platform_utils.hpp b/src/common/platform_utils.hpp index 219d1978..883baef8 100644 --- a/src/common/platform_utils.hpp +++ b/src/common/platform_utils.hpp @@ -37,6 +37,44 @@ #include "common_config.h" #endif // OPENXR_HAVE_COMMON_CONFIG +#if defined(__x86_64__) && defined(__ILP32__) +#define XR_ARCH_ABI "x32" +#elif defined(_M_X64) || defined(__x86_64__) +#define XR_ARCH_ABI "x86_64" +#elif defined(_M_IX86) || defined(__i386__) || defined(_X86_) +#define XR_ARCH_ABI "i686" +#elif (defined(__aarch64__) && defined(__LP64__)) || defined(_M_ARM64) +#define XR_ARCH_ABI "aarch64" +#elif (defined(__ARM_ARCH) && __ARM_ARCH >= 7 && (defined(__ARM_PCS_VFP) || defined(__ANDROID__))) || defined(_M_ARM) +#define XR_ARCH_ABI "armv7a-vfp" +#elif defined(__ARM_ARCH_5TE__) +#define XR_ARCH_ABI "armv5te" +#elif defined(__mips64) +#define XR_ARCH_ABI "mips64" +#elif defined(__mips) +#define XR_ARCH_ABI "mips" +#elif defined(__powerpc64__) && __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__ +#define XR_ARCH_ABI "ppc64" +#elif defined(__powerpc__) && __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__ +#define XR_ARCH_ABI "ppc64el" +#elif defined(__s390x__) || defined(__zarch__) +#define XR_ARCH_ABI "s390x" +#elif defined(__hppa__) +#define XR_ARCH_ABI "hppa" +#elif defined(__alpha__) +#define XR_ARCH_ABI "alpha" +#elif defined(__ia64__) || defined(_M_IA64) +#define XR_ARCH_ABI "ia64" +#elif defined(__m68k__) +#define XR_ARCH_ABI "m68k" +#elif defined(__riscv_xlen) && (__riscv_xlen == 64) +#define XR_ARCH_ABI "riscv64" +#elif defined(__sparc__) && defined(__arch64__) +#define XR_ARCH_ABI "sparc64" +#else +#error "No architecture string known!" +#endif + // Consumers of this file must ensure this function is implemented. For example, the loader will implement this function so that it // can route messages through the loader's logging system. void LogPlatformUtilsError(const std::string& message); @@ -47,6 +85,7 @@ void LogPlatformUtilsError(const std::string& message); #include #include #include +#include namespace detail { @@ -72,6 +111,31 @@ static inline char* ImplGetSecureEnv(const char* name) { } // namespace detail #endif // defined(XR_OS_LINUX) || defined(XR_OS_APPLE) + +#if defined(XR_OS_ANDROID) || defined(XR_OS_APPLE) + +#include + +namespace detail { + +static inline bool ImplTryRuntimeFilename(const char* rt_dir_prefix, uint16_t major_version, std::string& file_name) { + auto decorated_path = rt_dir_prefix + std::to_string(major_version) + "/active_runtime." XR_ARCH_ABI ".json"; + auto undecorated_path = rt_dir_prefix + std::to_string(major_version) + "/active_runtime.json"; + + struct stat buf {}; + if (0 == stat(decorated_path.c_str(), &buf)) { + file_name = decorated_path; + return true; + } + if (0 == stat(undecorated_path.c_str(), &buf)) { + file_name = undecorated_path; + return true; + } + return false; +} + +} // namespace detail +#endif // defined(XR_OS_ANDROID) || defined(XR_OS_APPLE) #if defined(XR_OS_LINUX) static inline std::string PlatformUtilsGetEnv(const char* name) { @@ -130,15 +194,8 @@ static inline bool PlatformUtilsSetEnv(const char* name, const char* value) { return (result == 0); } -// Prefix for the Apple global runtime JSON file name -static const std::string rt_dir_prefix = "/usr/local/share/openxr/"; -static const std::string rt_filename = "/active_runtime.json"; - static inline bool PlatformGetGlobalRuntimeFileName(uint16_t major_version, std::string& file_name) { - file_name = rt_dir_prefix; - file_name += std::to_string(major_version); - file_name += rt_filename; - return true; + return detail::ImplTryRuntimeFilename("/usr/local/share/openxr/", major_version, file_name); } #elif defined(XR_OS_WINDOWS) @@ -311,22 +368,19 @@ static inline bool PlatformUtilsSetEnv(const char* /* name */, const char* /* va return false; } -#include - // Intended to be only used as a fallback on Android, with a more open, "native" technique used in most cases static inline bool PlatformGetGlobalRuntimeFileName(uint16_t major_version, std::string& file_name) { // Prefix for the runtime JSON file name static const char* rt_dir_prefixes[] = {"/product", "/odm", "/oem", "/vendor", "/system"}; - static const std::string rt_filename = "/active_runtime.json"; + static const std::string subdir = "/etc/openxr/"; for (const auto prefix : rt_dir_prefixes) { - auto path = prefix + subdir + std::to_string(major_version) + rt_filename; - struct stat buf; - if (0 == stat(path.c_str(), &buf)) { - file_name = path; + const std::string rt_dir_prefix = prefix + subdir; + if (detail::ImplTryRuntimeFilename(rt_dir_prefix.c_str(), major_version, file_name)) { return true; } } + return false; } #else // Not Linux, Apple, nor Windows diff --git a/src/common/xr_dependencies.h b/src/common/xr_dependencies.h index 5c7bd047..6c9cf2d0 100644 --- a/src/common/xr_dependencies.h +++ b/src/common/xr_dependencies.h @@ -76,7 +76,10 @@ #include "wayland-client.h" #endif // XR_USE_PLATFORM_WAYLAND -#ifdef XR_USE_GRAPHICS_API_OPENGL +#ifdef XR_USE_PLATFORM_EGL +#include +#endif // XR_USE_PLATFORM_EGL + #if defined(XR_USE_PLATFORM_XLIB) || defined(XR_USE_PLATFORM_XCB) #ifdef Success #undef Success @@ -90,4 +93,3 @@ #undef None #endif // None #endif // defined(XR_USE_PLATFORM_XLIB) || defined(XR_USE_PLATFORM_XCB) -#endif // XR_USE_GRAPHICS_API_OPENGL diff --git a/src/common/xr_linear.h b/src/common/xr_linear.h index 5b0da645..ce65f8dd 100644 --- a/src/common/xr_linear.h +++ b/src/common/xr_linear.h @@ -1,5 +1,5 @@ -// Copyright (c) 2017 The Khronos Group Inc. -// Copyright (c) 2016 Oculus VR, LLC. +// Copyright (c) 2017-2023, The Khronos Group Inc. +// Copyright (c) 2016, Oculus VR, LLC. // // SPDX-License-Identifier: Apache-2.0 // @@ -23,15 +23,17 @@ #include +/* REUSE-IgnoreStart */ +/* The following has copyright notices that duplicate the header above */ + /* ================================================================================================ -Description : Vector, matrix and quaternion math. -Author : J.M.P. van Waveren -Date : 12/10/2016 -Language : C99 -Format : Indent 4 spaces - no tabs. -Copyright : Copyright (c) 2016 Oculus VR, LLC. All Rights reserved. +Description : Vector, matrix and quaternion math. +Orig. Author : J.M.P. van Waveren +Orig. Date : 12/10/2016 +Language : C99 +Copyright : Copyright (c) 2016 Oculus VR, LLC. All Rights reserved. DESCRIPTION @@ -145,6 +147,8 @@ inline static float XrRcpSqrt(const float x) { return rcp; } +inline static float XrVector2f_Length(const XrVector2f* v) { return sqrtf(v->x * v->x + v->y * v->y); } + inline static void XrVector3f_Set(XrVector3f* v, const float value) { v->x = value; v->y = value; diff --git a/src/conformance/conformance_cli/main.cpp b/src/conformance/conformance_cli/main.cpp index 5358c733..2b8dba84 100644 --- a/src/conformance/conformance_cli/main.cpp +++ b/src/conformance/conformance_cli/main.cpp @@ -14,9 +14,9 @@ // Favor the high performance NVIDIA or AMD GPUs extern "C" { // http://developer.download.nvidia.com/devzone/devcenter/gamegraphics/files/OptimusRenderingPolicies.pdf -_declspec(dllexport) DWORD NvOptimusEnablement = 0x00000001; +__declspec(dllexport) DWORD NvOptimusEnablement = 0x00000001; // https://gpuopen.com/learn/amdpowerxpressrequesthighperformance/ -_declspec(dllexport) DWORD AmdPowerXpressRequestHighPerformance = 0x00000001; +__declspec(dllexport) DWORD AmdPowerXpressRequestHighPerformance = 0x00000001; } #endif // defined(_WIN32) diff --git a/src/conformance/conformance_layer/CMakeLists.txt b/src/conformance/conformance_layer/CMakeLists.txt index 01fa7e22..bf7c9b19 100644 --- a/src/conformance/conformance_layer/CMakeLists.txt +++ b/src/conformance/conformance_layer/CMakeLists.txt @@ -25,7 +25,6 @@ run_xr_xml_generate(conformance_layer_generator.py gen_dispatch.cpp run_xr_xml_generate(conformance_layer_generator.py gen_dispatch.h ${CMAKE_CURRENT_SOURCE_DIR}/../../scripts/template_gen_dispatch.h) - add_library(XrApiLayer_runtime_conformance MODULE ${LOCAL_SOURCE} ${LOCAL_HEADERS} @@ -43,7 +42,7 @@ source_group("Headers" FILES ${LOCAL_HEADERS}) add_dependencies(XrApiLayer_runtime_conformance generate_openxr_header - xr_global_generated_files + xr_common_generated_files ) target_include_directories(XrApiLayer_runtime_conformance @@ -55,15 +54,15 @@ target_include_directories(XrApiLayer_runtime_conformance PRIVATE ${PROJECT_BINARY_DIR}/include #for common_config.h: PRIVATE ${PROJECT_BINARY_DIR}/src +#for xr_generated_dispatch_table_all.h: + PRIVATE ${CMAKE_CURRENT_BINARY_DIR}/../../api_layers ) target_link_libraries(XrApiLayer_runtime_conformance Threads::Threads) if(MSVC) - # Right now can't build this on MinGW because of directxcolors, etc. + # Right now can't build this on MinGW because of directxcolors, directxmath, etc. target_link_libraries(XrApiLayer_runtime_conformance d3d11 d3d12 d3dcompiler dxgi) -else() - target_compile_definitions(XrApiLayer_runtime_conformance PRIVATE MISSING_DIRECTX_COLORS) endif() # Flag generated files diff --git a/src/conformance/conformance_layer/D3D11GraphicsValidator.cpp b/src/conformance/conformance_layer/D3D11GraphicsValidator.cpp index ff1a26eb..131f8cb3 100644 --- a/src/conformance/conformance_layer/D3D11GraphicsValidator.cpp +++ b/src/conformance/conformance_layer/D3D11GraphicsValidator.cpp @@ -24,7 +24,6 @@ namespace Conformance { -#if !defined(MISSING_DIRECTX_COLORS) // Map type to typeless. const std::unordered_map g_typelessMap{ {DXGI_FORMAT_R32G32B32A32_FLOAT, DXGI_FORMAT_R32G32B32A32_TYPELESS}, @@ -133,7 +132,6 @@ namespace Conformance DXGI_FORMAT_V408 = 132, */ }; -#endif struct D3D11GraphicsValidator : IGraphicsValidator { @@ -148,12 +146,8 @@ namespace Conformance void ValidateSwapchainImageStructs(ConformanceHooksBase* conformanceHooks, uint64_t swapchainFormat, uint32_t count, XrSwapchainImageBaseHeader* images) const override { -#if !defined(MISSING_DIRECTX_COLORS) const auto it = g_typelessMap.find((DXGI_FORMAT)swapchainFormat); const DXGI_FORMAT expectedTextureFormat = it == g_typelessMap.end() ? (DXGI_FORMAT)swapchainFormat : it->second; -#else - (void)swapchainFormat; -#endif const XrSwapchainImageD3D11KHR* const d3d11Images = reinterpret_cast(images); for (uint32_t i = 0; i < count; i++) { @@ -166,16 +160,12 @@ namespace Conformance D3D11_TEXTURE2D_DESC desc; d3d11Images[i].texture->GetDesc(&desc); -#if !defined(MISSING_DIRECTX_COLORS) if (desc.Format != expectedTextureFormat) { conformanceHooks->ConformanceFailure( XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, "xrEnumerateSwapchainImages", "xrEnumerateSwapchainImages failed: ID3D11Texture2D format is not expected format %d: Swapchain : %d", expectedTextureFormat, desc.Format); } -#else - (void)desc; -#endif // TODO: Confirm texture is for same device(context)? } diff --git a/src/conformance/conformance_layer/HandleState.h b/src/conformance/conformance_layer/HandleState.h index 2d774fdd..88dfd5a5 100644 --- a/src/conformance/conformance_layer/HandleState.h +++ b/src/conformance/conformance_layer/HandleState.h @@ -23,6 +23,7 @@ #include #include +#include #include struct ICustomHandleState diff --git a/src/conformance/conformance_test/test_Swapchains.cpp b/src/conformance/conformance_test/test_Swapchains.cpp index 6ce6484b..db0b5a93 100644 --- a/src/conformance/conformance_test/test_Swapchains.cpp +++ b/src/conformance/conformance_test/test_Swapchains.cpp @@ -31,6 +31,7 @@ #include +#include #include XRC_DISABLE_MSVC_WARNING(4505) // unreferenced local function has been removed @@ -310,35 +311,34 @@ namespace Conformance { std::vector> ret; const auto defaultCreateInfo = MakeDefaultSwapchainCreateInfo(imageFormat, tp); - { + + auto emplaceCaseWithDimensions = [&](uint32_t width, uint32_t height, const char* message) { auto createInfo = defaultCreateInfo; - // Smallest compressed texture size is 4x4, use 8x8 to allow for future formats - createInfo.width = 8; - createInfo.height = 8; - ret.emplace_back("Very small texture, but larger than minimum compressed texture size", createInfo); - } + if (width != createInfo.width || height != createInfo.height) { - for (const auto& viewConfigView : session.viewConfigurationViewVector) { - { - auto createInfo = defaultCreateInfo; - createInfo.width = viewConfigView.recommendedImageRectWidth; - createInfo.height = viewConfigView.recommendedImageRectHeight; - ret.emplace_back("Recommended image size for view", createInfo); - } - { - auto createInfo = defaultCreateInfo; - createInfo.width = viewConfigView.maxImageRectWidth; - createInfo.height = viewConfigView.maxImageRectHeight; - ret.emplace_back("Max image size for view", createInfo); + createInfo.width = width; + createInfo.height = height; + ret.emplace_back(message, createInfo); } + }; + + // Smallest compressed texture size is 4x4, use 8x8 to allow for future formats + emplaceCaseWithDimensions(8, 8, "Very small texture, but larger than minimum compressed texture size"); + + for (const auto& viewConfigView : session.viewConfigurationViewVector) { + emplaceCaseWithDimensions(viewConfigView.recommendedImageRectWidth, viewConfigView.recommendedImageRectHeight, + "Recommended image size for view"); + emplaceCaseWithDimensions(viewConfigView.maxImageRectWidth, viewConfigView.maxImageRectHeight, "Max image size for view"); if (!tp.compressedFormat) { - auto createInfo = defaultCreateInfo; - { + if (viewConfigView.recommendedSwapchainSampleCount != defaultCreateInfo.sampleCount) { + auto createInfo = defaultCreateInfo; createInfo.sampleCount = viewConfigView.recommendedSwapchainSampleCount; ret.emplace_back("Recommended sample count", createInfo); } - { + + if (viewConfigView.maxSwapchainSampleCount != defaultCreateInfo.sampleCount) { + auto createInfo = defaultCreateInfo; createInfo.sampleCount = viewConfigView.maxSwapchainSampleCount; ret.emplace_back("Max sample count", createInfo); } @@ -347,32 +347,42 @@ namespace Conformance for (const auto& cf : tp.createFlagsVector) { auto createInfo = defaultCreateInfo; - createInfo.createFlags = cf; - ret.emplace_back("Non-default create flags", createInfo); + if (cf != createInfo.createFlags) { + createInfo.createFlags = cf; + ret.emplace_back("Non-default create flags", createInfo); + } } for (const auto& sc : tp.sampleCountVector) { auto createInfo = defaultCreateInfo; - createInfo.sampleCount = sc; - ret.emplace_back("Non-default sample count", createInfo); + if (sc != createInfo.sampleCount) { + createInfo.sampleCount = sc; + ret.emplace_back("Non-default sample count", createInfo); + } } for (const auto& uf : tp.usageFlagsVector) { auto createInfo = defaultCreateInfo; - createInfo.usageFlags = (XrSwapchainUsageFlags)uf; - ret.emplace_back("Non-default usage flags", createInfo); + if (uf != createInfo.usageFlags) { + createInfo.usageFlags = (XrSwapchainUsageFlags)uf; + ret.emplace_back("Non-default usage flags", createInfo); + } } for (const auto& ac : tp.arrayCountVector) { auto createInfo = defaultCreateInfo; - createInfo.arraySize = ac; - ret.emplace_back("Non-default array size", createInfo); + if (ac != createInfo.arraySize) { + createInfo.arraySize = ac; + ret.emplace_back("Non-default array size", createInfo); + } } for (const auto& mc : tp.mipCountVector) { auto createInfo = defaultCreateInfo; - createInfo.mipCount = mc; - ret.emplace_back("Non-default mip count", createInfo); + if (mc != createInfo.mipCount) { + createInfo.mipCount = mc; + ret.emplace_back("Non-default mip count", createInfo); + } } return ret; diff --git a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp index 8145d8a1..2b6fda21 100644 --- a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp @@ -1,4 +1,6 @@ -// Copyright (c) 2019-2023, The Khronos Group Inc. +// Copyright (c) 2017-2023, The Khronos Group Inc. +// Copyright (c) 2017-2019 Valve Corporation +// Copyright (c) 2017-2019 LunarG, Inc. // // SPDX-License-Identifier: Apache-2.0 // @@ -18,11 +20,15 @@ #include "conformance_utils.h" #include "conformance_framework.h" #include "utilities/throw_helpers.h" +#include "hex_and_handles.h" #include +#include +#include #include #include +#include #include #include #include @@ -31,12 +37,12 @@ namespace Conformance { // It would be nice to have these functions as lambdas per test case or section but - // lambdas will not account for XRAPI_CALL calling conventions for all systems. + // lambdas will not account for XRAPI_ATTR, XRAPI_CALL calling conventions for all systems. static XRAPI_ATTR XrBool32 XRAPI_CALL myOutputDebugString(XrDebugUtilsMessageSeverityFlagsEXT, XrDebugUtilsMessageTypeFlagsEXT, const XrDebugUtilsMessengerCallbackDataEXT* callbackData, void* userData) { - REQUIRE(userData == NULL); + REQUIRE(userData == nullptr); WARN(callbackData->message); return XR_FALSE; }; @@ -44,7 +50,7 @@ namespace Conformance static XRAPI_ATTR XrBool32 XRAPI_CALL myDebugBreak(XrDebugUtilsMessageSeverityFlagsEXT, XrDebugUtilsMessageTypeFlagsEXT, const XrDebugUtilsMessengerCallbackDataEXT*, void* userData) { - REQUIRE(userData == NULL); + REQUIRE(userData == nullptr); FAIL(); return XR_FALSE; }; @@ -52,38 +58,172 @@ namespace Conformance static XRAPI_ATTR XrBool32 XRAPI_CALL myStdOutLogger(XrDebugUtilsMessageSeverityFlagsEXT, XrDebugUtilsMessageTypeFlagsEXT, const XrDebugUtilsMessengerCallbackDataEXT* callbackData, void* userData) { - REQUIRE(userData == NULL); + REQUIRE(userData == nullptr); INFO(callbackData->message); return XR_FALSE; }; - static XRAPI_ATTR XrBool32 XRAPI_CALL addToStringPairVector(XrDebugUtilsMessageSeverityFlagsEXT, XrDebugUtilsMessageTypeFlagsEXT, - const XrDebugUtilsMessengerCallbackDataEXT* callbackData, void* userData) + struct DebugUtilsCallbackInfo + { + XrDebugUtilsMessageSeverityFlagsEXT messageSeverity; + XrDebugUtilsMessageTypeFlagsEXT messageTypes; + XrDebugUtilsMessengerCallbackDataEXT callbackData; + + std::vector objects; + std::vector sessionLabels; + + // All of the debug utils structs contain strings which are not valid + // for us to reference after the callback function has returned. We will + // store a vector of strings that were used with each of our callbacks + // to avoid this problem. + std::vector> strings; + }; + + static XRAPI_ATTR XrBool32 XRAPI_CALL addToDebugUtilsCallbackInfoVector(XrDebugUtilsMessageSeverityFlagsEXT messageSeverity, + XrDebugUtilsMessageTypeFlagsEXT messageTypes, + const XrDebugUtilsMessengerCallbackDataEXT* callbackData, + void* userData) { REQUIRE(userData != NULL); - auto pMessages = reinterpret_cast>*>(userData); + auto pMessages = reinterpret_cast*>(userData); + + DebugUtilsCallbackInfo callbackInfo; + callbackInfo.messageSeverity = messageSeverity; + callbackInfo.messageTypes = messageTypes; + callbackInfo.callbackData = *callbackData; + + if (callbackData->messageId != nullptr) { + auto tmp = std::make_shared(callbackData->messageId); + callbackInfo.strings.push_back(tmp); + callbackInfo.callbackData.messageId = tmp->c_str(); + } + + if (callbackData->functionName != nullptr) { + auto tmp = std::make_shared(callbackData->functionName); + callbackInfo.strings.push_back(tmp); + callbackInfo.callbackData.functionName = tmp->c_str(); + } + + if (callbackData->message != nullptr) { + auto tmp = std::make_shared(callbackData->message); + callbackInfo.strings.push_back(tmp); + callbackInfo.callbackData.message = tmp->c_str(); + } + + for (uint32_t i = 0; i < callbackData->objectCount; ++i) { + callbackInfo.objects.push_back(callbackData->objects[i]); + if (callbackData->objects[i].objectName != nullptr) { + auto tmp = std::make_shared(callbackData->objects[i].objectName); + callbackInfo.strings.push_back(tmp); + callbackInfo.objects[i].objectName = tmp->c_str(); + } + } + callbackInfo.callbackData.objects = callbackInfo.objects.data(); + + for (uint32_t i = 0; i < callbackData->sessionLabelCount; ++i) { + callbackInfo.sessionLabels.push_back(callbackData->sessionLabels[i]); + if (callbackData->sessionLabels[i].labelName != nullptr) { + auto tmp = std::make_shared(callbackData->sessionLabels[i].labelName); + callbackInfo.strings.push_back(tmp); + callbackInfo.sessionLabels[i].labelName = tmp->c_str(); + } + } + callbackInfo.callbackData.sessionLabels = callbackInfo.sessionLabels.data(); - std::string functionName = (callbackData->functionName != nullptr ? callbackData->functionName : ""); - std::string message = (callbackData->message != nullptr ? callbackData->message : ""); - std::pair pair = std::make_pair(std::move(functionName), std::move(message)); - pMessages->push_back(std::move(pair)); + pMessages->push_back(std::move(callbackInfo)); return XR_FALSE; }; - static XRAPI_ATTR XrBool32 XRAPI_CALL createInstanceCallback(XrDebugUtilsMessageSeverityFlagsEXT messageSeverity, - XrDebugUtilsMessageTypeFlagsEXT /* messageTypes */, - const XrDebugUtilsMessengerCallbackDataEXT* callbackData, void* userData) + static bool debugMessageExists(const std::vector& callbackInfos, + XrDebugUtilsMessageSeverityFlagsEXT messageSeverity, XrDebugUtilsMessageTypeFlagsEXT messageTypes, + const XrDebugUtilsMessengerCallbackDataEXT* callbackData) { - REQUIRE(userData == nullptr); - if ((messageSeverity & XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT) != 0) { - FAIL_CHECK("Conformance layer error: " << callbackData->functionName << ": " << callbackData->message); - } - else { - WARN("Conformance layer warning: " << callbackData->functionName << ": " << callbackData->message); + auto callbackDataMatches = [](const XrDebugUtilsMessengerCallbackDataEXT* a, + const XrDebugUtilsMessengerCallbackDataEXT* b) -> bool { + REQUIRE(a->type == XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT); + REQUIRE(b->type == XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT); + + // We are not validating next chains match, but that should be ok. + + if ((a->messageId == nullptr) != (b->messageId == nullptr)) { + return false; + } + if ((a->messageId != nullptr) && (b->messageId != nullptr)) { + if (strcmp(a->messageId, b->messageId) != 0) { + return false; + } + } + + if ((a->functionName == nullptr) != (b->functionName == nullptr)) { + return false; + } + if ((a->functionName != nullptr) && (b->functionName != nullptr)) { + if (strcmp(a->functionName, b->functionName) != 0) { + return false; + } + } + + if ((a->message == nullptr) != (b->message == nullptr)) { + return false; + } + if ((a->message != nullptr) && (b->message != nullptr)) { + if (strcmp(a->message, b->message) != 0) { + return false; + } + } + + if (a->objectCount != b->objectCount) { + return false; + } + for (uint32_t i = 0; i < a->objectCount; ++i) { + REQUIRE(a->objects[i].type == XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT); + REQUIRE(b->objects[i].type == XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT); + + if (a->objects[i].objectType != b->objects[i].objectType) { + return false; + } + if (a->objects[i].objectHandle != b->objects[i].objectHandle) { + return false; + } + if ((a->objects[i].objectName == nullptr) != (b->objects[i].objectName == nullptr)) { + return false; + } + if ((a->objects[i].objectName != nullptr) && (b->objects[i].objectName != nullptr)) { + if (strcmp(a->objects[i].objectName, b->objects[i].objectName) != 0) { + return false; + } + } + } + + if (a->sessionLabelCount != b->sessionLabelCount) { + return false; + } + for (uint32_t i = 0; i < a->sessionLabelCount; ++i) { + REQUIRE(a->sessionLabels[i].type == XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT); + REQUIRE(b->sessionLabels[i].type == XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT); + + if ((a->sessionLabels[i].labelName == nullptr) != (b->sessionLabels[i].labelName == nullptr)) { + return false; + } + if ((a->sessionLabels[i].labelName != nullptr) && (b->sessionLabels[i].labelName != nullptr)) { + if (strcmp(a->sessionLabels[i].labelName, b->sessionLabels[i].labelName) != 0) { + return false; + } + } + } + + return true; + }; + + for (const auto& callbackInfo : callbackInfos) { + if (callbackInfo.messageSeverity == messageSeverity && callbackInfo.messageTypes == messageTypes && + callbackDataMatches(&callbackInfo.callbackData, callbackData)) { + return true; + } } - return XR_TRUE; - }; + return false; + } TEST_CASE("XR_EXT_debug_utils", "") { @@ -100,51 +240,6 @@ namespace Conformance SKIP(XR_EXT_DEBUG_UTILS_EXTENSION_NAME " not supported"); } - SECTION("xrSubmitDebugUtilsMessageEXT") - { - std::vector> messages; - - auto testMessage = std::make_pair("worker", "testing!"); - - auto worker = [&](XrInstance instance) -> void { - auto pfnSubmitDebugUtilsMessageEXT = - GetInstanceExtensionFunction(instance, "xrSubmitDebugUtilsMessageEXT"); - - XrDebugUtilsMessengerCallbackDataEXT callbackData{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; - callbackData.functionName = testMessage.first.c_str(); - callbackData.message = testMessage.second.c_str(); - REQUIRE_RESULT(XR_SUCCESS, pfnSubmitDebugUtilsMessageEXT(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, - XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callbackData)); - }; - - AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); - - // Must call extension functions through a function pointer: - auto pfnCreateDebugUtilsMessengerEXT = - GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); - - XrDebugUtilsMessengerCreateInfoEXT callback = { - XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT, // type - NULL, // next - XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT | // messageSeverities - XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT | - XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, - XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT | // messageTypes - XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT | - XR_DEBUG_UTILS_MESSAGE_TYPE_CONFORMANCE_BIT_EXT, - addToStringPairVector, // userCallback - reinterpret_cast(&messages) // userData - }; - XrDebugUtilsMessengerEXT messenger = XR_NULL_HANDLE; - REQUIRE_RESULT(XR_SUCCESS, pfnCreateDebugUtilsMessengerEXT(instance, &callback, &messenger)); - - worker(instance); - - REQUIRE(messages.size() > 0); - - REQUIRE(messages.end() != std::find(messages.begin(), messages.end(), testMessage)); - } - SECTION("xrCreateInstance debug utils not enabled") { auto enabledApiLayers = StringVec(globalData.enabledAPILayerNames); @@ -173,7 +268,7 @@ namespace Conformance ValidateInstanceExtensionFunctionNotSupported(instance, "xrCreateDebugUtilsMessengerEXT"); } - SECTION("xrCreateInstance XrDebugUtilsMessengerCreateInfoEXT") + SECTION("Create/Destroy with xrCreateInstance/xrDestroyInstance") { // To capture events that occur while creating or destroying an instance an application can link // an XrDebugUtilsMessengerCreateInfoEXT structure to the next element of the XrInstanceCreateInfo @@ -182,15 +277,19 @@ namespace Conformance // is not passed as an option, but we have an explicit test for this behavior too. auto enabledApiLayers = StringVec(globalData.enabledAPILayerNames); + // Enable only the required platform extensions by default auto enabledExtensions = StringVec(globalData.requiredPlatformInstanceExtensions); + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT debugInfo{XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; debugInfo.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT; - debugInfo.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_CONFORMANCE_BIT_EXT; - debugInfo.userCallback = createInstanceCallback; - debugInfo.userData = nullptr; + debugInfo.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; + debugInfo.userCallback = addToDebugUtilsCallbackInfoVector; + debugInfo.userData = reinterpret_cast(&callbackInfo); XrInstance instance{XR_NULL_HANDLE}; CleanupInstanceOnScopeExit cleanup(instance); @@ -223,19 +322,661 @@ namespace Conformance GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); REQUIRE(pfnCreateDebugUtilsMessengerEXT != nullptr); } + { + // Get a function pointer to the submit function to test + PFN_xrSubmitDebugUtilsMessageEXT pfn_submit_dmsg; + REQUIRE_RESULT(XR_SUCCESS, xrGetInstanceProcAddr(instance, "xrSubmitDebugUtilsMessageEXT", + reinterpret_cast(&pfn_submit_dmsg))); + REQUIRE(pfn_submit_dmsg != nullptr); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + } REQUIRE_RESULT(XR_SUCCESS, xrDestroyInstance(instance)); instance = XR_NULL_HANDLE; } + SECTION("Create/Destroy with explicit call (xrCreateDebugUtilsMessengerEXT/xrDestroyDebugUtilsMessengerEXT)") + { + AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); + + // Get a function pointer to the various debug utils functions to test + auto pfn_create_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); + auto pfn_destroy_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrDestroyDebugUtilsMessengerEXT"); + auto pfn_submit_dmsg = GetInstanceExtensionFunction(instance, "xrSubmitDebugUtilsMessageEXT"); + + // Create the debug utils messenger + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT | + XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT | XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT | + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + + SECTION("Make sure appropriate messages only received when registered") + { + AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); + + // Get a function pointer to the various debug utils functions to test + auto pfn_create_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); + auto pfn_destroy_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrDestroyDebugUtilsMessengerEXT"); + auto pfn_submit_dmsg = GetInstanceExtensionFunction(instance, "xrSubmitDebugUtilsMessageEXT"); + + SECTION("Create the debug utils messenger, but only to receive general error messages") + { + // Create the debug utils messenger, but only to receive general error messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + + SECTION("Create the debug utils messenger, but only to receive validation warning messages") + { + // Create the debug utils messenger, but only to receive validation warning messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + + SECTION("Create the debug utils messenger, but only to receive performance verbose messages") + { + // Create the debug utils messenger, but only to receive performance verbose messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + + SECTION("Create the debug utils messenger, but only to info validation messages") + { + // Create the debug utils messenger, but only to receive validation info messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + // Test the various items + { + callback_data.messageId = "General Error"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_ERROR_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Validation Warning"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_WARNING_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Info"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "General Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + REQUIRE(!debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_GENERAL_BIT_EXT, &callback_data)); + } + { + callback_data.messageId = "Performance Verbose"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_INFO_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_VALIDATION_BIT_EXT, &callback_data)); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + } + + SECTION("Test Objects") + { + AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); + AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::createSpaces | AutoBasicSession::createSwapchains, + instance); + + // Get a function pointer to the various debug utils functions to test + auto pfn_create_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); + auto pfn_destroy_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrDestroyDebugUtilsMessengerEXT"); + auto pfn_submit_dmsg = GetInstanceExtensionFunction(instance, "xrSubmitDebugUtilsMessageEXT"); + + // Create the debug utils messenger, but only to receive validation warning messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + + std::array objects; + objects.fill({XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT}); + objects[0].objectType = XR_OBJECT_TYPE_INSTANCE; + objects[0].objectHandle = MakeHandleGeneric(instance.GetInstance()); + objects[0].objectName = nullptr; + objects[1].objectType = XR_OBJECT_TYPE_SESSION; + objects[1].objectHandle = MakeHandleGeneric(session.GetSession()); + objects[1].objectName = nullptr; + objects[2].objectType = XR_OBJECT_TYPE_DEBUG_UTILS_MESSENGER_EXT; + objects[2].objectHandle = MakeHandleGeneric(debug_utils_messenger); + objects[2].objectName = nullptr; + callback_data.objects = objects.data(); + callback_data.objectCount = static_cast(objects.size()); + + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + + SECTION("Test object names") + { + auto findMessageByMessageId = [](const std::vector& callbackInfos, + const char* messageId) -> const DebugUtilsCallbackInfo& { + { + size_t messageMatchCount = 0; + for (const auto& callbackInfo : callbackInfos) { + if (strcmp(callbackInfo.callbackData.messageId, messageId) == 0) { + messageMatchCount++; + } + } + REQUIRE(messageMatchCount == 1); + } + auto it = std::find_if(callbackInfos.begin(), callbackInfos.end(), [messageId](const auto& callbackInfo) { + return strcmp(callbackInfo.callbackData.messageId, messageId) == 0; + }); + REQUIRE(it != callbackInfos.end()); + return *it; + }; + + AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); + + auto pfn_create_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrCreateDebugUtilsMessengerEXT"); + auto pfn_destroy_debug_utils_messager_ext = + GetInstanceExtensionFunction(instance, "xrDestroyDebugUtilsMessengerEXT"); + auto pfn_submit_dmsg = GetInstanceExtensionFunction(instance, "xrSubmitDebugUtilsMessageEXT"); + auto pfn_set_obj_name = + GetInstanceExtensionFunction(instance, "xrSetDebugUtilsObjectNameEXT"); + auto pfn_begin_debug_utils_label_region_ext = GetInstanceExtensionFunction( + instance, "xrSessionBeginDebugUtilsLabelRegionEXT"); + auto pfn_end_debug_utils_label_region_ext = + GetInstanceExtensionFunction(instance, "xrSessionEndDebugUtilsLabelRegionEXT"); + auto pfn_insert_debug_utils_label_ext = + GetInstanceExtensionFunction(instance, "xrSessionInsertDebugUtilsLabelEXT"); + + // Create the debug utils messenger, but only to receive validation warning messages + std::vector callbackInfo; + XrDebugUtilsMessengerCreateInfoEXT dbg_msg_ci = {XR_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; + dbg_msg_ci.messageSeverities = XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT; + dbg_msg_ci.messageTypes = XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT; + dbg_msg_ci.userCallback = addToDebugUtilsCallbackInfoVector; + dbg_msg_ci.userData = reinterpret_cast(&callbackInfo); + XrDebugUtilsMessengerEXT debug_utils_messenger = XR_NULL_HANDLE; + REQUIRE_RESULT(XR_SUCCESS, pfn_create_debug_utils_messager_ext(instance, &dbg_msg_ci, &debug_utils_messenger)); + + XrDebugUtilsObjectNameInfoEXT object{XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT}; + object.objectType = XR_OBJECT_TYPE_INSTANCE; + object.objectHandle = MakeHandleGeneric(instance.GetInstance()); + object.objectName = "My Instance Obj"; + REQUIRE_RESULT(XR_SUCCESS, pfn_set_obj_name(instance, &object)); + + // TODO: validate objectName works (in a separate test case) + + { + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + callback_data.objectCount = 1; + callback_data.objects = &object; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + REQUIRE(debugMessageExists(callbackInfo, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + } + + { + static const char first_individual_label_name[] = "First individual label"; + static const char second_individual_label_name[] = "Second individual label"; + static const char third_individual_label_name[] = "Third individual label"; + static const char first_label_region_name[] = "First Label Region"; + static const char second_label_region_name[] = "Second Label Region"; + + AutoBasicSession session( + AutoBasicSession::createSession | AutoBasicSession::createSpaces | AutoBasicSession::createSwapchains, instance); + FrameIterator frameIterator(&session); + + auto timeout = (globalData.options.debugMode ? 3600s : 10s); + CAPTURE(timeout); + + // Create a label struct for initial testing + XrDebugUtilsLabelEXT first_label = {XR_TYPE_DEBUG_UTILS_LABEL_EXT}; + first_label.labelName = first_individual_label_name; + + // TODO: Invalid parameters might be better as a separate test case + SECTION("Invalid parameters") + { + // Try invalid session on each of the label functions + REQUIRE_RESULT(XR_ERROR_HANDLE_INVALID, pfn_begin_debug_utils_label_region_ext(XR_NULL_HANDLE, &first_label)); + REQUIRE_RESULT(XR_ERROR_HANDLE_INVALID, pfn_end_debug_utils_label_region_ext(XR_NULL_HANDLE)); + REQUIRE_RESULT(XR_ERROR_HANDLE_INVALID, pfn_insert_debug_utils_label_ext(XR_NULL_HANDLE, &first_label)); + + // Try with nullptr for the label + REQUIRE_RESULT(XR_ERROR_VALIDATION_FAILURE, pfn_begin_debug_utils_label_region_ext(session, nullptr)); + REQUIRE_RESULT(XR_ERROR_VALIDATION_FAILURE, pfn_insert_debug_utils_label_ext(session, nullptr)); + } + + // Set it up to put in the session and instance to any debug utils messages + XrDebugUtilsMessengerCallbackDataEXT callback_data{XR_TYPE_DEBUG_UTILS_MESSENGER_CALLBACK_DATA_EXT}; + callback_data.messageId = "General Error"; + callback_data.functionName = "MyTestFunctionName"; + callback_data.message = "General Error"; + std::array objects; + objects.fill({XR_TYPE_DEBUG_UTILS_OBJECT_NAME_INFO_EXT}); + objects[0].objectType = XR_OBJECT_TYPE_INSTANCE; + objects[0].objectHandle = MakeHandleGeneric(instance.GetInstance()); + objects[0].objectName = nullptr; + objects[1].objectType = XR_OBJECT_TYPE_SESSION; + objects[1].objectHandle = MakeHandleGeneric(session.GetSession()); + objects[1].objectName = nullptr; + callback_data.objectCount = static_cast(objects.size()); + callback_data.objects = objects.data(); + + // Start an individual label + REQUIRE_RESULT(XR_SUCCESS, pfn_insert_debug_utils_label_ext(session, &first_label)); + + // Trigger a message and make sure we see "First individual label" + { + callback_data.messageId = "First Individual Label"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 1); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(first_individual_label_name)); + } + + // Begin a label region + first_label.labelName = first_label_region_name; + REQUIRE_RESULT(XR_SUCCESS, pfn_begin_debug_utils_label_region_ext(session, &first_label)); + + // Trigger a message and make sure we see "Label Region" and not "First individual label" + { + callback_data.messageId = "First Label Region"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 1); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(first_label_region_name)); + } + + // Begin the session now. + { + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + XrSessionBeginInfo session_begin_info = {XR_TYPE_SESSION_BEGIN_INFO}; + session_begin_info.primaryViewConfigurationType = GetGlobalData().GetOptions().viewConfigurationValue; + REQUIRE_RESULT(XR_SUCCESS, xrBeginSession(session, &session_begin_info)); + } + + XrDebugUtilsLabelEXT individual_label{XR_TYPE_DEBUG_UTILS_LABEL_EXT}; + individual_label.labelName = second_individual_label_name; + REQUIRE_RESULT(XR_SUCCESS, pfn_insert_debug_utils_label_ext(session, &individual_label)); + + // Trigger a message and make sure we see "Second individual" and "First Label Region" and not "First + // individual label" + { + callback_data.messageId = "Second Individual and First Region"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + // From: https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#session-labels + // The labels listed inside sessionLabels are organized in time order, with the most recently + // generated label appearing first, and the oldest label appearing last. + REQUIRE(cb.callbackData.sessionLabelCount == 2); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(second_individual_label_name)); + REQUIRE_THAT(cb.callbackData.sessionLabels[1].labelName, Catch::Matchers::Equals(first_label_region_name)); + } + + individual_label.labelName = third_individual_label_name; + REQUIRE_RESULT(XR_SUCCESS, pfn_insert_debug_utils_label_ext(session, &individual_label)); + + // Trigger a message and make sure we see "Third individual" and "First Label Region" and not "First + // individual label" or "Second individual label" + { + callback_data.messageId = "Third Individual and First Region"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 2); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(third_individual_label_name)); + REQUIRE_THAT(cb.callbackData.sessionLabels[1].labelName, Catch::Matchers::Equals(first_label_region_name)); + } + + // Begin a label region + { + XrDebugUtilsLabelEXT second_label_region = {XR_TYPE_DEBUG_UTILS_LABEL_EXT}; + second_label_region.labelName = second_label_region_name; + REQUIRE_RESULT(XR_SUCCESS, pfn_begin_debug_utils_label_region_ext(session, &second_label_region)); + } + + // Trigger a message and make sure we see "Second Label Region" and "First Label Region" + { + callback_data.messageId = "Second and First Label Regions"; + pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 2); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(second_label_region_name)); + REQUIRE_THAT(cb.callbackData.sessionLabels[1].labelName, Catch::Matchers::Equals(first_label_region_name)); + } + + // End the last (most recent) label region + { + REQUIRE_RESULT(XR_SUCCESS, pfn_end_debug_utils_label_region_ext(session)); + // TODO: need a test for end a label region that has not been started + } + + // Trigger a message and make sure we see "First Label Region" + { + callback_data.messageId = "First Label Region 2"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 1); + REQUIRE_THAT(cb.callbackData.sessionLabels[0].labelName, Catch::Matchers::Equals(first_label_region_name)); + } + + // Now clean-up (the session) + { + REQUIRE_RESULT(XR_SUCCESS, xrRequestExitSession(session)); + + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + REQUIRE_RESULT(XR_SUCCESS, xrEndSession(session)); + } + + // End the last label region + { + REQUIRE_RESULT(XR_SUCCESS, pfn_end_debug_utils_label_region_ext(session)); + } + + // Trigger a message and make sure we see no labels + { + callback_data.messageId = "No Labels"; + REQUIRE_RESULT(XR_SUCCESS, pfn_submit_dmsg(instance, XR_DEBUG_UTILS_MESSAGE_SEVERITY_VERBOSE_BIT_EXT, + XR_DEBUG_UTILS_MESSAGE_TYPE_PERFORMANCE_BIT_EXT, &callback_data)); + const auto& cb = findMessageByMessageId(callbackInfo, callback_data.messageId); + REQUIRE(cb.callbackData.sessionLabelCount == 0); + } + + session.Shutdown(); + } + + // Destroy what we created + REQUIRE_RESULT(XR_SUCCESS, pfn_destroy_debug_utils_messager_ext(debug_utils_messenger)); + } + // https://www.khronos.org/registry/OpenXR/specs/1.0/html/xrspec.html#XR_EXT_debug_utils // The OpenXR spec provides some examples of how to use the extension; they are not full // examples but let's make sure that something equivalent to them works. // Example 1 / multiple callbacks +#define CHK_XR(expr) XRC_CHECK_THROW_XRCMD(expr) + SECTION("Examples") { -#define CHK_XR(expr) XRC_CHECK_THROW_XRCMD(expr) SECTION("Example 1: Multiple callbacks") { AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); @@ -411,8 +1152,8 @@ namespace Conformance // (in this case it's the "Session Active" label) pfnSessionEndDebugUtilsLabelRegionEXT(session); } -#undef CHK_XR } +#undef CHK_XR } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp index ad2e8e72..e37130fa 100644 --- a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp @@ -36,7 +36,7 @@ namespace Conformance MakeSystemPropertiesBoolChecker(XrSystemHandTrackingPropertiesEXT{XR_TYPE_SYSTEM_HAND_TRACKING_PROPERTIES_EXT}, &XrSystemHandTrackingPropertiesEXT::supportsHandTracking); - TEST_CASE("XR_EXT_hand_tracking", "") + TEST_CASE("XR_EXT_hand_tracking-create-destroy", "") { GlobalData& globalData = GetGlobalData(); if (!globalData.IsInstanceExtensionSupported(XR_EXT_HAND_TRACKING_EXTENSION_NAME)) { @@ -81,41 +81,54 @@ namespace Conformance createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); REQUIRE(XR_ERROR_FEATURE_UNSUPPORTED == xrCreateHandTrackerEXT(session, &createInfo, &tracker)); } - - // This runtime does support hand tracking, but this headset does not - // support hand tracking, which is fine. - SKIP("System does not support hand tracking"); } - - std::array handTracker; - for (size_t i = 0; i < HAND_COUNT; ++i) { - XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; - createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; - createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); - REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); - REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); + else { + std::array handTracker; + for (size_t i = 0; i < HAND_COUNT; ++i) { + XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; + createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; + createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); + REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); + REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); + } } } + } - SECTION("Query joint locations") - { - AutoBasicInstance instance({XR_EXT_HAND_TRACKING_EXTENSION_NAME}, AutoBasicInstance::createSystemId); + TEST_CASE("XR_EXT_hand_tracking-simple-queries") + { + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_HAND_TRACKING_EXTENSION_NAME)) { + SKIP(XR_EXT_HAND_TRACKING_EXTENSION_NAME " not supported"); + } - auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); - auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); - auto xrLocateHandJointsEXT = GetInstanceExtensionFunction(instance, "xrLocateHandJointsEXT"); + AutoBasicInstance instance({XR_EXT_HAND_TRACKING_EXTENSION_NAME}, AutoBasicInstance::createSystemId); - XrSystemId systemId = instance.systemId; - if (!SystemSupportsHandTracking(instance, systemId)) { - // This runtime does support hand tracking, but this headset does not - // support hand tracking, which is fine. - SKIP("System does not support hand tracking"); - } + auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); + auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); + auto xrLocateHandJointsEXT = GetInstanceExtensionFunction(instance, "xrLocateHandJointsEXT"); - AutoBasicSession session(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | - AutoBasicSession::createSwapchains, - instance); + XrSystemId systemId = instance.systemId; + if (!SystemSupportsHandTracking(instance, systemId)) { + // This runtime does support hand tracking, but this headset does not + // support hand tracking, which is fine. + SKIP("System does not support hand tracking"); + } + + AutoBasicSession session(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | + AutoBasicSession::createSwapchains, + instance); + std::array handTracker; + for (size_t i = 0; i < HAND_COUNT; ++i) { + XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; + createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; + createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); + REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); + } + + SECTION("Query joint locations") + { XrSpace localSpace = XR_NULL_HANDLE; XrReferenceSpaceCreateInfo localSpaceCreateInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; @@ -123,14 +136,6 @@ namespace Conformance localSpaceCreateInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL; REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); - std::array handTracker; - for (size_t i = 0; i < HAND_COUNT; ++i) { - XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; - createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; - createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); - REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); - } - // Wait until the runtime is ready for us to begin a session auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); @@ -190,29 +195,10 @@ namespace Conformance } } } - - for (size_t i = 0; i < HAND_COUNT; ++i) { - REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); - } } SECTION("Query invalid joint sets") { - AutoBasicInstance instance({XR_EXT_HAND_TRACKING_EXTENSION_NAME}, AutoBasicInstance::createSystemId); - - auto xrCreateHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrCreateHandTrackerEXT"); - auto xrDestroyHandTrackerEXT = GetInstanceExtensionFunction(instance, "xrDestroyHandTrackerEXT"); - auto xrLocateHandJointsEXT = GetInstanceExtensionFunction(instance, "xrLocateHandJointsEXT"); - - XrSystemId systemId = instance.systemId; - if (!SystemSupportsHandTracking(instance, systemId)) { - // This runtime does support hand tracking, but this headset does not - // support hand tracking, which is fine. - SKIP("System does not support hand tracking"); - } - - AutoBasicSession session(AutoBasicSession::beginSession, instance); - XrSpace localSpace = XR_NULL_HANDLE; XrReferenceSpaceCreateInfo localSpaceCreateInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; @@ -220,14 +206,6 @@ namespace Conformance localSpaceCreateInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL; REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); - std::array handTracker; - for (size_t i = 0; i < HAND_COUNT; ++i) { - XrHandTrackerCreateInfoEXT createInfo{XR_TYPE_HAND_TRACKER_CREATE_INFO_EXT}; - createInfo.handJointSet = XR_HAND_JOINT_SET_DEFAULT_EXT; - createInfo.hand = (i == 0 ? XR_HAND_LEFT_EXT : XR_HAND_RIGHT_EXT); - REQUIRE(XR_SUCCESS == xrCreateHandTrackerEXT(session, &createInfo, &handTracker[i])); - } - // Wait until the runtime is ready for us to begin a session auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); @@ -271,10 +249,10 @@ namespace Conformance locateInfo.time = frameIterator.frameState.predictedDisplayTime; REQUIRE(XR_ERROR_VALIDATION_FAILURE == xrLocateHandJointsEXT(handTracker[hand], &locateInfo, &locations)); } + } - for (size_t i = 0; i < HAND_COUNT; ++i) { - REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); - } + for (size_t i = 0; i < HAND_COUNT; ++i) { + REQUIRE(XR_SUCCESS == xrDestroyHandTrackerEXT(handTracker[i])); } } diff --git a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp index 5391e7fd..3d2cb6c5 100644 --- a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp @@ -46,7 +46,7 @@ namespace Conformance constexpr XrVector3f Up{0, 1, 0}; // Purpose: Ensure that the action space for palm can be used for placing a hand representation. - TEST_CASE("XR_EXT_palm_pose", "[scenario][interactive][no_auto]") + TEST_CASE("XR_EXT_palm_pose", "[XR_EXT_palm_pose][scenario][interactive][no_auto]") { GlobalData& globalData = GetGlobalData(); if (!globalData.IsInstanceExtensionSupported(XR_EXT_PALM_POSE_EXTENSION_NAME)) { @@ -408,4 +408,157 @@ namespace Conformance RenderLoop(compositionHelper.GetSession(), update).Loop(); } + + TEST_CASE("XR_EXT_palm_pose-noninteractive", "[XR_EXT_palm_pose]") + { + GlobalData& globalData = GetGlobalData(); + if (!globalData.IsInstanceExtensionSupported(XR_EXT_PALM_POSE_EXTENSION_NAME)) { + SKIP(); + } + + AutoBasicInstance instance({XR_EXT_PALM_POSE_EXTENSION_NAME}); + + // Set up the actions. + const std::array subactionPaths{StringToPath(instance, "/user/hand/left"), StringToPath(instance, "/user/hand/right")}; + + XrActionSet actionSet; + XrAction gripPoseAction, aimPoseAction, handModelPoseAction; + { + XrActionSetCreateInfo actionSetInfo{XR_TYPE_ACTION_SET_CREATE_INFO}; + strcpy(actionSetInfo.actionSetName, "conformance_test"); + strcpy(actionSetInfo.localizedActionSetName, "Conformance Test"); + XRC_CHECK_THROW_XRCMD(xrCreateActionSet(instance, &actionSetInfo, &actionSet)); + + XrActionCreateInfo actionInfo{XR_TYPE_ACTION_CREATE_INFO}; + actionInfo.actionType = XR_ACTION_TYPE_POSE_INPUT; + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + + strcpy(actionInfo.actionName, "grip_pose"); + strcpy(actionInfo.localizedActionName, "grip pose"); + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + XRC_CHECK_THROW_XRCMD(xrCreateAction(actionSet, &actionInfo, &gripPoseAction)); + + strcpy(actionInfo.actionName, "aim_pose"); + strcpy(actionInfo.localizedActionName, "aim pose"); + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + XRC_CHECK_THROW_XRCMD(xrCreateAction(actionSet, &actionInfo, &aimPoseAction)); + + strcpy(actionInfo.actionName, "palm_pose"); + strcpy(actionInfo.localizedActionName, "palm pose"); + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + XRC_CHECK_THROW_XRCMD(xrCreateAction(actionSet, &actionInfo, &handModelPoseAction)); + } + + const std::vector bindings = { + {gripPoseAction, StringToPath(instance, "/user/hand/left/input/grip/pose")}, + {gripPoseAction, StringToPath(instance, "/user/hand/right/input/grip/pose")}, + {aimPoseAction, StringToPath(instance, "/user/hand/left/input/aim/pose")}, + {aimPoseAction, StringToPath(instance, "/user/hand/right/input/aim/pose")}, + {handModelPoseAction, StringToPath(instance, "/user/hand/left/input/palm_ext/pose")}, + {handModelPoseAction, StringToPath(instance, "/user/hand/right/input/palm_ext/pose")}, + }; + + XrInteractionProfileSuggestedBinding suggestedBindings{XR_TYPE_INTERACTION_PROFILE_SUGGESTED_BINDING}; + suggestedBindings.interactionProfile = StringToPath(instance, "/interaction_profiles/khr/simple_controller"); + suggestedBindings.suggestedBindings = bindings.data(); + suggestedBindings.countSuggestedBindings = (uint32_t)bindings.size(); + XRC_CHECK_THROW_XRCMD(xrSuggestInteractionProfileBindings(instance, &suggestedBindings)); + + AutoBasicSession session(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | + AutoBasicSession::createSwapchains, + instance); + + // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. + auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); + CAPTURE(timeout); + + // Get frames iterating to the point of app focused state. This will draw frames along the way. + FrameIterator frameIterator(&session); + FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); + REQUIRE(runResult == FrameIterator::RunResult::Success); + + XrSessionActionSetsAttachInfo attachInfo{XR_TYPE_SESSION_ACTION_SETS_ATTACH_INFO}; + attachInfo.actionSets = &actionSet; + attachInfo.countActionSets = 1; + XRC_CHECK_THROW_XRCMD(xrAttachSessionActionSets(session, &attachInfo)); + + XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; + XrActiveActionSet activeActionSet{actionSet}; + syncInfo.activeActionSets = &activeActionSet; + syncInfo.countActiveActionSets = 1; + + REQUIRE_RESULT(xrSyncActions(session, &syncInfo), XR_SUCCESS); + + // Set up the spaces. + std::array oculusBaseControllerSpaces; + + std::array gripSpaces; + std::array aimSpaces; + std::array handSpaces; + + XrSpace localSpace{XR_NULL_HANDLE}; + XrReferenceSpaceCreateInfo createSpaceInfo{XR_TYPE_REFERENCE_SPACE_CREATE_INFO}; + createSpaceInfo.referenceSpaceType = XR_REFERENCE_SPACE_TYPE_LOCAL; + createSpaceInfo.poseInReferenceSpace = XrPosefCPP(); + REQUIRE_RESULT(xrCreateReferenceSpace(session, &createSpaceInfo, &localSpace), XR_SUCCESS); + + for (int i = 0; i < 2; ++i) { + XrActionSpaceCreateInfo spaceCreateInfo{XR_TYPE_ACTION_SPACE_CREATE_INFO}; + spaceCreateInfo.subactionPath = subactionPaths[i]; + spaceCreateInfo.action = aimPoseAction; + spaceCreateInfo.poseInActionSpace = {{0.f, 0.f, 0.f, 1.f}, {0.f, 0.f, 0.55f}}; + XRC_CHECK_THROW_XRCMD(xrCreateActionSpace(session, &spaceCreateInfo, &oculusBaseControllerSpaces[i])); + } + + for (int i = 0; i < 2; ++i) { + XrActionSpaceCreateInfo spaceCreateInfo{XR_TYPE_ACTION_SPACE_CREATE_INFO}; + spaceCreateInfo.subactionPath = subactionPaths[i]; + spaceCreateInfo.action = gripPoseAction; + spaceCreateInfo.poseInActionSpace = XrPosefCPP(); + XRC_CHECK_THROW_XRCMD(xrCreateActionSpace(session, &spaceCreateInfo, &gripSpaces[i])); + } + + for (int i = 0; i < 2; ++i) { + XrActionSpaceCreateInfo spaceCreateInfo{XR_TYPE_ACTION_SPACE_CREATE_INFO}; + spaceCreateInfo.subactionPath = subactionPaths[i]; + spaceCreateInfo.action = aimPoseAction; + spaceCreateInfo.poseInActionSpace = XrPosefCPP(); + XRC_CHECK_THROW_XRCMD(xrCreateActionSpace(session, &spaceCreateInfo, &aimSpaces[i])); + } + + for (int i = 0; i < 2; ++i) { + XrActionSpaceCreateInfo spaceCreateInfo{XR_TYPE_ACTION_SPACE_CREATE_INFO}; + spaceCreateInfo.subactionPath = subactionPaths[i]; + spaceCreateInfo.action = handModelPoseAction; + spaceCreateInfo.poseInActionSpace = XrPosefCPP(); + XRC_CHECK_THROW_XRCMD(xrCreateActionSpace(session, &spaceCreateInfo, &handSpaces[i])); + } + + for (int i = 0; i < 2; ++i) { + CAPTURE(i == 0 ? "Left hand" : "Right hand"); + { + XrSpaceLocation gripSpaceLocation{XR_TYPE_SPACE_LOCATION}; + XRC_CHECK_THROW_XRCMD(xrLocateSpace(gripSpaces[i], oculusBaseControllerSpaces[i], + frameIterator.frameState.predictedDisplayTime, &gripSpaceLocation)); + // Not going to check the result here - controller might not even be active... + } + { + XrSpaceLocation aimSpaceLocation{XR_TYPE_SPACE_LOCATION}; + XRC_CHECK_THROW_XRCMD(xrLocateSpace(aimSpaces[i], oculusBaseControllerSpaces[i], + frameIterator.frameState.predictedDisplayTime, &aimSpaceLocation)); + // Not going to check the result here - controller might not even be active... + } + { + XrSpaceLocation palmSpaceLocation{XR_TYPE_SPACE_LOCATION}; + XRC_CHECK_THROW_XRCMD(xrLocateSpace(handSpaces[i], oculusBaseControllerSpaces[i], + frameIterator.frameState.predictedDisplayTime, &palmSpaceLocation)); + // Not going to check the result here - controller might not even be active... + } + } + } + } // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_MSFT_controller_model.cpp b/src/conformance/conformance_test/test_XR_MSFT_controller_model.cpp new file mode 100644 index 00000000..ce6833fc --- /dev/null +++ b/src/conformance/conformance_test/test_XR_MSFT_controller_model.cpp @@ -0,0 +1,273 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 + +#include "action_utils.h" +#include "composition_utils.h" +#include "conformance_framework.h" +#include "conformance_utils.h" +#include "report.h" +#include "two_call.h" +#include "two_call_struct_metadata.h" +#include "two_call_struct_tests.h" + +#include "common/hex_and_handles.h" +#include "utilities/utils.h" + +#include +#include + +#include +#include +#include +#include +#include + +namespace Conformance +{ + struct ExtensionDataForXR_MSFT_controller_model + { + XrInstance instance; + PFN_xrGetControllerModelKeyMSFT xrGetControllerModelKeyMSFT_; + PFN_xrGetControllerModelPropertiesMSFT xrGetControllerModelPropertiesMSFT_; + PFN_xrGetControllerModelStateMSFT xrGetControllerModelStateMSFT_; + PFN_xrLoadControllerModelMSFT xrLoadControllerModelMSFT_; + + ExtensionDataForXR_MSFT_controller_model(XrInstance instance_) + : instance(instance_) + , xrGetControllerModelKeyMSFT_( + GetInstanceExtensionFunction(instance, "xrGetControllerModelKeyMSFT")) + , xrGetControllerModelPropertiesMSFT_(GetInstanceExtensionFunction( + instance, "xrGetControllerModelPropertiesMSFT")) + , xrGetControllerModelStateMSFT_( + GetInstanceExtensionFunction(instance, "xrGetControllerModelStateMSFT")) + , xrLoadControllerModelMSFT_( + GetInstanceExtensionFunction(instance, "xrLoadControllerModelMSFT")) + { + } + + void CheckInvalidModelKey(XrSession session, XrControllerModelKeyMSFT modelKey) const + { + INFO("Known-invalid model key: " << Uint64ToHexString(modelKey)); + uint32_t countOutput = 0; + CHECK(XR_ERROR_CONTROLLER_MODEL_KEY_INVALID_MSFT == xrLoadControllerModelMSFT_(session, modelKey, 0, &countOutput, NULL)); + + XrControllerModelPropertiesMSFT modelProperties{XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}; + CHECK(XR_ERROR_CONTROLLER_MODEL_KEY_INVALID_MSFT == xrGetControllerModelPropertiesMSFT_(session, modelKey, &modelProperties)); + + XrControllerModelStateMSFT modelState{XR_TYPE_CONTROLLER_MODEL_STATE_MSFT}; + CHECK(XR_ERROR_CONTROLLER_MODEL_KEY_INVALID_MSFT == xrGetControllerModelStateMSFT_(session, modelKey, &modelState)); + } + + void CheckValidModelKeys(XrSession session, const std::vector& modelKeys) + { + + // Check two call struct for controller model properties and states, + // plus regular two-call for the model itself + auto modelPropertiesTwoCallData = getTwoCallStructData(); + auto modelStateTwoCallData = getTwoCallStructData(); + for (auto modelKey : modelKeys) { + INFO("Model key: " << Uint64ToHexString(modelKey)); + CheckTwoCallStructConformance(modelPropertiesTwoCallData, {XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}, + "xrGetControllerModelPropertiesMSFT", false, + [&](XrControllerModelPropertiesMSFT* properties) { + return xrGetControllerModelPropertiesMSFT_(session, modelKey, properties); + }); + + CheckTwoCallStructConformance( + modelStateTwoCallData, {XR_TYPE_CONTROLLER_MODEL_STATE_MSFT}, "xrGetControllerModelStateMSFT", false, + [&](XrControllerModelStateMSFT* state) { return xrGetControllerModelStateMSFT_(session, modelKey, state); }); + CHECK_TWO_CALL(uint8_t, {}, xrLoadControllerModelMSFT_, session, modelKey); + } + + // Try inventing some model keys that will be invalid. + + for (auto modelKey : modelKeys) { + XrControllerModelKeyMSFT fakeModelKey = modelKey + 1234; + if (std::find(modelKeys.begin(), modelKeys.end(), fakeModelKey) == modelKeys.end()) { + // We invented an invalid key + INFO("Invented model key: " << Uint64ToHexString(fakeModelKey)); + CheckInvalidModelKey(session, fakeModelKey); + } + } + } + }; + + TEST_CASE("XR_MSFT_controller_model-simple", "") + { + GlobalData& globalData = GetGlobalData(); + + if (!globalData.IsInstanceExtensionSupported(XR_MSFT_CONTROLLER_MODEL_EXTENSION_NAME)) { + SKIP(XR_MSFT_CONTROLLER_MODEL_EXTENSION_NAME " not supported"); + } + + AutoBasicInstance instance({"XR_MSFT_controller_model"}); + AutoBasicSession session(AutoBasicSession::OptionFlags::createSession, instance); + ExtensionDataForXR_MSFT_controller_model ext(instance); + ext.CheckInvalidModelKey(session, XR_NULL_CONTROLLER_MODEL_KEY_MSFT); + } + + TEST_CASE("XR_MSFT_controller_model", "") + { + GlobalData& globalData = GetGlobalData(); + + if (!globalData.IsInstanceExtensionSupported(XR_MSFT_CONTROLLER_MODEL_EXTENSION_NAME)) { + SKIP(XR_MSFT_CONTROLLER_MODEL_EXTENSION_NAME " not supported"); + } + + CompositionHelper compositionHelper("XR_MSFT_controller_model", {"XR_MSFT_controller_model"}); + XrInstance instance = compositionHelper.GetInstance(); + + ExtensionDataForXR_MSFT_controller_model ext(instance); + + ActionLayerManager actionLayerManager(compositionHelper); + XrPath simpleKHR = StringToPath(instance, "/interaction_profiles/microsoft/motion_controller"); + XrPath leftHandPath{StringToPath(instance, "/user/hand/left")}; + std::shared_ptr leftHandInputDevice = + CreateTestDevice(&actionLayerManager, &compositionHelper.GetInteractionManager(), instance, compositionHelper.GetSession(), + simpleKHR, leftHandPath, cWMRControllerIPData); + + XrPath rightHandPath{StringToPath(instance, "/user/hand/right")}; + std::shared_ptr rightHandInputDevice = + CreateTestDevice(&actionLayerManager, &compositionHelper.GetInteractionManager(), instance, compositionHelper.GetSession(), + simpleKHR, rightHandPath, cWMRControllerIPData); + + const std::vector subactionPaths{leftHandPath, rightHandPath}; + + XrActionSet actionSet; + XrAction gripPoseAction; + { + XrActionSetCreateInfo actionSetInfo{XR_TYPE_ACTION_SET_CREATE_INFO}; + strcpy(actionSetInfo.actionSetName, "interaction_test"); + strcpy(actionSetInfo.localizedActionSetName, "Interaction Test"); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrCreateActionSet(instance, &actionSetInfo, &actionSet)); + + XrActionCreateInfo actionInfo{XR_TYPE_ACTION_CREATE_INFO}; + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + actionInfo.actionType = XR_ACTION_TYPE_POSE_INPUT; + strcpy(actionInfo.actionName, "grip_pose"); + strcpy(actionInfo.localizedActionName, "Grip pose"); + actionInfo.subactionPaths = subactionPaths.data(); + actionInfo.countSubactionPaths = (uint32_t)subactionPaths.size(); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrCreateAction(actionSet, &actionInfo, &gripPoseAction)); + } + + compositionHelper.BeginSession(); + actionLayerManager.WaitForSessionFocusWithMessage(); + + compositionHelper.GetInteractionManager().AddActionSet(actionSet); + compositionHelper.GetInteractionManager().AddActionBindings( + StringToPath(instance, "/interaction_profiles/microsoft/motion_controller"), + {{{gripPoseAction, StringToPath(instance, "/user/hand/left/input/grip")}, + {gripPoseAction, StringToPath(instance, "/user/hand/right/input/grip")}}}); + compositionHelper.GetInteractionManager().AttachActionSets(); + + XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; + XrActiveActionSet activeActionSet{actionSet}; + syncInfo.activeActionSets = &activeActionSet; + syncInfo.countActiveActionSets = 1; + + XrActionSpaceCreateInfo actionSpaceCreateInfo{XR_TYPE_ACTION_SPACE_CREATE_INFO}; + actionSpaceCreateInfo.poseInActionSpace = XrPosef{Quat::Identity, {0, 0, 0}}; + + std::vector gripSpaces; + for (std::shared_ptr controller : {leftHandInputDevice, rightHandInputDevice}) { + actionSpaceCreateInfo.subactionPath = leftHandInputDevice->TopLevelPath(); + actionSpaceCreateInfo.action = gripPoseAction; + XrSpace gripSpace; + XRC_CHECK_THROW_XRCMD(xrCreateActionSpace(compositionHelper.GetSession(), &actionSpaceCreateInfo, &gripSpace)); + gripSpaces.push_back(gripSpace); + } + + actionLayerManager.SyncActionsUntilFocusWithMessage(syncInfo); + + XrSession session = compositionHelper.GetSession(); + + std::vector modelKeys; + std::map pathsAndKeys; + // std::vector> pathsAndKeys; + + bool gotAllKeys = WaitUntilPredicateWithTimeout( + [&]() { + actionLayerManager.IterateFrame(); + + xrSyncActions(compositionHelper.GetSession(), &syncInfo); + + for (XrPath subactionPath : subactionPaths) { + if (pathsAndKeys.count(subactionPath)) { + continue; + } + XrControllerModelKeyStateMSFT modelKeyState{XR_TYPE_CONTROLLER_MODEL_KEY_STATE_MSFT}; + CHECK_RESULT_UNQUALIFIED_SUCCESS(ext.xrGetControllerModelKeyMSFT_(session, subactionPath, &modelKeyState)); + if (modelKeyState.modelKey != XR_NULL_CONTROLLER_MODEL_KEY_MSFT) { + // we got one + modelKeys.emplace_back(modelKeyState.modelKey); + pathsAndKeys.emplace(subactionPath, modelKeyState.modelKey); + // pathsAndKeys.emplace_back(subactionPath, modelKeyState.modelKey); + } + else { + } + } + + return (pathsAndKeys.size() == subactionPaths.size()); + }, + 20s, kActionWaitDelay); + + if (pathsAndKeys.empty()) { + WARN("Cannot do further testing on XR_MSFT_controller_model: no bound subaction paths have controller model keys"); + return; + } + else if (!gotAllKeys) { + WARN("Only some bound subaction paths have controller model keys"); + } + + // Check two call struct for controller model properties and states, + // plus regular two-call for the model itself + auto modelPropertiesTwoCallData = getTwoCallStructData(); + auto modelStateTwoCallData = getTwoCallStructData(); + for (auto modelKey : modelKeys) { + INFO("Model key: " << Uint64ToHexString(modelKey)); + CheckTwoCallStructConformance(modelPropertiesTwoCallData, {XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}, + "xrGetControllerModelPropertiesMSFT", false, [&](XrControllerModelPropertiesMSFT* properties) { + return ext.xrGetControllerModelPropertiesMSFT_(session, modelKey, properties); + }); + + CheckTwoCallStructConformance( + modelStateTwoCallData, {XR_TYPE_CONTROLLER_MODEL_STATE_MSFT}, "xrGetControllerModelStateMSFT", false, + [&](XrControllerModelStateMSFT* state) { return ext.xrGetControllerModelStateMSFT_(session, modelKey, state); }); + CHECK_TWO_CALL(uint8_t, {}, ext.xrLoadControllerModelMSFT_, session, modelKey); + + CheckTwoCallStructConformance(modelPropertiesTwoCallData, {XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}, + "xrGetControllerModelPropertiesMSFT", false, [&](XrControllerModelPropertiesMSFT* properties) { + return ext.xrGetControllerModelPropertiesMSFT_(session, modelKey, properties); + }); + + XrControllerModelPropertiesMSFT modelProperties{XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}; + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(ext.xrGetControllerModelPropertiesMSFT_(session, modelKey, &modelProperties)); + std::vector nodeBuffer(modelProperties.nodeCountOutput); + modelProperties.nodeCapacityInput = (uint32_t)nodeBuffer.size(); + modelProperties.nodeProperties = nodeBuffer.data(); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(ext.xrGetControllerModelPropertiesMSFT_(session, modelKey, &modelProperties)); + XrControllerModelStateMSFT modelState{XR_TYPE_CONTROLLER_MODEL_STATE_MSFT}; + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(ext.xrGetControllerModelStateMSFT_(session, modelKey, &modelState)); + uint32_t modelBufferSize; + REQUIRE_RESULT_UNQUALIFIED_SUCCESS(ext.xrLoadControllerModelMSFT_(session, modelKey, 0, &modelBufferSize, nullptr)); + std::vector modelBuffer(modelBufferSize); + REQUIRE_RESULT_UNQUALIFIED_SUCCESS( + ext.xrLoadControllerModelMSFT_(session, modelKey, modelBufferSize, &modelBufferSize, modelBuffer.data())); + //! @todo Check that the model is valid, that it contains the nodes mentioned in the properties, and that the properties list is the same length as the state list + } + + // Try inventing some model keys that will be invalid. + + for (auto modelKey : modelKeys) { + XrControllerModelKeyMSFT fakeModelKey = modelKey + 1234; + if (std::find(modelKeys.begin(), modelKeys.end(), fakeModelKey) == modelKeys.end()) { + // We invented an invalid key + INFO("Invented model key: " << Uint64ToHexString(fakeModelKey)); + ext.CheckInvalidModelKey(session, fakeModelKey); + } + } + } +} // namespace Conformance diff --git a/src/conformance/framework/CMakeLists.txt b/src/conformance/framework/CMakeLists.txt index 2281d7c8..44ca88f6 100644 --- a/src/conformance/framework/CMakeLists.txt +++ b/src/conformance/framework/CMakeLists.txt @@ -81,11 +81,8 @@ if(WIN32) target_compile_options(conformance_framework PUBLIC /Zc:wchar_t /Zc:forScope /W4 /WX /wd4996) - # Right now can't build this on MinGW because of directxcolors, etc. + # Right now can't build this on MinGW because of directxcolors, directxmath, etc. target_link_libraries(conformance_framework PUBLIC d3d11 d3d12 d3dcompiler dxgi) - else() - target_compile_definitions(conformance_framework - PUBLIC MISSING_DIRECTX_COLORS) endif() endif() diff --git a/src/conformance/framework/action_utils.cpp b/src/conformance/framework/action_utils.cpp index 8d210d0a..fddbb4d5 100644 --- a/src/conformance/framework/action_utils.cpp +++ b/src/conformance/framework/action_utils.cpp @@ -82,7 +82,6 @@ namespace Conformance const std::string message = "Waiting for " + hand + " controller to " + (expectLocatability ? "gain" : "lose") + " tracking..."; bool success = WaitWithMessage(message.c_str(), [&]() { - GetRenderLoop().IterateFrame(); REQUIRE_RESULT(xrLocateSpace(space, localSpace, GetRenderLoop().GetLastPredictedDisplayTime(), location), XR_SUCCESS); constexpr XrSpaceLocationFlags LocatableFlags = XR_SPACE_LOCATION_ORIENTATION_VALID_BIT | XR_SPACE_LOCATION_POSITION_VALID_BIT; diff --git a/src/conformance/framework/graphics_plugin_d3d11.cpp b/src/conformance/framework/graphics_plugin_d3d11.cpp index 6b2ac971..a2dee28e 100644 --- a/src/conformance/framework/graphics_plugin_d3d11.cpp +++ b/src/conformance/framework/graphics_plugin_d3d11.cpp @@ -14,7 +14,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#if defined(XR_USE_GRAPHICS_API_D3D11) && !defined(MISSING_DIRECTX_COLORS) +#if defined(XR_USE_GRAPHICS_API_D3D11) #include "graphics_plugin.h" #include "common/xr_linear.h" diff --git a/src/conformance/framework/graphics_plugin_d3d12.cpp b/src/conformance/framework/graphics_plugin_d3d12.cpp index e24c89c8..cd7b438d 100644 --- a/src/conformance/framework/graphics_plugin_d3d12.cpp +++ b/src/conformance/framework/graphics_plugin_d3d12.cpp @@ -14,7 +14,7 @@ // See the License for the specific language governing permissions and // limitations under the License. -#if defined(XR_USE_GRAPHICS_API_D3D12) && !defined(MISSING_DIRECTX_COLORS) +#if defined(XR_USE_GRAPHICS_API_D3D12) #include "graphics_plugin.h" #include "common/xr_linear.h" diff --git a/src/conformance/framework/graphics_plugin_opengl.cpp b/src/conformance/framework/graphics_plugin_opengl.cpp index 5bdc4881..3ce587ab 100644 --- a/src/conformance/framework/graphics_plugin_opengl.cpp +++ b/src/conformance/framework/graphics_plugin_opengl.cpp @@ -23,6 +23,7 @@ #include "report.h" #include "swapchain_image_data.h" #include "utilities/Geometry.h" +#include "utilities/swapchain_format_data.h" #include "utilities/swapchain_parameters.h" #include "utilities/throw_helpers.h" #include "xr_dependencies.h" @@ -37,8 +38,6 @@ #include "gfxwrapper_opengl.h" -// clang-format off - // Note: mapping of OpenXR usage flags to OpenGL // // XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT: can be bound to a framebuffer as color @@ -53,227 +52,94 @@ // Note: no GL formats are "mutableFormats" in the sense of SwapchainCreateTestParameters as this is intended for TYPELESS, // however, some are "supportsMutableFormat" -// For now don't test XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT on GL since the semantics are unclear and some runtimes don't support this flag. -#define XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT 0 - -#define XRC_ALL_CREATE_FLAGS \ -{ \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT | XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ -} - -// the app might request any combination of flags -#define XRC_COLOR_UA_COPY_SAMPLED_MUTABLE_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ -} -#define XRC_COLOR_UA_SAMPLED_MUTABLE_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ -} -#define XRC_COLOR_COPY_SAMPLED_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ -} -#define XRC_COLOR_COPY_SAMPLED_MUTABLE_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ -} -#define XRC_COLOR_SAMPLED_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ -} -#define XRC_DEPTH_COPY_SAMPLED_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ -} -#define XRC_DEPTH_SAMPLED_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT, \ - XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ -} - -#define XRC_COMPRESSED_SAMPLED_MUTABLE_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ - XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ - XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, \ -} -#define XRC_COMPRESSED_SAMPLED_USAGE_FLAGS \ -{ \ - XR_SWAPCHAIN_USAGE_SAMPLED_BIT, \ -} - -#define ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_COLOR_UA_COPY_SAMPLED_MUTABLE_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT(X) ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT2(X, #X) - -#define ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_COLOR_UA_SAMPLED_MUTABLE_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(X) ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT2(X, #X) - -#define ADD_GL_COLOR_COPY_SAMPLED_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_COLOR_COPY_SAMPLED_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COLOR_COPY_SAMPLED_FORMAT(X) ADD_GL_COLOR_COPY_SAMPLED_FORMAT2(X, #X) - -#define ADD_GL_COLOR_COPY_SAMPLED_MUTABLE_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_COLOR_COPY_SAMPLED_MUTABLE_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COLOR_COPY_SAMPLED_MUTABLE_FORMAT(X) ADD_GL_COLOR_COPY_SAMPLED_MUTABLE_FORMAT2(X, #X) - -#define ADD_GL_COLOR_SAMPLED_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_COLOR_SAMPLED_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COLOR_SAMPLED_FORMAT(X) ADD_GL_COLOR_SAMPLED_FORMAT2(X, #X) - -#define ADD_GL_DEPTH_COPY_SAMPLED_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, NO_MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_DEPTH_COPY_SAMPLED_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_DEPTH_COPY_SAMPLED_FORMAT(X) ADD_GL_DEPTH_COPY_SAMPLED_FORMAT2(X, #X) - -#define ADD_GL_DEPTH_SAMPLED_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, NO_MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, FORMAT, XRC_DEPTH_SAMPLED_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_DEPTH_SAMPLED_FORMAT(X) ADD_GL_DEPTH_SAMPLED_FORMAT2(X, #X) - -#define ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, FORMAT, XRC_COMPRESSED_SAMPLED_MUTABLE_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(X) ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT2(X, #X) - -#define ADD_GL_COMPRESSED_SAMPLED_FORMAT2(FORMAT, NAME) \ -{ \ - {FORMAT}, \ - { \ - NAME, IMMUTABLE, NO_MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, FORMAT, XRC_COMPRESSED_SAMPLED_USAGE_FLAGS, XRC_ALL_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ -} -#define ADD_GL_COMPRESSED_SAMPLED_FORMAT(X) ADD_GL_COMPRESSED_SAMPLED_FORMAT2(X, #X) - -// clang-format on - namespace Conformance { + + // Only texture formats which are in OpenGL core and which are either color or depth renderable or + // of a specific compressed format are listed below. Runtimes can support additional formats, but those + // will not get tested. + // + // For now don't test XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT on GL since the semantics are unclear and some runtimes don't support this flag. + // TODO in the future remove this workaround? +#define WORKAROUND NotMutable() + static const SwapchainFormatDataMap& GetSwapchainFormatData() + { + static SwapchainFormatDataMap map{ + XRC_SWAPCHAIN_FORMAT(GL_RGBA8).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA16).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGB10_A2).WORKAROUND.ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_R8).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R16).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG8).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG16).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGB10_A2UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R16F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG16F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGB16F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA16F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R32F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG32F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA32F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R11F_G11F_B10F).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R8I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R8UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R16I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R16UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R32I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_R32UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG8I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG8UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG16I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG16UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG32I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RG32UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA8I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA8UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA16I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA16UI).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA32I).WORKAROUND.ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGBA32UI).WORKAROUND.ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_RGBA4).WORKAROUND.NotMutable().NoUnorderedAccess().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_RGB5_A1).WORKAROUND.NotMutable().NoUnorderedAccess().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_SRGB8).WORKAROUND.NoUnorderedAccess().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_SRGB8_ALPHA8).WORKAROUND.NoUnorderedAccess().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_RGB565).WORKAROUND.NotMutable().NoUnorderedAccess().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT16).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT24).WORKAROUND.Depth().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT32F).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH24_STENCIL8).WORKAROUND.DepthStencil().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH32F_STENCIL8).WORKAROUND.DepthStencil().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_STENCIL_INDEX8).WORKAROUND.Stencil().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RED_RGTC1).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RED_RGTC1).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RG_RGTC2).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RG_RGTC2).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_BPTC_UNORM).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT).WORKAROUND.Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT).WORKAROUND.Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB8_ETC2).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ETC2).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA8_ETC2_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_R11_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_R11_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RG11_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RG11_EAC).WORKAROUND.Compressed().NotMutable().ToPair(), + }; + return map; + } + std::string glResultString(GLenum err) { switch (err) { @@ -979,129 +845,20 @@ namespace Conformance deleteGLContext(); } - // Only texture formats which are in OpenGL core and which are either color or depth renderable or - // of a specific compressed format are listed below. Runtimes can support additional formats, but those - // will not get tested. - typedef std::map SwapchainTestMap; - SwapchainTestMap openGLSwapchainTestMap{ - ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT(GL_RGBA8), - ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT(GL_RGBA16), - ADD_GL_COLOR_UA_COPY_SAMPLED_MUTABLE_FORMAT(GL_RGB10_A2), - - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R8), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R16), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG8), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG16), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGB10_A2UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R16F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG16F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGB16F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA16F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R32F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG32F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA32F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R11F_G11F_B10F), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R8I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R8UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R16I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R16UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R32I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_R32UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG8I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG8UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG16I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG16UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG32I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RG32UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA8I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA8UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA16I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA16UI), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA32I), - ADD_GL_COLOR_UA_SAMPLED_MUTABLE_FORMAT(GL_RGBA32UI), - - ADD_GL_COLOR_COPY_SAMPLED_FORMAT(GL_RGBA4), - ADD_GL_COLOR_COPY_SAMPLED_FORMAT(GL_RGB5_A1), - - ADD_GL_COLOR_COPY_SAMPLED_MUTABLE_FORMAT(GL_SRGB8), - ADD_GL_COLOR_COPY_SAMPLED_MUTABLE_FORMAT(GL_SRGB8_ALPHA8), - - ADD_GL_COLOR_SAMPLED_FORMAT(GL_RGB565), - - ADD_GL_DEPTH_COPY_SAMPLED_FORMAT(GL_DEPTH_COMPONENT16), - ADD_GL_DEPTH_COPY_SAMPLED_FORMAT(GL_DEPTH_COMPONENT24), - - ADD_GL_DEPTH_SAMPLED_FORMAT(GL_DEPTH_COMPONENT32F), - ADD_GL_DEPTH_SAMPLED_FORMAT(GL_DEPTH24_STENCIL8), - ADD_GL_DEPTH_SAMPLED_FORMAT(GL_DEPTH32F_STENCIL8), - ADD_GL_DEPTH_SAMPLED_FORMAT(GL_STENCIL_INDEX8), - - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_RED_RGTC1), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_SIGNED_RED_RGTC1), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_RG_RGTC2), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_SIGNED_RG_RGTC2), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_RGBA_BPTC_UNORM), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT), - ADD_GL_COMPRESSED_SAMPLED_MUTABLE_FORMAT(GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT), - - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_RGB8_ETC2), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_SRGB8_ETC2), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_RGBA8_ETC2_EAC), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_R11_EAC), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_SIGNED_R11_EAC), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_RG11_EAC), - ADD_GL_COMPRESSED_SAMPLED_FORMAT(GL_COMPRESSED_SIGNED_RG11_EAC), - }; - std::string OpenGLGraphicsPlugin::GetImageFormatName(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = openGLSwapchainTestMap.find(imageFormat); - - if (it != openGLSwapchainTestMap.end()) { - return it->second.imageFormatName; - } - - return std::string("unknown"); + return ::Conformance::GetImageFormatName(GetSwapchainFormatData(), imageFormat); } bool OpenGLGraphicsPlugin::IsImageFormatKnown(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = openGLSwapchainTestMap.find(imageFormat); - - return (it != openGLSwapchainTestMap.end()); + return ::Conformance::IsImageFormatKnown(GetSwapchainFormatData(), imageFormat); } bool OpenGLGraphicsPlugin::GetSwapchainCreateTestParameters(XrInstance /*instance*/, XrSession /*session*/, XrSystemId /*systemId*/, int64_t imageFormat, SwapchainCreateTestParameters* swapchainTestParameters) { - // Swapchain image format support by the runtime is specified by the xrEnumerateSwapchainFormats function. - // Runtimes should support R8G8B8A8 and R8G8B8A8 sRGB formats if possible. - - SwapchainTestMap::iterator it = openGLSwapchainTestMap.find(imageFormat); - - // Verify that the image format is known. If it's not known then this test needs to be - // updated to recognize new OpenGL formats. - CAPTURE(imageFormat); - CHECK_MSG(it != openGLSwapchainTestMap.end(), "Unknown OpenGL image format."); - if (it == openGLSwapchainTestMap.end()) { - return false; - } - - // We may now proceed with creating swapchains with the format. - SwapchainCreateTestParameters& tp = it->second; - tp.arrayCountVector = {1, 2}; - if (!tp.compressedFormat) { - tp.mipCountVector = {1, 2}; - } - else { - tp.mipCountVector = {1}; - } - - *swapchainTestParameters = tp; + *swapchainTestParameters = ::Conformance::GetSwapchainCreateTestParameters(GetSwapchainFormatData(), imageFormat); return true; } @@ -1189,8 +946,9 @@ namespace Conformance int64_t OpenGLGraphicsPlugin::SelectDepthSwapchainFormat(const int64_t* imageFormatArray, size_t count) const { // List of supported depth swapchain formats. - const std::array f{GL_DEPTH24_STENCIL8, GL_DEPTH32F_STENCIL8, GL_DEPTH_COMPONENT24, GL_DEPTH_COMPONENT32F, - GL_DEPTH_COMPONENT16}; + const std::array f{ + GL_DEPTH24_STENCIL8, GL_DEPTH32F_STENCIL8, GL_DEPTH_COMPONENT24, GL_DEPTH_COMPONENT32F, GL_DEPTH_COMPONENT16, + }; const int64_t* formatArrayEnd = imageFormatArray + count; auto it = std::find_first_of(imageFormatArray, formatArrayEnd, f.begin(), f.end()); diff --git a/src/conformance/framework/graphics_plugin_opengles.cpp b/src/conformance/framework/graphics_plugin_opengles.cpp index e6800012..5632dd29 100644 --- a/src/conformance/framework/graphics_plugin_opengles.cpp +++ b/src/conformance/framework/graphics_plugin_opengles.cpp @@ -25,6 +25,7 @@ #include "report.h" #include "swapchain_image_data.h" #include "utilities/Geometry.h" +#include "utilities/swapchain_format_data.h" #include "utilities/swapchain_parameters.h" #include "utilities/throw_helpers.h" #include "xr_dependencies.h" @@ -581,251 +582,355 @@ namespace Conformance return std::make_shared(platformPlugin); } - // Shorthand constants for usage below. - static const uint64_t XRC_COLOR_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_MUTABLE = (XRC_COLOR_TEXTURE_USAGE | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_COMPRESSED = - (XR_SWAPCHAIN_USAGE_SAMPLED_BIT); // Compressed textures can't be rendered to, so no XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT. - - static const uint64_t XRC_DEPTH_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - -#define XRC_COLOR_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ - } - -#define XRC_DEPTH_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ - } + // For now don't test XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT on GL since the semantics are unclear and some runtimes don't support this flag. + // TODO in the future remove this workaround? +#define WORKAROUND NotMutable() + typedef std::map SwapchainTestMap; -// one for texture formats which are not known to the GL.h (where a more modern header would be needed): -#define ADD_GL_COLOR_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, X, \ - {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ - } -#define ADD_GL_COLOR_FORMAT(X) ADD_GL_COLOR_FORMAT2(X, #X) - -#define ADD_GL_COLOR_COMPRESSED_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, NO_MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, X, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, \ - XRC_COLOR_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ - } -#define ADD_GL_COLOR_COMPRESSED_FORMAT(X) ADD_GL_COLOR_COMPRESSED_FORMAT2(X, #X) - -#define ADD_GL_DEPTH_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, NO_MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, X, {XRC_DEPTH_TEXTURE_USAGE}, \ - XRC_DEPTH_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ + static const SwapchainFormatDataMap& GetSwapchainFormatData() + { + static SwapchainFormatDataMap map{ + + // + // 8 bits per component + // + XRC_SWAPCHAIN_FORMAT(GL_R8).WORKAROUND.ToPair(), // 1-component, 8-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RG8).WORKAROUND.ToPair(), // 2-component, 8-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB8).WORKAROUND.ToPair(), // 3-component, 8-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGBA8).WORKAROUND.ToPair(), // 4-component, 8-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_R8_SNORM).WORKAROUND.ToPair(), // 1-component, 8-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RG8_SNORM).WORKAROUND.ToPair(), // 2-component, 8-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB8_SNORM).WORKAROUND.ToPair(), // 3-component, 8-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RGBA8_SNORM).WORKAROUND.ToPair(), // 4-component, 8-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_R8UI).WORKAROUND.ToPair(), // 1-component, 8-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RG8UI).WORKAROUND.ToPair(), // 2-component, 8-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGB8UI).WORKAROUND.ToPair(), // 3-component, 8-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA8UI).WORKAROUND.ToPair(), // 4-component, 8-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_R8I).WORKAROUND.ToPair(), // 1-component, 8-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RG8I).WORKAROUND.ToPair(), // 2-component, 8-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGB8I).WORKAROUND.ToPair(), // 3-component, 8-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA8I).WORKAROUND.ToPair(), // 4-component, 8-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_SR8).WORKAROUND.ToPair(), // 1-component, 8-bit sRGB + XRC_SWAPCHAIN_FORMAT(GL_SRG8).WORKAROUND.ToPair(), // 2-component, 8-bit sRGB + XRC_SWAPCHAIN_FORMAT(GL_SRGB8).WORKAROUND.ToPair(), // 3-component, 8-bit sRGB + XRC_SWAPCHAIN_FORMAT(GL_SRGB8_ALPHA8).WORKAROUND.ToPair(), // 4-component, 8-bit sRGB + + // + // 16 bits per component + // + XRC_SWAPCHAIN_FORMAT(GL_R16).WORKAROUND.ToPair(), // 1-component, 16-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RG16).WORKAROUND.ToPair(), // 2-component, 16-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB16).WORKAROUND.ToPair(), // 3-component, 16-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGBA16).WORKAROUND.ToPair(), // 4-component, 16-bit unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_R16_SNORM).WORKAROUND.ToPair(), // 1-component, 16-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RG16_SNORM).WORKAROUND.ToPair(), // 2-component, 16-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB16_SNORM).WORKAROUND.ToPair(), // 3-component, 16-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_RGBA16_SNORM).WORKAROUND.ToPair(), // 4-component, 16-bit signed normalized + XRC_SWAPCHAIN_FORMAT(GL_R16UI).WORKAROUND.ToPair(), // 1-component, 16-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RG16UI).WORKAROUND.ToPair(), // 2-component, 16-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGB16UI).WORKAROUND.ToPair(), // 3-component, 16-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA16UI).WORKAROUND.ToPair(), // 4-component, 16-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_R16I).WORKAROUND.ToPair(), // 1-component, 16-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RG16I).WORKAROUND.ToPair(), // 2-component, 16-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGB16I).WORKAROUND.ToPair(), // 3-component, 16-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA16I).WORKAROUND.ToPair(), // 4-component, 16-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_R16F).WORKAROUND.ToPair(), // 1-component, 16-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RG16F).WORKAROUND.ToPair(), // 2-component, 16-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RGB16F).WORKAROUND.ToPair(), // 3-component, 16-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RGBA16F).WORKAROUND.ToPair(), // 4-component, 16-bit floating-point + + // + // 32 bits per component + // + XRC_SWAPCHAIN_FORMAT(GL_R32UI).WORKAROUND.ToPair(), // 1-component, 32-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RG32UI).WORKAROUND.ToPair(), // 2-component, 32-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGB32UI).WORKAROUND.ToPair(), // 3-component, 32-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA32UI).WORKAROUND.ToPair(), // 4-component, 32-bit unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_R32I).WORKAROUND.ToPair(), // 1-component, 32-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RG32I).WORKAROUND.ToPair(), // 2-component, 32-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGB32I).WORKAROUND.ToPair(), // 3-component, 32-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_RGBA32I).WORKAROUND.ToPair(), // 4-component, 32-bit signed integer + XRC_SWAPCHAIN_FORMAT(GL_R32F).WORKAROUND.ToPair(), // 1-component, 32-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RG32F).WORKAROUND.ToPair(), // 2-component, 32-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RGB32F).WORKAROUND.ToPair(), // 3-component, 32-bit floating-point + XRC_SWAPCHAIN_FORMAT(GL_RGBA32F).WORKAROUND.ToPair(), // 4-component, 32-bit floating-point + + // + // Packed + // + XRC_SWAPCHAIN_FORMAT(GL_RGB5).WORKAROUND.ToPair(), // 3-component 5:5:5, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB565).WORKAROUND.ToPair(), // 3-component 5:6:5, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB10).WORKAROUND.ToPair(), // 3-component 10:10:10, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGBA4).WORKAROUND.ToPair(), // 4-component 4:4:4:4, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB5_A1).WORKAROUND.ToPair(), // 4-component 5:5:5:1, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB10_A2).WORKAROUND.ToPair(), // 4-component 10:10:10:2, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_RGB10_A2UI).WORKAROUND.ToPair(), // 4-component 10:10:10:2, unsigned integer + XRC_SWAPCHAIN_FORMAT(GL_R11F_G11F_B10F).WORKAROUND.ToPair(), // 3-component 11:11:10, floating-point + XRC_SWAPCHAIN_FORMAT(GL_RGB9_E5).WORKAROUND.ToPair(), // 3-component/exp 9:9:9/5, floating-point + + // + // S3TC/DXT/BC + // + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB_S3TC_DXT1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_S3TC_DXT1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus 1-bit alpha, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_S3TC_DXT5_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus line through 1D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_S3TC_DXT3_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus 4-bit alpha, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_S3TC_DXT1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus 1-bit alpha, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus line through 1D space, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 3D space plus 4-bit alpha, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_LUMINANCE_LATC1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 1D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_LUMINANCE_ALPHA_LATC2_EXT) + .Compressed() + .NotMutable() + .ToPair(), // two lines through 1D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_LUMINANCE_LATC1_EXT) + .Compressed() + .NotMutable() + .ToPair(), // line through 1D space, 4x4 blocks, signed normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_LUMINANCE_ALPHA_LATC2_EXT) + .Compressed() + .NotMutable() + .ToPair(), // two lines through 1D space, 4x4 blocks, signed normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RED_RGTC1) + .Compressed() + .NotMutable() + .ToPair(), // line through 1D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RG_RGTC2) + .Compressed() + .NotMutable() + .ToPair(), // two lines through 1D space, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RED_RGTC1) + .Compressed() + .NotMutable() + .ToPair(), // line through 1D space, 4x4 blocks, signed normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RG_RGTC2) + .Compressed() + .NotMutable() + .ToPair(), // two lines through 1D space, 4x4 blocks, signed normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT) + .Compressed() + .NotMutable() + .ToPair(), // 3-component, 4x4 blocks, unsigned floating-point + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT) + .Compressed() + .NotMutable() + .ToPair(), // 3-component, 4x4 blocks, signed floating-point + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_BPTC_UNORM) + .Compressed() + .NotMutable() + .ToPair(), // 4-component, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM).Compressed().NotMutable().ToPair(), // 4-component, 4x4 blocks, sRGB + + // + // ETC + // + XRC_SWAPCHAIN_FORMAT(GL_ETC1_RGB8_OES).Compressed().NotMutable().ToPair(), // 3-component ETC1, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB8_ETC2) + .Compressed() + .NotMutable() + .ToPair(), // 3-component ETC2, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ETC2 with 1-bit alpha, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA8_ETC2_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ETC2, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ETC2).Compressed().NotMutable().ToPair(), // 3-component ETC2, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ETC2 with 1-bit alpha, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ETC2, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_R11_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 1-component ETC, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RG11_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 2-component ETC, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_R11_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 1-component ETC, 4x4 blocks, signed normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SIGNED_RG11_EAC) + .Compressed() + .NotMutable() + .ToPair(), // 2-component ETC, 4x4 blocks, signed normalized + + // + // ASTC + // + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_4x4_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 4x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_5x4_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 5x4 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_5x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 5x5 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_6x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 6x5 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_6x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 6x6 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x5 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x6 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x8_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x8 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x5 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x6 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x8_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x8 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x10_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x10 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_12x10_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 12x10 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_RGBA_ASTC_12x12_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 12x12 blocks, unsigned normalized + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 4x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x4_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 5x4 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 5x5 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 6x5 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 6x6 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x5 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x6 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x8_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 8x8 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x5_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x5 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x6_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x6 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x8_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x8 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x10_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 10x10 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x10_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 12x10 blocks, sRGB + XRC_SWAPCHAIN_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x12_KHR) + .Compressed() + .NotMutable() + .ToPair(), // 4-component ASTC, 12x12 blocks, sRGB + + // + // Depth/stencil + // + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT16).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT24).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT32F).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH_COMPONENT32F_NV).WORKAROUND.Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_STENCIL_INDEX8).WORKAROUND.Stencil().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH24_STENCIL8).WORKAROUND.DepthStencil().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH32F_STENCIL8).WORKAROUND.DepthStencil().ToPair(), + XRC_SWAPCHAIN_FORMAT(GL_DEPTH32F_STENCIL8_NV).WORKAROUND.DepthStencil().ToPair(), + }; + return map; } -#define ADD_GL_DEPTH_FORMAT(X) ADD_GL_DEPTH_FORMAT2(X, #X) - - typedef std::map SwapchainTestMap; - SwapchainTestMap openGLESSwapchainTestMap{ - //{ {type}, { name, false (typeless), color, type, flags, flagV, arrayV, sampleV, mipV} }, - - // - // 8 bits per component - // - ADD_GL_COLOR_FORMAT(GL_R8), // 1-component, 8-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RG8), // 2-component, 8-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB8), // 3-component, 8-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGBA8), // 4-component, 8-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_R8_SNORM), // 1-component, 8-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RG8_SNORM), // 2-component, 8-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RGB8_SNORM), // 3-component, 8-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RGBA8_SNORM), // 4-component, 8-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_R8UI), // 1-component, 8-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RG8UI), // 2-component, 8-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGB8UI), // 3-component, 8-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGBA8UI), // 4-component, 8-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_R8I), // 1-component, 8-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RG8I), // 2-component, 8-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGB8I), // 3-component, 8-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGBA8I), // 4-component, 8-bit signed integer - ADD_GL_COLOR_FORMAT(GL_SR8), // 1-component, 8-bit sRGB - ADD_GL_COLOR_FORMAT(GL_SRG8), // 2-component, 8-bit sRGB - ADD_GL_COLOR_FORMAT(GL_SRGB8), // 3-component, 8-bit sRGB - ADD_GL_COLOR_FORMAT(GL_SRGB8_ALPHA8), // 4-component, 8-bit sRGB - - // - // 16 bits per component - // - ADD_GL_COLOR_FORMAT(GL_R16), // 1-component, 16-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RG16), // 2-component, 16-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB16), // 3-component, 16-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGBA16), // 4-component, 16-bit unsigned normalized - ADD_GL_COLOR_FORMAT(GL_R16_SNORM), // 1-component, 16-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RG16_SNORM), // 2-component, 16-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RGB16_SNORM), // 3-component, 16-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_RGBA16_SNORM), // 4-component, 16-bit signed normalized - ADD_GL_COLOR_FORMAT(GL_R16UI), // 1-component, 16-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RG16UI), // 2-component, 16-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGB16UI), // 3-component, 16-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGBA16UI), // 4-component, 16-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_R16I), // 1-component, 16-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RG16I), // 2-component, 16-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGB16I), // 3-component, 16-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGBA16I), // 4-component, 16-bit signed integer - ADD_GL_COLOR_FORMAT(GL_R16F), // 1-component, 16-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RG16F), // 2-component, 16-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RGB16F), // 3-component, 16-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RGBA16F), // 4-component, 16-bit floating-point - - // - // 32 bits per component - // - ADD_GL_COLOR_FORMAT(GL_R32UI), // 1-component, 32-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RG32UI), // 2-component, 32-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGB32UI), // 3-component, 32-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_RGBA32UI), // 4-component, 32-bit unsigned integer - ADD_GL_COLOR_FORMAT(GL_R32I), // 1-component, 32-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RG32I), // 2-component, 32-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGB32I), // 3-component, 32-bit signed integer - ADD_GL_COLOR_FORMAT(GL_RGBA32I), // 4-component, 32-bit signed integer - ADD_GL_COLOR_FORMAT(GL_R32F), // 1-component, 32-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RG32F), // 2-component, 32-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RGB32F), // 3-component, 32-bit floating-point - ADD_GL_COLOR_FORMAT(GL_RGBA32F), // 4-component, 32-bit floating-point - - // - // Packed - // - ADD_GL_COLOR_FORMAT(GL_RGB5), // 3-component 5:5:5, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB565), // 3-component 5:6:5, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB10), // 3-component 10:10:10, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGBA4), // 4-component 4:4:4:4, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB5_A1), // 4-component 5:5:5:1, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB10_A2), // 4-component 10:10:10:2, unsigned normalized - ADD_GL_COLOR_FORMAT(GL_RGB10_A2UI), // 4-component 10:10:10:2, unsigned integer - ADD_GL_COLOR_FORMAT(GL_R11F_G11F_B10F), // 3-component 11:11:10, floating-point - ADD_GL_COLOR_FORMAT(GL_RGB9_E5), // 3-component/exp 9:9:9/5, floating-point - - // - // S3TC/DXT/BC - // - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGB_S3TC_DXT1_EXT), // line through 3D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_RGBA_S3TC_DXT1_EXT), // line through 3D space plus 1-bit alpha, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_RGBA_S3TC_DXT5_EXT), // line through 3D space plus line through 1D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_RGBA_S3TC_DXT3_EXT), // line through 3D space plus 4-bit alpha, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB_S3TC_DXT1_EXT), // line through 3D space, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT1_EXT), // line through 3D space plus 1-bit alpha, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT3_EXT), // line through 3D space plus line through 1D space, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB_ALPHA_S3TC_DXT5_EXT), // line through 3D space plus 4-bit alpha, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_LUMINANCE_LATC1_EXT), // line through 1D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_LUMINANCE_ALPHA_LATC2_EXT), // two lines through 1D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SIGNED_LUMINANCE_LATC1_EXT), // line through 1D space, 4x4 blocks, signed normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_SIGNED_LUMINANCE_ALPHA_LATC2_EXT), // two lines through 1D space, 4x4 blocks, signed normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RED_RGTC1), // line through 1D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RG_RGTC2), // two lines through 1D space, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SIGNED_RED_RGTC1), // line through 1D space, 4x4 blocks, signed normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SIGNED_RG_RGTC2), // two lines through 1D space, 4x4 blocks, signed normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGB_BPTC_UNSIGNED_FLOAT), // 3-component, 4x4 blocks, unsigned floating-point - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGB_BPTC_SIGNED_FLOAT), // 3-component, 4x4 blocks, signed floating-point - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_BPTC_UNORM), // 4-component, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB_ALPHA_BPTC_UNORM), // 4-component, 4x4 blocks, sRGB - - // - // ETC - // - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_ETC1_RGB8_OES), // 3-component ETC1, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGB8_ETC2), // 3-component ETC2, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_RGB8_PUNCHTHROUGH_ALPHA1_ETC2), // 4-component ETC2 with 1-bit alpha, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA8_ETC2_EAC), // 4-component ETC2, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ETC2), // 3-component ETC2, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT( - GL_COMPRESSED_SRGB8_PUNCHTHROUGH_ALPHA1_ETC2), // 4-component ETC2 with 1-bit alpha, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ETC2_EAC), // 4-component ETC2, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_R11_EAC), // 1-component ETC, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RG11_EAC), // 2-component ETC, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SIGNED_R11_EAC), // 1-component ETC, 4x4 blocks, signed normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SIGNED_RG11_EAC), // 2-component ETC, 4x4 blocks, signed normalized - - // - // ASTC - // - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_4x4_KHR), // 4-component ASTC, 4x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_5x4_KHR), // 4-component ASTC, 5x4 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_5x5_KHR), // 4-component ASTC, 5x5 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_6x5_KHR), // 4-component ASTC, 6x5 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_6x6_KHR), // 4-component ASTC, 6x6 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x5_KHR), // 4-component ASTC, 8x5 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x6_KHR), // 4-component ASTC, 8x6 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_8x8_KHR), // 4-component ASTC, 8x8 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x5_KHR), // 4-component ASTC, 10x5 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x6_KHR), // 4-component ASTC, 10x6 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x8_KHR), // 4-component ASTC, 10x8 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_10x10_KHR), // 4-component ASTC, 10x10 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_12x10_KHR), // 4-component ASTC, 12x10 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_RGBA_ASTC_12x12_KHR), // 4-component ASTC, 12x12 blocks, unsigned normalized - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_4x4_KHR), // 4-component ASTC, 4x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x4_KHR), // 4-component ASTC, 5x4 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_5x5_KHR), // 4-component ASTC, 5x5 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x5_KHR), // 4-component ASTC, 6x5 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_6x6_KHR), // 4-component ASTC, 6x6 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x5_KHR), // 4-component ASTC, 8x5 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x6_KHR), // 4-component ASTC, 8x6 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_8x8_KHR), // 4-component ASTC, 8x8 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x5_KHR), // 4-component ASTC, 10x5 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x6_KHR), // 4-component ASTC, 10x6 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x8_KHR), // 4-component ASTC, 10x8 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_10x10_KHR), // 4-component ASTC, 10x10 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x10_KHR), // 4-component ASTC, 12x10 blocks, sRGB - ADD_GL_COLOR_COMPRESSED_FORMAT(GL_COMPRESSED_SRGB8_ALPHA8_ASTC_12x12_KHR), // 4-component ASTC, 12x12 blocks, sRGB - - // - // Depth/stencil - // - ADD_GL_DEPTH_FORMAT(GL_DEPTH_COMPONENT16), ADD_GL_DEPTH_FORMAT(GL_DEPTH_COMPONENT24), ADD_GL_DEPTH_FORMAT(GL_DEPTH_COMPONENT32F), - ADD_GL_DEPTH_FORMAT(GL_DEPTH_COMPONENT32F_NV), ADD_GL_DEPTH_FORMAT(GL_STENCIL_INDEX8), ADD_GL_DEPTH_FORMAT(GL_DEPTH24_STENCIL8), - ADD_GL_DEPTH_FORMAT(GL_DEPTH32F_STENCIL8), ADD_GL_DEPTH_FORMAT(GL_DEPTH32F_STENCIL8_NV)}; -#undef ADD_GL_COLOR_FORMAT -#undef ADD_GL_COLOR_FORMAT2 -#undef ADD_GL_DEPTH_FORMAT -#undef ADD_GL_DEPTH_FORMAT2 // Returns a name for an image format. std::string OpenGLESGraphicsPlugin::GetImageFormatName(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = openGLESSwapchainTestMap.find(imageFormat); - - if (it != openGLESSwapchainTestMap.end()) { - return it->second.imageFormatName; - } - - return std::string("unknown"); + return ::Conformance::GetImageFormatName(GetSwapchainFormatData(), imageFormat); } bool OpenGLESGraphicsPlugin::IsImageFormatKnown(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = openGLESSwapchainTestMap.find(imageFormat); - - return (it != openGLESSwapchainTestMap.end()); + return ::Conformance::IsImageFormatKnown(GetSwapchainFormatData(), imageFormat); } // Retrieves SwapchainCreateTestParameters for the caller, handling plaform-specific functionality @@ -838,29 +943,7 @@ namespace Conformance // Swapchain image format support by the runtime is specified by the xrEnumerateSwapchainFormats function. // Runtimes should support R8G8B8A8 and R8G8B8A8 sRGB formats if possible. - SwapchainTestMap::iterator it = openGLESSwapchainTestMap.find(imageFormat); - - // Verify that the image format is known. If it's not known then this test needs to be - // updated to recognize new OpenGL formats. - CAPTURE(imageFormat); - CHECK_MSG(it != openGLESSwapchainTestMap.end(), "Unknown OpenGLES image format."); - if (it == openGLESSwapchainTestMap.end()) { - return false; - } - - CAPTURE(it->second.imageFormatName); - - // We may now proceed with creating swapchains with the format. - SwapchainCreateTestParameters& tp = it->second; - tp.arrayCountVector = {1, 2}; - if (!tp.compressedFormat) { - tp.mipCountVector = {1, 2}; - } - else { - tp.mipCountVector = {1}; - } - - *swapchainTestParameters = tp; + *swapchainTestParameters = ::Conformance::GetSwapchainCreateTestParameters(GetSwapchainFormatData(), imageFormat); return true; } @@ -926,7 +1009,18 @@ namespace Conformance int64_t OpenGLESGraphicsPlugin::SelectColorSwapchainFormat(const int64_t* imageFormatArray, size_t count) const { // List of supported color swapchain formats. - const std::array f{GL_RGBA8, GL_SRGB8_ALPHA8}; + // The order of this list does not effect the priority of selecting formats, the runtime list defines that. + const std::array f{ + GL_RGB10_A2, + GL_RGBA16, + GL_RGBA16F, + GL_RGBA32F, + + // The two below should only be used as a fallback, as they are linear color formats without enough bits for color + // depth, thus leading to banding. + GL_RGBA8, + GL_SRGB8_ALPHA8, + }; const int64_t* formatArrayEnd = imageFormatArray + count; auto it = std::find_first_of(imageFormatArray, formatArrayEnd, f.begin(), f.end()); @@ -942,7 +1036,12 @@ namespace Conformance int64_t OpenGLESGraphicsPlugin::SelectDepthSwapchainFormat(const int64_t* imageFormatArray, size_t count) const { // List of supported depth swapchain formats. - const std::array f{GL_DEPTH24_STENCIL8, GL_DEPTH_COMPONENT24, GL_DEPTH_COMPONENT16, GL_DEPTH_COMPONENT32F}; + const std::array f{ + GL_DEPTH24_STENCIL8, + GL_DEPTH_COMPONENT24, + GL_DEPTH_COMPONENT16, + GL_DEPTH_COMPONENT32F, + }; const int64_t* formatArrayEnd = imageFormatArray + count; auto it = std::find_first_of(imageFormatArray, formatArrayEnd, f.begin(), f.end()); diff --git a/src/conformance/framework/graphics_plugin_vulkan.cpp b/src/conformance/framework/graphics_plugin_vulkan.cpp index 92ca5ddc..4db260a5 100644 --- a/src/conformance/framework/graphics_plugin_vulkan.cpp +++ b/src/conformance/framework/graphics_plugin_vulkan.cpp @@ -14,17 +14,17 @@ // See the License for the specific language governing permissions and // limitations under the License. -#include "graphics_plugin.h" - #ifdef XR_USE_GRAPHICS_API_VULKAN #include "RGBAImage.h" #include "common/hex_and_handles.h" #include "common/xr_linear.h" +#include "graphics_plugin.h" #include "graphics_plugin_impl_helpers.h" #include "report.h" #include "swapchain_image_data.h" #include "utilities/Geometry.h" +#include "utilities/swapchain_format_data.h" #include "utilities/swapchain_parameters.h" #include "utilities/throw_helpers.h" #include "utilities/vulkan_utils.h" @@ -41,6 +41,7 @@ #include #include #include +#include #include #include #include @@ -175,12 +176,14 @@ namespace Conformance VulkanSwapchainImageData::Reset(); } - void BindRenderTarget(uint32_t index, uint32_t arraySlice, const VkRect2D& renderArea, VkRenderPassBeginInfo* renderPassBeginInfo) + void BindRenderTarget(uint32_t index, uint32_t arraySlice, const VkRect2D& renderArea, VkImageAspectFlags secondAttachmentAspect, + VkRenderPassBeginInfo* renderPassBeginInfo) { RenderTarget& rt = m_slices[arraySlice].m_renderTarget[index]; RenderPass& rp = m_slices[arraySlice].m_rp; if (rt.fb == VK_NULL_HANDLE) { - rt.Create(m_namer, m_vkDevice, GetTypedImage(index).image, GetDepthImageForColorIndex(index).image, arraySlice, m_size, rp); + rt.Create(m_namer, m_vkDevice, GetTypedImage(index).image, GetDepthImageForColorIndex(index).image, secondAttachmentAspect, + arraySlice, m_size, rp); } renderPassBeginInfo->renderPass = rp.pass; renderPassBeginInfo->framebuffer = rt.fb; @@ -206,6 +209,11 @@ namespace Conformance SwapchainImageDataBase::Reset(); } + int64_t GetDepthFormat() const + { + return m_depthFormat; + } + protected: const XrSwapchainImageVulkanKHR& GetFallbackDepthSwapchainImage(uint32_t i) override { @@ -285,6 +293,10 @@ namespace Conformance hWnd = nullptr; UnregisterClassW(L"conformance_test", hInst); } + if (hUser32Dll) { + ::FreeLibrary(hUser32Dll); + hUser32Dll = nullptr; + } #endif m_vkDevice = nullptr; @@ -299,6 +311,7 @@ namespace Conformance #if defined(VK_USE_PLATFORM_WIN32_KHR) HINSTANCE hInst{NULL}; HWND hWnd{NULL}; + HINSTANCE hUser32Dll{NULL}; #endif const VkExtent2D size{640, 480}; VkInstance m_vkInstance{VK_NULL_HANDLE}; @@ -328,8 +341,13 @@ namespace Conformance // adjust the window size and show at InitDevice time #if defined(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2) - // Make sure we're 1:1 for HMD pixels - SetThreadDpiAwarenessContext(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2); + typedef DPI_AWARENESS_CONTEXT(WINAPI * PFN_SetThreadDpiAwarenessContext)(DPI_AWARENESS_CONTEXT); + hUser32Dll = ::LoadLibraryA("user32.dll"); + if (PFN_SetThreadDpiAwarenessContext SetThreadDpiAwarenessContextFn = + reinterpret_cast(::GetProcAddress(hUser32Dll, "SetThreadDpiAwarenessContext"))) { + // Make sure we're 1:1 for HMD pixels + SetThreadDpiAwarenessContextFn(DPI_AWARENESS_CONTEXT_PER_MONITOR_AWARE_V2); + } #endif RECT rect{0, 0, (LONG)size.width, (LONG)size.height}; AdjustWindowRect(&rect, WS_OVERLAPPEDWINDOW, false); @@ -619,6 +637,9 @@ namespace Conformance void RenderView(const XrCompositionLayerProjectionView& layerView, const XrSwapchainImageBaseHeader* colorSwapchainImage, const RenderParams& params) override; + /// Get data on a known swapchain format + const SwapchainFormatData& FindFormatData(int64_t format) const; + #if defined(USE_CHECKPOINTS) void Checkpoint(std::string msg) { @@ -1348,288 +1369,182 @@ namespace Conformance return nullptr; } - // Shorthand constants for usage below. - static const uint64_t XRC_COLOR_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_MUTABLE = (XRC_COLOR_TEXTURE_USAGE | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_COMPRESSED = - (XR_SWAPCHAIN_USAGE_SAMPLED_BIT); // Compressed textures can't be rendered to, so no XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT. - - static const uint64_t XRC_DEPTH_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - -#define XRC_COLOR_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ - } + static const SwapchainFormatDataMap& GetSwapchainFormatData() + { -#define XRC_DEPTH_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ + // Add SwapchainCreateTestParameters for other Vulkan formats if they are supported by a runtime + static SwapchainFormatDataMap map{{ + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8A8_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8A8_SRGB).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_B8G8R8A8_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_B8G8R8A8_SRGB).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8_SRGB).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_B8G8R8_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_B8G8R8_SRGB).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8_UNORM).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_UNORM).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8A8_SNORM).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8A8_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8G8B8A8_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R8_SRGB).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16_UNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_UNORM).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16_SNORM).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_SNORM).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_UINT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_SINT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_SFLOAT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32_SINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32A32_SINT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32_UINT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32A32_UINT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32_SFLOAT).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R32G32B32A32_SFLOAT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R5G5B5A1_UNORM_PACK16).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R5G6B5_UNORM_PACK16).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A2B10G10R10_UNORM_PACK32).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R4G4B4A4_UNORM_PACK16).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A1R5G5B5_UNORM_PACK16).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A2R10G10B10_UNORM_PACK32).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A2R10G10B10_UINT_PACK32).ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A2B10G10R10_UNORM_PACK32).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_A2B10G10R10_UINT_PACK32).ToPair(), + // Runtimes with D3D11 back-ends map VK_FORMAT_B10G11R11_UFLOAT_PACK32 to DXGI_FORMAT_R11G11B10_FLOAT and that format doesn't have a TYPELESS equivalent. + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_B10G11R11_UFLOAT_PACK32).NotMutable().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_E5B9G9R9_UFLOAT_PACK32).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_R16G16B16A16_SFLOAT).ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_D16_UNORM).Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_D24_UNORM_S8_UINT).DepthStencil().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_X8_D24_UNORM_PACK32).Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_S8_UINT).Stencil().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_D32_SFLOAT).Depth().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_D32_SFLOAT_S8_UINT).DepthStencil().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8A1_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8A8_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8A1_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ETC2_R8G8B8A8_SRGB_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_EAC_R11_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_EAC_R11G11_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_EAC_R11_SNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_EAC_R11G11_SNORM_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_4x4_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_5x4_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_5x5_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_6x5_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_6x6_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x5_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x6_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x8_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x5_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x6_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x8_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x10_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_12x10_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_12x12_UNORM_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_4x4_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_5x4_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_5x5_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_6x5_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_6x6_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x5_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x6_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_8x8_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x5_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x6_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x8_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_10x10_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_12x10_SRGB_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_ASTC_12x12_SRGB_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC1_RGBA_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC1_RGBA_SRGB_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC2_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC2_SRGB_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC3_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC3_SRGB_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC6H_UFLOAT_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC6H_SFLOAT_BLOCK).Compressed().ToPair(), + + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC7_UNORM_BLOCK).Compressed().ToPair(), + XRC_SWAPCHAIN_FORMAT(VK_FORMAT_BC7_SRGB_BLOCK).Compressed().ToPair(), + }}; + return map; } -#define ADD_VK_COLOR_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, X, \ - {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ - } -#define ADD_VK_COLOR_FORMAT(X) ADD_VK_COLOR_FORMAT2(X, #X) - -#define ADD_VK_COLOR_IMMUTABLE_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, X, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, \ - {}, {}, \ - { \ - } \ - } \ - } -#define ADD_VK_COLOR_IMMUTABLE_FORMAT(X) ADD_VK_COLOR_IMMUTABLE_FORMAT2(X, #X) - -#define ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, X, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, \ - XRC_COLOR_CREATE_FLAGS, {}, {}, \ - { \ - } \ - } \ - } -#define ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(X) ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT2(X, #X) - -#define ADD_VK_DEPTH_FORMAT2(X, Y) \ - { \ - {X}, \ - { \ - Y, IMMUTABLE, MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, X, {XRC_DEPTH_TEXTURE_USAGE}, XRC_DEPTH_CREATE_FLAGS, \ - {}, {}, \ - { \ - } \ - } \ - } -#define ADD_VK_DEPTH_FORMAT(X) ADD_VK_DEPTH_FORMAT2(X, #X) - - // clang-format off - // Add SwapchainCreateTestParameters for other Vulkan formats if they are supported by a runtime - typedef std::map SwapchainTestMap; - SwapchainTestMap vkSwapchainTestMap{ - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8A8_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8A8_SRGB), - ADD_VK_COLOR_FORMAT(VK_FORMAT_B8G8R8A8_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_B8G8R8A8_SRGB), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8_SRGB), - ADD_VK_COLOR_FORMAT(VK_FORMAT_B8G8R8_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_B8G8R8_SRGB), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8_UNORM), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_UNORM), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8A8_SNORM), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8A8_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8G8B8A8_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R8_SRGB), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16_UNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_UNORM), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16_SNORM), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_SNORM), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_UINT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_SINT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_SFLOAT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32_SINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32A32_SINT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32_UINT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32A32_UINT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32_SFLOAT), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R32G32B32A32_SFLOAT), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R5G5B5A1_UNORM_PACK16), - ADD_VK_COLOR_FORMAT(VK_FORMAT_R5G6B5_UNORM_PACK16), - ADD_VK_COLOR_FORMAT(VK_FORMAT_A2B10G10R10_UNORM_PACK32), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R4G4B4A4_UNORM_PACK16), - ADD_VK_COLOR_FORMAT(VK_FORMAT_A1R5G5B5_UNORM_PACK16), - ADD_VK_COLOR_FORMAT(VK_FORMAT_A2R10G10B10_UNORM_PACK32), - ADD_VK_COLOR_FORMAT(VK_FORMAT_A2R10G10B10_UINT_PACK32), - ADD_VK_COLOR_FORMAT(VK_FORMAT_A2B10G10R10_UNORM_PACK32), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_A2B10G10R10_UINT_PACK32), - // Runtimes with D3D11 back-ends map VK_FORMAT_B10G11R11_UFLOAT_PACK32 to DXGI_FORMAT_R11G11B10_FLOAT and that format doesn't have a TYPELESS equivalent. - //ADD_VK_COLOR_FORMAT(VK_FORMAT_B10G11R11_UFLOAT_PACK32), - ADD_VK_COLOR_IMMUTABLE_FORMAT(VK_FORMAT_B10G11R11_UFLOAT_PACK32), - ADD_VK_COLOR_FORMAT(VK_FORMAT_E5B9G9R9_UFLOAT_PACK32), - - ADD_VK_COLOR_FORMAT(VK_FORMAT_R16G16B16A16_SFLOAT), - - ADD_VK_DEPTH_FORMAT(VK_FORMAT_D16_UNORM), - ADD_VK_DEPTH_FORMAT(VK_FORMAT_D24_UNORM_S8_UINT), - - ADD_VK_DEPTH_FORMAT(VK_FORMAT_X8_D24_UNORM_PACK32), - ADD_VK_DEPTH_FORMAT(VK_FORMAT_S8_UINT), - - ADD_VK_DEPTH_FORMAT(VK_FORMAT_D32_SFLOAT), - ADD_VK_DEPTH_FORMAT(VK_FORMAT_D32_SFLOAT_S8_UINT), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8A1_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8A8_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8A1_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ETC2_R8G8B8A8_SRGB_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_EAC_R11_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_EAC_R11G11_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_EAC_R11_SNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_EAC_R11G11_SNORM_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_4x4_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_5x4_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_5x5_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_6x5_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_6x6_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x5_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x6_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x8_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x5_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x6_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x8_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x10_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_12x10_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_12x12_UNORM_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_4x4_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_5x4_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_5x5_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_6x5_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_6x6_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x5_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x6_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_8x8_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x5_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x6_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x8_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_10x10_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_12x10_SRGB_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_ASTC_12x12_SRGB_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC1_RGBA_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC1_RGBA_SRGB_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC2_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC2_SRGB_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC3_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC3_SRGB_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC6H_UFLOAT_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC6H_SFLOAT_BLOCK), - - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC7_UNORM_BLOCK), - ADD_VK_COLOR_COMPRESSED_UNRENDERABLE_FORMAT(VK_FORMAT_BC7_SRGB_BLOCK), - }; - // clang-format on - std::string VulkanGraphicsPlugin::GetImageFormatName(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = vkSwapchainTestMap.find(imageFormat); - - if (it != vkSwapchainTestMap.end()) { - return it->second.imageFormatName; - } - - return std::string("unknown"); + return ::Conformance::GetImageFormatName(GetSwapchainFormatData(), imageFormat); } bool VulkanGraphicsPlugin::IsImageFormatKnown(int64_t imageFormat) const { - SwapchainTestMap::const_iterator it = vkSwapchainTestMap.find(imageFormat); - - return (it != vkSwapchainTestMap.end()); + return ::Conformance::IsImageFormatKnown(GetSwapchainFormatData(), imageFormat); } bool VulkanGraphicsPlugin::GetSwapchainCreateTestParameters(XrInstance /*instance*/, XrSession /*session*/, XrSystemId /*systemId*/, int64_t imageFormat, SwapchainCreateTestParameters* swapchainTestParameters) { - // Swapchain image format support by the runtime is specified by the xrEnumerateSwapchainFormats function. - // Runtimes should support R8G8B8A8 and R8G8B8A8 sRGB formats if possible. - - SwapchainTestMap::iterator it = vkSwapchainTestMap.find(imageFormat); - - // Verify that the image format is known. If it's not known then this test needs to be - // updated to recognize new DXGI formats. - CAPTURE(imageFormat); - XRC_CHECK_THROW_MSG(it != vkSwapchainTestMap.end(), "Unknown Vulkan image format."); - if (it == vkSwapchainTestMap.end()) { - return false; - } - - // Verify that imageFormat is not a typeless type. Only regular types are allowed to - // be returned by the runtime for enumerated image formats. Note Vulkan doesn't really - // have a "typeless" format to worry about so this should never be hit. - CAPTURE(it->second.imageFormatName); - XRC_CHECK_THROW_MSG(!it->second.mutableFormat, "Typeless Vulkan image formats must not be enumerated by runtimes."); - if (it->second.mutableFormat) { - return false; - } - - // We may now proceed with creating swapchains with the format. - SwapchainCreateTestParameters& tp = it->second; - tp.arrayCountVector = {1, 2}; - if (tp.colorFormat && !tp.compressedFormat) { - tp.mipCountVector = {1, 2}; - } - else { - tp.mipCountVector = {1}; - } - - *swapchainTestParameters = tp; + *swapchainTestParameters = ::Conformance::GetSwapchainCreateTestParameters(GetSwapchainFormatData(), imageFormat); return true; } @@ -1692,10 +1607,10 @@ namespace Conformance const std::array f{VK_FORMAT_R8G8B8A8_SRGB, VK_FORMAT_B8G8R8A8_SRGB, VK_FORMAT_R8G8B8A8_UNORM, VK_FORMAT_B8G8R8A8_UNORM}; - const int64_t* formatArrayEnd = formatArray + count; - auto it = std::find_first_of(formatArray, formatArrayEnd, f.begin(), f.end()); + span formatArraySpan{formatArray, count}; + auto it = std::find_first_of(formatArraySpan.begin(), formatArraySpan.end(), f.begin(), f.end()); - if (it == formatArrayEnd) { + if (it == formatArraySpan.end()) { assert(false); // Assert instead of throw as we need to switch to the big table which can't fail. return formatArray[0]; } @@ -1710,10 +1625,10 @@ namespace Conformance const std::array f{VK_FORMAT_D32_SFLOAT, VK_FORMAT_D24_UNORM_S8_UINT, VK_FORMAT_D16_UNORM, VK_FORMAT_D32_SFLOAT_S8_UINT}; - const int64_t* formatArrayEnd = formatArray + count; - auto it = std::find_first_of(formatArray, formatArrayEnd, f.begin(), f.end()); + span formatArraySpan{formatArray, count}; + auto it = std::find_first_of(formatArraySpan.begin(), formatArraySpan.end(), f.begin(), f.end()); - if (it == formatArrayEnd) { + if (it == formatArraySpan.end()) { assert(false); // Assert instead of throw as we need to switch to the big table which can't fail. return formatArray[0]; } @@ -1892,6 +1807,45 @@ namespace Conformance vkCmdSetScissor(m_cmdBuffer.buf, 0, 1, &rect); } + /// Compute image layout for the "second image" format (depth and/or stencil) + static inline VkImageLayout ComputeLayout(const SwapchainFormatData& secondFormatData) + { + // can't use VK_IMAGE_LAYOUT_DEPTH_ATTACHMENT_OPTIMAL / + // VK_IMAGE_LAYOUT_STENCIL_ATTACHMENT_OPTIMAL + // because they're Vulkan 1.2 + if (secondFormatData.GetDepthFormat() || secondFormatData.GetStencilFormat()) { + return VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL; + } + if (secondFormatData.GetColorFormat()) { + return VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL; + } + throw std::runtime_error("No idea what layout to use for depth/stencil image"); + } + + /// Compute image aspect flags for the "second image" format (depth and/or stencil) + static inline VkImageAspectFlags ComputeAspectFlags(const SwapchainFormatData& secondFormatData) + { + VkImageAspectFlags secondAttachmentAspect = 0; + if (secondFormatData.GetDepthFormat()) { + secondAttachmentAspect |= VK_IMAGE_ASPECT_DEPTH_BIT; + } + if (secondFormatData.GetStencilFormat()) { + secondAttachmentAspect |= VK_IMAGE_ASPECT_STENCIL_BIT; + } + if (secondFormatData.GetColorFormat()) { + throw std::logic_error("this is just for the second image"); + } + return secondAttachmentAspect; + } + + const SwapchainFormatData& VulkanGraphicsPlugin::FindFormatData(int64_t format) const + { + auto& vkFormatMap = GetSwapchainFormatData(); + auto it = vkFormatMap.find(format); + assert(it != vkFormatMap.end()); + return it->second; + } + void VulkanGraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f bgColor) { @@ -1906,13 +1860,20 @@ namespace Conformance VkRect2D renderArea = {{0, 0}, {swapchainData->Width(), swapchainData->Height()}}; SetViewportAndScissor(renderArea); + // may be depth, stencil, or both + int64_t secondImageFormat = swapchainData->GetDepthFormat(); + const SwapchainFormatData& secondFormatData = FindFormatData(secondImageFormat); + + VkImageAspectFlags secondAttachmentAspect = ComputeAspectFlags(secondFormatData); + // Bind eye render target VkRenderPassBeginInfo renderPassBeginInfo{VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO}; - swapchainData->BindRenderTarget(imageIndex, imageArrayIndex, renderArea, &renderPassBeginInfo); + swapchainData->BindRenderTarget(imageIndex, imageArrayIndex, renderArea, secondAttachmentAspect, &renderPassBeginInfo); if (!swapchainData->DepthSwapchainEnabled()) { - // Ensure depth is in the right layout - swapchainData->TransitionLayout(imageIndex, &m_cmdBuffer, VK_IMAGE_LAYOUT_DEPTH_STENCIL_ATTACHMENT_OPTIMAL); + // Ensure self-made fallback depth is in the right layout + VkImageLayout layout = ComputeLayout(secondFormatData); + swapchainData->TransitionLayout(imageIndex, &m_cmdBuffer, layout); } vkCmdBeginRenderPass(m_cmdBuffer.buf, &renderPassBeginInfo, VK_SUBPASS_CONTENTS_INLINE); @@ -1929,7 +1890,7 @@ namespace Conformance clearValues[1].depthStencil.stencil = 0; std::array clearAttachments{{ {VK_IMAGE_ASPECT_COLOR_BIT, 0, clearValues[0]}, - {VK_IMAGE_ASPECT_DEPTH_BIT, 0, clearValues[1]}, + {secondAttachmentAspect, 0, clearValues[1]}, }}; // imageArrayIndex already included in the VkImageView VkClearRect clearRect{renderArea, 0, 1}; @@ -1968,13 +1929,18 @@ namespace Conformance VkRect2D renderArea = {{r.offset.x, r.offset.y}, {uint32_t(r.extent.width), uint32_t(r.extent.height)}}; SetViewportAndScissor(renderArea); + // may be depth, stencil, or both + int64_t secondImageFormat = swapchainData->GetDepthFormat(); + const SwapchainFormatData& secondFormatData = FindFormatData(secondImageFormat); + VkImageAspectFlags secondAttachmentAspect = ComputeAspectFlags(secondFormatData); + // Just bind the eye render target, ClearImageSlice will have cleared it. VkRenderPassBeginInfo renderPassBeginInfo{VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO}; // aka slice auto imageArrayIndex = layerView.subImage.imageArrayIndex; - swapchainData->BindRenderTarget(imageIndex, imageArrayIndex, renderArea, &renderPassBeginInfo); + swapchainData->BindRenderTarget(imageIndex, imageArrayIndex, renderArea, secondAttachmentAspect, &renderPassBeginInfo); vkCmdBeginRenderPass(m_cmdBuffer.buf, &renderPassBeginInfo, VK_SUBPASS_CONTENTS_INLINE); diff --git a/src/conformance/framework/two_call_struct_metadata.h b/src/conformance/framework/two_call_struct_metadata.h index 9d929ec2..f256ebf4 100644 --- a/src/conformance/framework/two_call_struct_metadata.h +++ b/src/conformance/framework/two_call_struct_metadata.h @@ -14,14 +14,38 @@ namespace Conformance static inline auto getTwoCallStructData(const XrVisibilityMaskKHR&) { - XrVisibilityMaskKHR visibilityMask{XR_TYPE_VISIBILITY_MASK_KHR}; - return TwoCallStruct(visibilityMask, - CapacityInputCountOutput(NAME_AND_MEMPTR(XrVisibilityMaskKHR::indexCapacityInput), - NAME_AND_MEMPTR(XrVisibilityMaskKHR::indexCountOutput)) - .Array(NAME_AND_MEMPTR(XrVisibilityMaskKHR::indices)), - CapacityInputCountOutput(NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertexCapacityInput), - NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertexCountOutput)) - .Array(NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertices))); + static const XrVisibilityMaskKHR visibilityMask{XR_TYPE_VISIBILITY_MASK_KHR}; + static const auto data = TwoCallStruct(visibilityMask, + CapacityInputCountOutput(NAME_AND_MEMPTR(XrVisibilityMaskKHR::indexCapacityInput), + NAME_AND_MEMPTR(XrVisibilityMaskKHR::indexCountOutput)) + .Array(NAME_AND_MEMPTR(XrVisibilityMaskKHR::indices)), + CapacityInputCountOutput(NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertexCapacityInput), + NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertexCountOutput)) + .Array(NAME_AND_MEMPTR(XrVisibilityMaskKHR::vertices))); + return data; + } + + /// Get the two-call-struct metadata for XrControllerModelPropertiesMSFT + static inline auto getTwoCallStructData(const XrControllerModelPropertiesMSFT&) + { + static const XrControllerModelPropertiesMSFT modelProperties{XR_TYPE_CONTROLLER_MODEL_PROPERTIES_MSFT}; + static const auto data = TwoCallStruct( + modelProperties, + CapacityInputCountOutput(NAME_AND_MEMPTR(XrControllerModelPropertiesMSFT::nodeCapacityInput), + NAME_AND_MEMPTR(XrControllerModelPropertiesMSFT::nodeCountOutput)) + .Array(NAME_AND_MEMPTR(XrControllerModelPropertiesMSFT::nodeProperties), {XR_TYPE_CONTROLLER_MODEL_NODE_PROPERTIES_MSFT})); + return data; + } + + /// Get the two-call-struct metadata for XrControllerModelStateMSFT + static inline auto getTwoCallStructData(const XrControllerModelStateMSFT&) + { + static const XrControllerModelStateMSFT modelState{XR_TYPE_CONTROLLER_MODEL_STATE_MSFT}; + static const auto data = TwoCallStruct( + modelState, CapacityInputCountOutput(NAME_AND_MEMPTR(XrControllerModelStateMSFT::nodeCapacityInput), + NAME_AND_MEMPTR(XrControllerModelStateMSFT::nodeCountOutput)) + .Array(NAME_AND_MEMPTR(XrControllerModelStateMSFT::nodeStates), {XR_TYPE_CONTROLLER_MODEL_NODE_STATE_MSFT})); + return data; } #undef NAME_AND_MEMPTR diff --git a/src/conformance/platform_specific/android_resources/drawable/ic_launcher_foreground.xml b/src/conformance/platform_specific/android_resources/drawable/ic_launcher_foreground.xml index 69c547db..ea4d6e26 100644 --- a/src/conformance/platform_specific/android_resources/drawable/ic_launcher_foreground.xml +++ b/src/conformance/platform_specific/android_resources/drawable/ic_launcher_foreground.xml @@ -5,10 +5,10 @@ android:viewportHeight="108" android:tint="#FFFFFF"> diff --git a/src/conformance/platform_specific/android_resources/mipmap-anydpi-v26/ic_launcher_round.xml b/src/conformance/platform_specific/android_resources/mipmap-anydpi-v26/ic_launcher_round.xml index 1ecaaf59..d61597bd 100644 --- a/src/conformance/platform_specific/android_resources/mipmap-anydpi-v26/ic_launcher_round.xml +++ b/src/conformance/platform_specific/android_resources/mipmap-anydpi-v26/ic_launcher_round.xml @@ -1,9 +1,9 @@ diff --git a/src/conformance/platform_specific/android_resources/values/colors.xml b/src/conformance/platform_specific/android_resources/values/colors.xml index a784fbec..c2497a05 100644 --- a/src/conformance/platform_specific/android_resources/values/colors.xml +++ b/src/conformance/platform_specific/android_resources/values/colors.xml @@ -1,6 +1,6 @@ diff --git a/src/conformance/utilities/CMakeLists.txt b/src/conformance/utilities/CMakeLists.txt index 9a86511f..99128825 100644 --- a/src/conformance/utilities/CMakeLists.txt +++ b/src/conformance/utilities/CMakeLists.txt @@ -10,6 +10,7 @@ add_library( event_reader.cpp Geometry.cpp stringification.cpp + swapchain_format_data.cpp types_and_constants.cpp utils.cpp) diff --git a/src/conformance/utilities/d3d_common.cpp b/src/conformance/utilities/d3d_common.cpp index ca878e4c..8a82a881 100644 --- a/src/conformance/utilities/d3d_common.cpp +++ b/src/conformance/utilities/d3d_common.cpp @@ -2,19 +2,18 @@ // // SPDX-License-Identifier: Apache-2.0 -#if (defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12)) && !defined(MISSING_DIRECTX_COLORS) +#if defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12) #include "d3d_common.h" -#include "swapchain_parameters.h" #include "common/xr_linear.h" +#include "swapchain_format_data.h" +#include "swapchain_parameters.h" #include "throw_helpers.h" #include // For Microsoft::WRL::ComPtr -#include -#include - +#include #include using namespace Microsoft::WRL; @@ -89,163 +88,145 @@ namespace Conformance } } - // Shorthand constants for usage below. - static const uint64_t XRC_COLOR_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_MUTABLE = (XRC_COLOR_TEXTURE_USAGE | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT); - - static const uint64_t XRC_COLOR_TEXTURE_USAGE_COMPRESSED = - (XR_SWAPCHAIN_USAGE_SAMPLED_BIT); // Compressed textures can't be rendered to, so no XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT. - - static const uint64_t XRC_DEPTH_TEXTURE_USAGE = (XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT); - -#define XRC_COLOR_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ - } - -#define XRC_DEPTH_CREATE_FLAGS \ - { \ - 0, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT \ - } - SwapchainTestMap& GetDxgiSwapchainTestMap() { - // clang-format off static SwapchainTestMap dxgiSwapchainTestMap{ - {{DXGI_FORMAT_R32G32B32A32_TYPELESS}, {"DXGI_FORMAT_R32G32B32A32_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32A32_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32A32_FLOAT}, {"DXGI_FORMAT_R32G32B32A32_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32A32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32A32_UINT}, {"DXGI_FORMAT_R32G32B32A32_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32A32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32A32_SINT}, {"DXGI_FORMAT_R32G32B32A32_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32A32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R32G32B32_TYPELESS}, {"DXGI_FORMAT_R32G32B32_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32_FLOAT}, {"DXGI_FORMAT_R32G32B32_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32_UINT}, {"DXGI_FORMAT_R32G32B32_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32B32_SINT}, {"DXGI_FORMAT_R32G32B32_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32B32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R16G16B16A16_TYPELESS}, {"DXGI_FORMAT_R16G16B16A16_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16B16A16_FLOAT}, {"DXGI_FORMAT_R16G16B16A16_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16B16A16_UNORM}, {"DXGI_FORMAT_R16G16B16A16_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16B16A16_UINT}, {"DXGI_FORMAT_R16G16B16A16_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16B16A16_SNORM}, {"DXGI_FORMAT_R16G16B16A16_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16B16A16_SINT}, {"DXGI_FORMAT_R16G16B16A16_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16B16A16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R32G32_TYPELESS}, {"DXGI_FORMAT_R32G32_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32_FLOAT}, {"DXGI_FORMAT_R32G32_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32_UINT}, {"DXGI_FORMAT_R32G32_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32G32_SINT}, {"DXGI_FORMAT_R32G32_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R32G8X24_TYPELESS}, {"DXGI_FORMAT_R32G8X24_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G8X24_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_D32_FLOAT_S8X24_UINT}, {"DXGI_FORMAT_D32_FLOAT_S8X24_UINT", IMMUTABLE, MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32G8X24_TYPELESS, {XRC_DEPTH_TEXTURE_USAGE}, XRC_DEPTH_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS}, {"DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_X32_TYPELESS_G8X24_UINT}, {"DXGI_FORMAT_X32_TYPELESS_G8X24_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R10G10B10A2_TYPELESS}, {"DXGI_FORMAT_R10G10B10A2_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R10G10B10A2_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R10G10B10A2_UNORM}, {"DXGI_FORMAT_R10G10B10A2_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R10G10B10A2_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R10G10B10A2_UINT}, {"DXGI_FORMAT_R10G10B10A2_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R10G10B10A2_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - // This doesn't have a typeless equivalent, so it's created as-is by the runtime. - {{DXGI_FORMAT_R11G11B10_FLOAT}, {"DXGI_FORMAT_R11G11B10_FLOAT", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R11G11B10_FLOAT, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R8G8B8A8_TYPELESS}, {"DXGI_FORMAT_R8G8B8A8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8B8A8_UNORM}, {"DXGI_FORMAT_R8G8B8A8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8B8A8_UNORM_SRGB}, {"DXGI_FORMAT_R8G8B8A8_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8B8A8_UINT}, {"DXGI_FORMAT_R8G8B8A8_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8B8A8_SNORM}, {"DXGI_FORMAT_R8G8B8A8_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8B8A8_SINT}, {"DXGI_FORMAT_R8G8B8A8_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8B8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R16G16_TYPELESS}, {"DXGI_FORMAT_R16G16_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16_FLOAT}, {"DXGI_FORMAT_R16G16_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16_UNORM}, {"DXGI_FORMAT_R16G16_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16_UINT}, {"DXGI_FORMAT_R16G16_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16_SNORM}, {"DXGI_FORMAT_R16G16_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16G16_SINT}, {"DXGI_FORMAT_R16G16_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16G16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R32_TYPELESS}, {"DXGI_FORMAT_R32_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_D32_FLOAT}, {"DXGI_FORMAT_D32_FLOAT", IMMUTABLE, MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_TYPELESS, {XRC_DEPTH_TEXTURE_USAGE}, XRC_DEPTH_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32_FLOAT}, {"DXGI_FORMAT_R32_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32_UINT}, {"DXGI_FORMAT_R32_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R32_SINT}, {"DXGI_FORMAT_R32_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R32_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R24G8_TYPELESS}, {"DXGI_FORMAT_R24G8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R24G8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_D24_UNORM_S8_UINT}, {"DXGI_FORMAT_D24_UNORM_S8_UINT", IMMUTABLE, MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R24G8_TYPELESS, {XRC_DEPTH_TEXTURE_USAGE}, XRC_DEPTH_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R24_UNORM_X8_TYPELESS}, {"DXGI_FORMAT_R24_UNORM_X8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R24G8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_X24_TYPELESS_G8_UINT}, {"DXGI_FORMAT_X24_TYPELESS_G8_UINT", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R24G8_TYPELESS, {}, {}, {}, {}, {}}}, - - {{DXGI_FORMAT_R8G8_TYPELESS}, {"DXGI_FORMAT_R8G8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8_UNORM}, {"DXGI_FORMAT_R8G8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8_UINT}, {"DXGI_FORMAT_R8G8_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8_SNORM}, {"DXGI_FORMAT_R8G8_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8_SINT}, {"DXGI_FORMAT_R8G8_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R16_TYPELESS}, {"DXGI_FORMAT_R16_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R16_FLOAT}, {"DXGI_FORMAT_R16_FLOAT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_D16_UNORM}, {"DXGI_FORMAT_D16_UNORM", IMMUTABLE, MUT_SUPPORT, NON_COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_DEPTH_TEXTURE_USAGE}, XRC_DEPTH_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16_UNORM}, {"DXGI_FORMAT_R16_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16_UINT}, {"DXGI_FORMAT_R16_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16_SNORM}, {"DXGI_FORMAT_R16_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R16_SINT}, {"DXGI_FORMAT_R16_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R16_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_R8_TYPELESS}, {"DXGI_FORMAT_R8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_R8_UNORM}, {"DXGI_FORMAT_R8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8_UINT}, {"DXGI_FORMAT_R8_UINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8_SNORM}, {"DXGI_FORMAT_R8_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8_SINT}, {"DXGI_FORMAT_R8_SINT", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_A8_UNORM}, {"DXGI_FORMAT_A8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - // These don't have typeless equivalents, so they are created as-is by the runtime. - {{DXGI_FORMAT_R1_UNORM}, {"DXGI_FORMAT_R1_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R1_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R9G9B9E5_SHAREDEXP}, {"DXGI_FORMAT_R9G9B9E5_SHAREDEXP", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R9G9B9E5_SHAREDEXP, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R8G8_B8G8_UNORM}, {"DXGI_FORMAT_R8G8_B8G8_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R8G8_B8G8_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_G8R8_G8B8_UNORM}, {"DXGI_FORMAT_G8R8_G8B8_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_G8R8_G8B8_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC1_TYPELESS}, {"DXGI_FORMAT_BC1_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC1_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC1_UNORM}, {"DXGI_FORMAT_BC1_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC1_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC1_UNORM_SRGB}, {"DXGI_FORMAT_BC1_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC1_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC2_TYPELESS}, {"DXGI_FORMAT_BC2_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC2_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC2_UNORM}, {"DXGI_FORMAT_BC2_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC2_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC2_UNORM_SRGB}, {"DXGI_FORMAT_BC2_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC2_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC3_TYPELESS}, {"DXGI_FORMAT_BC3_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC3_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC3_UNORM}, {"DXGI_FORMAT_BC3_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC3_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC3_UNORM_SRGB}, {"DXGI_FORMAT_BC3_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC3_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC4_TYPELESS}, {"DXGI_FORMAT_BC4_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC4_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC4_UNORM}, {"DXGI_FORMAT_BC4_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC4_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC4_SNORM}, {"DXGI_FORMAT_BC4_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC4_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC5_TYPELESS}, {"DXGI_FORMAT_BC5_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC5_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC5_UNORM}, {"DXGI_FORMAT_BC5_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC5_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC5_SNORM}, {"DXGI_FORMAT_BC5_SNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC5_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - // These don't have typeless equivalents, so they are created as-is by the runtime. - {{DXGI_FORMAT_B5G6R5_UNORM}, {"DXGI_FORMAT_B5G6R5_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B5G6R5_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_B5G5R5A1_UNORM}, {"DXGI_FORMAT_B5G5R5A1_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B5G5R5A1_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM}, {"DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_B8G8R8A8_TYPELESS}, {"DXGI_FORMAT_B8G8R8A8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8A8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_B8G8R8A8_UNORM}, {"DXGI_FORMAT_B8G8R8A8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_B8G8R8A8_UNORM_SRGB}, {"DXGI_FORMAT_B8G8R8A8_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8A8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_B8G8R8X8_TYPELESS}, {"DXGI_FORMAT_B8G8R8X8_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8X8_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_B8G8R8X8_UNORM}, {"DXGI_FORMAT_B8G8R8X8_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8X8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_B8G8R8X8_UNORM_SRGB}, {"DXGI_FORMAT_B8G8R8X8_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B8G8R8X8_TYPELESS, {XRC_COLOR_TEXTURE_USAGE, XRC_COLOR_TEXTURE_USAGE_MUTABLE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC6H_TYPELESS}, {"DXGI_FORMAT_BC6H_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC6H_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC6H_UF16}, {"DXGI_FORMAT_BC6H_UF16", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC6H_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC6H_SF16}, {"DXGI_FORMAT_BC6H_SF16", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC6H_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - {{DXGI_FORMAT_BC7_TYPELESS}, {"DXGI_FORMAT_BC7_TYPELESS", MUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC7_TYPELESS, {}, {}, {}, {}, {}}}, - {{DXGI_FORMAT_BC7_UNORM}, {"DXGI_FORMAT_BC7_UNORM", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC7_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - {{DXGI_FORMAT_BC7_UNORM_SRGB}, {"DXGI_FORMAT_BC7_UNORM_SRGB", IMMUTABLE, MUT_SUPPORT, COLOR, COMPRESSED, NO_RENDERING_SUPPORT, DXGI_FORMAT_BC7_TYPELESS, {XRC_COLOR_TEXTURE_USAGE_COMPRESSED}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - // This doesn't have a typeless equivalent, so it's created as-is by the runtime. - {{DXGI_FORMAT_B4G4R4A4_UNORM}, {"DXGI_FORMAT_B4G4R4A4_UNORM", IMMUTABLE, NO_MUT_SUPPORT, COLOR, UNCOMPRESSED, RENDERING_SUPPORT, DXGI_FORMAT_B4G4R4A4_UNORM, {XRC_COLOR_TEXTURE_USAGE}, XRC_COLOR_CREATE_FLAGS, {}, {}, {}}}, - - }; - // clang-format on + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32A32_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32A32_FLOAT).ExpectedFormat(DXGI_FORMAT_R32G32B32A32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32A32_UINT).ExpectedFormat(DXGI_FORMAT_R32G32B32A32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32A32_SINT).ExpectedFormat(DXGI_FORMAT_R32G32B32A32_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32_FLOAT).ExpectedFormat(DXGI_FORMAT_R32G32B32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32_UINT).ExpectedFormat(DXGI_FORMAT_R32G32B32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32B32_SINT).ExpectedFormat(DXGI_FORMAT_R32G32B32_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_FLOAT).ExpectedFormat(DXGI_FORMAT_R16G16B16A16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_UINT).ExpectedFormat(DXGI_FORMAT_R16G16B16A16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_SINT).ExpectedFormat(DXGI_FORMAT_R16G16B16A16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_UNORM).ExpectedFormat(DXGI_FORMAT_R16G16B16A16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16B16A16_SNORM).ExpectedFormat(DXGI_FORMAT_R16G16B16A16_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32_FLOAT).ExpectedFormat(DXGI_FORMAT_R32G32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32_UINT).ExpectedFormat(DXGI_FORMAT_R32G32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G32_SINT).ExpectedFormat(DXGI_FORMAT_R32G32_TYPELESS).Build(), + + // 32bit channel, 8bit channel, 24bit ignored. All typeless. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32G8X24_TYPELESS).Typeless().Build(), + // 32bit float depth, 8 bit uint stencil, 24bit ignored. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_D32_FLOAT_S8X24_UINT).ExpectedFormat(DXGI_FORMAT_R32G8X24_TYPELESS).DepthStencil().Build(), + // 32bit float red, 8bit ignored, 24bit ignored. Not typeless because used parts are typed? + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS).Typeless().Build(), + // typeless unused 32bit component, 8bit uint green, and 24bit unused. Not typeless because used parts are typed? + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_X32_TYPELESS_G8X24_UINT).ExpectedFormat(DXGI_FORMAT_R32_FLOAT_X8X24_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R10G10B10A2_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R10G10B10A2_UNORM).ExpectedFormat(DXGI_FORMAT_R10G10B10A2_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R10G10B10A2_UINT).ExpectedFormat(DXGI_FORMAT_R10G10B10A2_TYPELESS).Build(), + + // This doesn't have a typeless equivalent, so it's created as-is by the runtime. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R11G11B10_FLOAT).NotMutable().Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_UNORM).ExpectedFormat(DXGI_FORMAT_R8G8B8A8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_UNORM_SRGB).ExpectedFormat(DXGI_FORMAT_R8G8B8A8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_UINT).ExpectedFormat(DXGI_FORMAT_R8G8B8A8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_SINT).ExpectedFormat(DXGI_FORMAT_R8G8B8A8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8B8A8_SNORM).ExpectedFormat(DXGI_FORMAT_R8G8B8A8_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_FLOAT).ExpectedFormat(DXGI_FORMAT_R16G16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_UINT).ExpectedFormat(DXGI_FORMAT_R16G16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_SINT).ExpectedFormat(DXGI_FORMAT_R16G16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_UNORM).ExpectedFormat(DXGI_FORMAT_R16G16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16G16_SNORM).ExpectedFormat(DXGI_FORMAT_R16G16_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32_FLOAT).ExpectedFormat(DXGI_FORMAT_R32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_D32_FLOAT).ExpectedFormat(DXGI_FORMAT_R32_TYPELESS).Depth().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32_UINT).ExpectedFormat(DXGI_FORMAT_R32_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R32_SINT).ExpectedFormat(DXGI_FORMAT_R32_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R24G8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_D24_UNORM_S8_UINT).ExpectedFormat(DXGI_FORMAT_R24G8_TYPELESS).Depth().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R24_UNORM_X8_TYPELESS).ExpectedFormat(DXGI_FORMAT_R24G8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_X24_TYPELESS_G8_UINT).ExpectedFormat(DXGI_FORMAT_R24G8_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_UINT).ExpectedFormat(DXGI_FORMAT_R8G8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_SINT).ExpectedFormat(DXGI_FORMAT_R8G8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_UNORM).ExpectedFormat(DXGI_FORMAT_R8G8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_SNORM).ExpectedFormat(DXGI_FORMAT_R8G8_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_FLOAT).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_D16_UNORM).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Depth().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_UINT).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_SINT).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_UNORM).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R16_SNORM).ExpectedFormat(DXGI_FORMAT_R16_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8_UINT).ExpectedFormat(DXGI_FORMAT_R8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8_SINT).ExpectedFormat(DXGI_FORMAT_R8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8_UNORM).ExpectedFormat(DXGI_FORMAT_R8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8_SNORM).ExpectedFormat(DXGI_FORMAT_R8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_A8_UNORM).ExpectedFormat(DXGI_FORMAT_R8_TYPELESS).Build(), + + // These don't have typeless equivalents, so they are created as-is by the runtime. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R1_UNORM).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R9G9B9E5_SHAREDEXP).NotMutable().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R8G8_B8G8_UNORM).NotMutable().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_G8R8_G8B8_UNORM).NotMutable().Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC1_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC1_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC1_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC1_UNORM_SRGB).Compressed().ExpectedFormat(DXGI_FORMAT_BC1_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC2_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC2_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC2_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC2_UNORM_SRGB).Compressed().ExpectedFormat(DXGI_FORMAT_BC2_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC3_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC3_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC3_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC3_UNORM_SRGB).Compressed().ExpectedFormat(DXGI_FORMAT_BC3_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC4_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC4_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC4_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC4_SNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC4_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC5_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC5_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC5_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC5_SNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC5_TYPELESS).Build(), + + // These don't have typeless equivalents, so they are created as-is by the runtime. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B5G6R5_UNORM).NotMutable().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B5G5R5A1_UNORM).NotMutable().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_R10G10B10_XR_BIAS_A2_UNORM).NotMutable().Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8A8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8A8_UNORM).ExpectedFormat(DXGI_FORMAT_B8G8R8A8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8A8_UNORM_SRGB).ExpectedFormat(DXGI_FORMAT_B8G8R8A8_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8X8_TYPELESS).Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8X8_UNORM).ExpectedFormat(DXGI_FORMAT_B8G8R8X8_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B8G8R8X8_UNORM_SRGB).ExpectedFormat(DXGI_FORMAT_B8G8R8X8_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC6H_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC6H_UF16).Compressed().ExpectedFormat(DXGI_FORMAT_BC6H_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC6H_SF16).Compressed().ExpectedFormat(DXGI_FORMAT_BC6H_TYPELESS).Build(), + + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC7_TYPELESS).Compressed().Typeless().Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC7_UNORM).Compressed().ExpectedFormat(DXGI_FORMAT_BC7_TYPELESS).Build(), + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_BC7_UNORM_SRGB).Compressed().ExpectedFormat(DXGI_FORMAT_BC7_TYPELESS).Build(), + + // This doesn't have a typeless equivalent, so it's created as-is by the runtime. + XRC_SWAPCHAIN_FORMAT(DXGI_FORMAT_B4G4R4A4_UNORM).NotMutable().Build(), + + }; return dxgiSwapchainTestMap; } @@ -263,35 +244,20 @@ namespace Conformance // (e.g. DXGI_FORMAT_R8G8B8A8_TYPELESS). Only concrete formats are returned, and only concrete // formats may be specified by applications for swapchain creation. - SwapchainTestMap& dxgiSwapchainTestMap = GetDxgiSwapchainTestMap(); - SwapchainTestMap::iterator it = dxgiSwapchainTestMap.find(imageFormat); + const SwapchainTestMap& dxgiSwapchainTestMap = GetDxgiSwapchainTestMap(); + SwapchainTestMap::const_iterator it = dxgiSwapchainTestMap.find(imageFormat); // Verify that the image format is known. If it's not known then this test needs to be // updated to recognize new DXGI formats. XRC_CHECK_THROW_MSG(it != dxgiSwapchainTestMap.end(), "Unknown DXGI image format."); - if (it == dxgiSwapchainTestMap.end()) { - return false; - } // Verify that imageFormat is not a typeless type. Only regular types are allowed to // be returned by the runtime for enumerated image formats. XRC_CHECK_THROW_MSG(!it->second.mutableFormat, "Typeless DXGI image formats must not be enumerated by runtimes: " + it->second.imageFormatName); - if (it->second.mutableFormat) { - return false; - } // We may now proceed with creating swapchains with the format. - SwapchainCreateTestParameters& tp = it->second; - tp.arrayCountVector = {1, 2}; - if (tp.colorFormat && !tp.compressedFormat) { - tp.mipCountVector = {1, 2}; - } - else { - tp.mipCountVector = {1}; - } - - *swapchainTestParameters = tp; + *swapchainTestParameters = it->second; return true; } diff --git a/src/conformance/utilities/d3d_common.h b/src/conformance/utilities/d3d_common.h index 29245aea..2fa3ccad 100644 --- a/src/conformance/utilities/d3d_common.h +++ b/src/conformance/utilities/d3d_common.h @@ -3,12 +3,15 @@ // SPDX-License-Identifier: Apache-2.0 #pragma once -#if (defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12)) && !defined(MISSING_DIRECTX_COLORS) +#if defined(XR_USE_GRAPHICS_API_D3D11) || defined(XR_USE_GRAPHICS_API_D3D12) #include #include "common/xr_linear.h" #include "swapchain_parameters.h" +#include +#include + #include #include #include diff --git a/src/conformance/utilities/swapchain_format_data.cpp b/src/conformance/utilities/swapchain_format_data.cpp new file mode 100644 index 00000000..83507f49 --- /dev/null +++ b/src/conformance/utilities/swapchain_format_data.cpp @@ -0,0 +1,329 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#include "common/hex_and_handles.h" +#include "swapchain_parameters.h" +#include "swapchain_format_data.h" +#include "throw_helpers.h" + +#include +#include +#include +#include +#include +#include + +namespace Conformance +{ + + // the app might request any combination of flags + static constexpr auto XRC_COLOR_UA_COPY_SAMPLED_MUTABLE_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_UNORDERED_ACCESS_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | + XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + }; + static constexpr auto XRC_COLOR_COPY_SAMPLED_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + }; + static constexpr auto XRC_COLOR_COPY_SAMPLED_MUTABLE_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + }; + static constexpr auto XRC_DEPTH_COPY_SAMPLED_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT, + XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_SRC_BIT | XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT | + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + }; + static constexpr auto XRC_COMPRESSED_SAMPLED_MUTABLE_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, + }; + static constexpr auto XRC_COMPRESSED_SAMPLED_USAGE_FLAGS = { + XR_SWAPCHAIN_USAGE_SAMPLED_BIT, + }; + + static constexpr std::array kArrayOf1{{1}}; + static constexpr std::array kArrayOf1And2{{1, 2}}; + static constexpr std::array kDefaultCreateFlags{ + {0, XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT, XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT, + XR_SWAPCHAIN_CREATE_PROTECTED_CONTENT_BIT | XR_SWAPCHAIN_CREATE_STATIC_IMAGE_BIT}}; + + SwapchainFormatData::SwapchainFormatData(int64_t format, const char* name) + : m_imageFormat(format) + , m_imageFormatName(name) + , m_expectedCreatedImageFormat(format) + , m_createFlagsVector(kDefaultCreateFlags.begin(), kDefaultCreateFlags.end()) + { + } + + SwapchainCreateTestParameters GetSwapchainCreateTestParameters(const SwapchainFormatDataMap& formatData, int64_t imageFormat) + { + SwapchainFormatDataMap::const_iterator it = formatData.find(imageFormat); + + // Verify that the image format is known. If it's not known then this test needs to be + // updated to recognize new formats. + if (it == formatData.end()) { + XRC_THROW("Unknown image format: " + std::to_string(imageFormat)); + } + + // Verify that imageFormat is not a typeless type. Only regular types are allowed to + // be returned by the runtime for enumerated image formats. + if (it->second.IsTypeless()) { + XRC_THROW("Typeless image formats must not be enumerated by runtimes: " + std::string{it->second.GetImageFormatName()}); + } + + // We may now proceed with creating swapchains with the format. + return it->second.ToTestParameters(); + } + + const char* GetImageFormatName(const SwapchainFormatDataMap& formatData, int64_t imageFormat) + { + SwapchainFormatDataMap::const_iterator it = formatData.find(imageFormat); + + if (it != formatData.end()) { + return it->second.GetImageFormatName(); + } + + return "unknown"; + } + + bool IsImageFormatKnown(const SwapchainFormatDataMap& formatData, int64_t imageFormat) + { + SwapchainFormatDataMap::const_iterator it = formatData.find(imageFormat); + + return it != formatData.end(); + } + + SwapchainCreateTestParametersBuilder::SwapchainCreateTestParametersBuilder(int64_t imageFormat, const char* imageFormatName) + : m_data(imageFormat, imageFormatName) + { + UpdateDefaultUsageFlagVector(); + } + + std::pair SwapchainCreateTestParametersBuilder::Build() const + { + return m_data.Build(); + } + + std::pair SwapchainCreateTestParametersBuilder::ToPair() const + { + return m_data.ToPair(); + } + + std::string SwapchainCreateTestParametersBuilder::ToString() const + { + return m_data.ToString(); + } + + void SwapchainCreateTestParametersBuilder::UpdateDefaultUsageFlagVector() + { + if (m_data.m_isTypeless) { + m_data.m_usageFlagsVector = {}; + } + else if (m_data.m_compressedFormat) { + + if (m_data.m_supportsMutableFormat) { + // compressed, mutable + m_data.m_usageFlagsVector = XRC_COMPRESSED_SAMPLED_MUTABLE_USAGE_FLAGS; + } + else { + // compressed, not mutable + m_data.m_usageFlagsVector = XRC_COMPRESSED_SAMPLED_USAGE_FLAGS; + } + } + else { // not compressed + if (m_data.m_colorFormat) { + if (m_data.m_supportsMutableFormat) { + // not compressed, color, mutable + m_data.m_usageFlagsVector = + m_data.m_allowUA ? XRC_COLOR_UA_COPY_SAMPLED_MUTABLE_USAGE_FLAGS : XRC_COLOR_COPY_SAMPLED_MUTABLE_USAGE_FLAGS; + } + else { + + // not compressed, color, not mutable + m_data.m_usageFlagsVector = XRC_COLOR_COPY_SAMPLED_USAGE_FLAGS; + } + } + else { + + // not compressed, depth/stencil + m_data.m_usageFlagsVector = XRC_DEPTH_COPY_SAMPLED_USAGE_FLAGS; + } + } + } + + SwapchainCreateTestParameters SwapchainFormatData::ToTestParameters() const + { + + span mipCountVector{kArrayOf1.begin(), kArrayOf1.end()}; + if (m_colorFormat && !m_compressedFormat) { + mipCountVector = {kArrayOf1And2.begin(), kArrayOf1And2.end()}; + } + + span arrayCountVector{kArrayOf1And2.begin(), kArrayOf1And2.end()}; + return SwapchainCreateTestParameters{ + std::string{m_imageFormatName}, // + m_isTypeless ? SwapchainFormatMutability::MUTABLE : SwapchainFormatMutability::IMMUTABLE, // + m_supportsMutableFormat ? SwapchainFormatSupportsMutability::MUT_SUPPORT + : SwapchainFormatSupportsMutability::NO_MUT_SUPPORT, // + m_colorFormat ? SwapchainFormatIsColor::COLOR : SwapchainFormatIsColor::NON_COLOR, + m_compressedFormat ? SwapchainFormatIsCompressed::COMPRESSED : SwapchainFormatIsCompressed::UNCOMPRESSED, + m_compressedFormat ? SwapchainFormatSupportsRendering::NO_RENDERING_SUPPORT + : SwapchainFormatSupportsRendering::RENDERING_SUPPORT, + m_expectedCreatedImageFormat, + {m_usageFlagsVector.begin(), m_usageFlagsVector.end()}, + {m_createFlagsVector.begin(), m_createFlagsVector.end()}, + {arrayCountVector.begin(), arrayCountVector.end()}, + {/* sampleCountVector - unused */}, + {mipCountVector.begin(), mipCountVector.end()}, + m_depthFormat, + m_stencilFormat, + }; + } + + std::pair SwapchainFormatData::Build() const + { + + return std::make_pair(m_imageFormat, ToTestParameters()); + } + + std::pair SwapchainFormatData::ToPair() const + { + + return {m_imageFormat, *this}; + } + + std::string SwapchainFormatData::ToString() const + { + + std::ostringstream oss; + oss << m_imageFormatName << " (" << to_hex(m_imageFormat) << "):"; + if (m_compressedFormat) { + oss << " compressed"; + } + + // what kind of thing: color, depth, stencil + if (m_colorFormat) { + oss << " color"; + } + else if (m_depthFormat && m_stencilFormat) { + oss << " depth/stencil"; + } + else if (m_depthFormat) { + oss << " depth"; + } + else if (m_stencilFormat) { + oss << " stencil"; + } + + if (m_isTypeless) { + oss << " typeless"; + } + + oss << " texture format"; + + if (!m_supportsMutableFormat) { + oss << " (no mutable format support)"; + } + if (!m_allowUA) { + oss << " (no UA support)"; + } + if (m_expectedCreatedImageFormat != m_imageFormat) { + oss << " (expected to be created as " << to_hex(m_expectedCreatedImageFormat) << ")"; + } + return oss.str(); + } + +} // namespace Conformance diff --git a/src/conformance/utilities/swapchain_format_data.h b/src/conformance/utilities/swapchain_format_data.h new file mode 100644 index 00000000..2a2baf28 --- /dev/null +++ b/src/conformance/utilities/swapchain_format_data.h @@ -0,0 +1,308 @@ +// Copyright (c) 2019-2023, The Khronos Group Inc. +// +// SPDX-License-Identifier: Apache-2.0 +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +#pragma once + +#include +#include +#include "swapchain_parameters.h" +#include "utils.h" + +#include +#include +#include +#include +#include + +namespace Conformance +{ + using nonstd::span; + + /// Minimal data structure storing details about a swapchain image format. + /// + /// May eventually replace @ref SwapchainImageTestParam + class SwapchainFormatData + { + public: + /// The graphics-API-specific numeric value of the image format + int64_t GetImageFormat() const + { + return m_imageFormat; + } + + /// String-ified version of the C identifier. + const char* GetImageFormatName() const + { + return m_imageFormatName; + } + + /// The graphics-API-specific created image format returned by `xrCreateSwapchain`, may be different from @ref GetImageFormat() + int64_t GetExpectedCreatedImageFormat() const + { + return m_expectedCreatedImageFormat; + } + + /// Whether "unordered access" usage flag is allowed + bool SupportsUnorderedAccess() const + { + return m_allowUA; + } + + /// Whether the image format is a mutable (a.k.a. typeless) type. + bool IsTypeless() const + { + return m_isTypeless; + } + + /// Whether the image format supports creation with XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT. + bool SupportsMutableFormatBit() const + { + return m_supportsMutableFormat; + } + + /// Whether the format is a color-specific format + bool GetColorFormat() const + { + return m_colorFormat; + } + + /// Whether the format can be use as a depth buffer: implies not color + bool GetDepthFormat() const + { + return m_depthFormat; + } + + /// Whether the format can be use as a stencil buffer: implies not color + bool GetStencilFormat() const + { + return m_stencilFormat; + } + + /// Whether the format is a compressed format (and thus cannot be rendered to) + bool GetCompressedFormat() const + { + return m_compressedFormat; + } + + /// XrSwapchainUsageFlags to exercise for this format. + /// Defaults to all combinations, including 0, of the core flags. + span GetUsageFlagsTestValues() const + { + return m_usageFlagsVector; + } + + /// XrSwapchainCreateFlags + span GetCreateFlagsTestValues() const + { + return m_createFlagsVector; + } + + /// Convert to a @ref SwapchainCreateTestParameters instance + SwapchainCreateTestParameters ToTestParameters() const; + + /// Convert to a pair of the numeric format and @ref SwapchainCreateTestParameters instance + std::pair Build() const; + + /// Return pair of the numeric format and and ourself. + std::pair ToPair() const; + + /// Describe this entry + std::string ToString() const; + + private: + friend class SwapchainCreateTestParametersBuilder; + SwapchainFormatData(int64_t format, const char* name); + + /// The graphics-API-specific numeric value of the image format + int64_t m_imageFormat; + + /// String-ified version of the C identifier. + const char* m_imageFormatName; + + /// The graphics-API-specific created image format returned by `xrCreateSwapchain`, may be different from @ref m_imageFormat in some cases. + int64_t m_expectedCreatedImageFormat; + + /// Whether "unordered access" usage flag is allowed + bool m_allowUA = true; + + /// Whether the image format is a mutable (a.k.a. typeless) type. + bool m_isTypeless = false; + + /// Whether the image format supports creation with XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT. + bool m_supportsMutableFormat = true; + + /// Whether the format is a color-specific format + bool m_colorFormat = true; + + /// Whether the format can be use as a depth buffer: implies not color + bool m_depthFormat = false; + + /// Whether the format can be use as a stencil buffer: implies not color + bool m_stencilFormat = false; + + /// Whether the format is a compressed format (and thus cannot be rendered to) + bool m_compressedFormat = false; + + /// XrSwapchainUsageFlags to exercise for this format. + /// Defaults to all combinations, including 0, of the core flags. + /// @todo Stop making so many copies of this, generate it from the other data instead + span m_usageFlagsVector; + + /// XrSwapchainCreateFlags + /// @todo Stop making so many copies of this, generate it from the other data instead + span m_createFlagsVector; + }; + + /// A map of swapchain format (numeric value) to @ref SwapchainFormatData + using SwapchainFormatDataMap = std::map; + + /// Look up the swapchain create test parameters in an map (API-specific). + /// + /// Throws if the format cannot be found. + SwapchainCreateTestParameters GetSwapchainCreateTestParameters(const SwapchainFormatDataMap& formatData, int64_t imageFormat); + + /// Returns a name for an image format. Returns "unknown" for unknown formats. + const char* GetImageFormatName(const SwapchainFormatDataMap& formatData, int64_t imageFormat); + + /// Returns true if the format is known to the plugin. Can be false if the runtime supports extra formats unknown to the conformance tests + /// (e.g. in APIs which have optional extensions). + bool IsImageFormatKnown(const SwapchainFormatDataMap& formatData, int64_t imageFormat); + + class SwapchainCreateTestParametersBuilder + { + public: + SwapchainCreateTestParametersBuilder(int64_t imageFormat, const char* imageFormatName); + + using Self = SwapchainCreateTestParametersBuilder; + + /// Mark this as not supporting "unordered access" + Self& NoUnorderedAccess() + { + m_data.m_allowUA = false; + UpdateDefaultUsageFlagVector(); + return *this; + } + + /// Mark this as being a "typeless" format (just channels of widths, no implied interpretation) + /// + /// Also sets some default usage flags. + Self& Typeless() + { + m_data.m_isTypeless = true; + /// @todo is this actually right? It's what the old d3d code did. + m_data.m_createFlagsVector = {}; + UpdateDefaultUsageFlagVector(); + return *this; + } + + /// Mark this as supporting depth buffer usage (and un-marking for color buffer usage) + /// + /// Also sets some default usage flags. + Self& Depth() + { + m_data.m_depthFormat = true; + NotColor(); + return *this; + } + + /// Mark this as supporting stencil buffer usage (and un-marking for color buffer usage) + /// + /// Also sets some default usage flags. + Self& Stencil() + { + m_data.m_stencilFormat = true; + NotColor(); + return *this; + } + + /// Mark this as supporting depth and stencil buffer usage (and un-marking for color buffer usage) + /// + /// Also sets some default usage flags. + /// + /// Equivalent to calling both @ref Depth() and @ref Stencil() + Self& DepthStencil() + { + m_data.m_stencilFormat = true; + m_data.m_depthFormat = true; + NotColor(); + return *this; + } + + /// Record that we expect the runtime to allocate this as the specified different format (normally a typeless version if one exists) + Self& ExpectedFormat(int64_t format) + { + assert(format == m_data.m_expectedCreatedImageFormat || !m_data.m_isTypeless); + m_data.m_expectedCreatedImageFormat = format; + return *this; + } + + /// Mark this as a format for which we should not test the `XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT` + Self& NotMutable() + { + assert(!m_data.m_isTypeless); + m_data.m_supportsMutableFormat = false; + UpdateDefaultUsageFlagVector(); + return *this; + } + + /// Mark this as a compressed format that we should not test rendering to + Self& Compressed() + { + m_data.m_compressedFormat = true; + UpdateDefaultUsageFlagVector(); + return *this; + } + + /// Populate the usage flags combinations to test. + /// + /// @note Call this method *after* any other builder methods other than @ref Build, since many of them update the usage flags + Self& UsageFlags(span usageFlagCombinationsToTest) + { + m_data.m_usageFlagsVector = usageFlagCombinationsToTest; + return *this; + } + + /// Populate the create flags combinations to test + Self& CreateFlags(span createFlagCombinationsToTest) + { + m_data.m_createFlagsVector = createFlagCombinationsToTest; + return *this; + } + + /// Convert to a pair of the numeric format and @ref SwapchainCreateTestParameters instance + std::pair Build() const; + + /// Return pair of the numeric format and and ourself. + std::pair ToPair() const; + + /// Describe this entry + std::string ToString() const; + + private: + void NotColor() + { + m_data.m_colorFormat = false; + UpdateDefaultUsageFlagVector(); + } + + void UpdateDefaultUsageFlagVector(); + SwapchainFormatData m_data; + }; + +/// Wraps constructor of @ref SwapchainCreateTestParametersBuilder to stringify format name +#define XRC_SWAPCHAIN_FORMAT(FORMAT) (::Conformance::SwapchainCreateTestParametersBuilder(FORMAT, #FORMAT)) + +} // namespace Conformance diff --git a/src/conformance/utilities/swapchain_parameters.h b/src/conformance/utilities/swapchain_parameters.h index 5b2a83d4..72390906 100644 --- a/src/conformance/utilities/swapchain_parameters.h +++ b/src/conformance/utilities/swapchain_parameters.h @@ -89,6 +89,12 @@ namespace Conformance /// Used only by color buffers. std::vector mipCountVector; + + /// Is this format usable as a depth buffer? + bool useAsDepth = false; + + /// Is this format usable as a stencil buffer? + bool useAsStencil = false; }; } // namespace Conformance diff --git a/src/conformance/utilities/vulkan_utils.h b/src/conformance/utilities/vulkan_utils.h index 99933130..2a4acceb 100644 --- a/src/conformance/utilities/vulkan_utils.h +++ b/src/conformance/utilities/vulkan_utils.h @@ -753,13 +753,13 @@ namespace Conformance swap(m_vkDevice, other.m_vkDevice); return *this; } - void Create(const VulkanDebugObjectNamer& namer, VkDevice device, VkImage aColorImage, VkImage aDepthImage, uint32_t baseArrayLayer, - VkExtent2D size, RenderPass& renderPass) + void Create(const VulkanDebugObjectNamer& namer, VkDevice device, VkImage aColorImage, VkImage aDepthOrStencilImage, + VkImageAspectFlags depthOrStencilImageAspect, uint32_t baseArrayLayer, VkExtent2D size, RenderPass& renderPass) { m_vkDevice = device; colorImage = aColorImage; - depthImage = aDepthImage; + depthImage = aDepthOrStencilImage; std::array attachments{}; uint32_t attachmentCount = 0; @@ -794,13 +794,25 @@ namespace Conformance depthViewInfo.components.g = VK_COMPONENT_SWIZZLE_G; depthViewInfo.components.b = VK_COMPONENT_SWIZZLE_B; depthViewInfo.components.a = VK_COMPONENT_SWIZZLE_A; - depthViewInfo.subresourceRange.aspectMask = VK_IMAGE_ASPECT_DEPTH_BIT; + depthViewInfo.subresourceRange.aspectMask = depthOrStencilImageAspect; depthViewInfo.subresourceRange.baseMipLevel = 0; depthViewInfo.subresourceRange.levelCount = 1; depthViewInfo.subresourceRange.baseArrayLayer = baseArrayLayer; depthViewInfo.subresourceRange.layerCount = 1; XRC_CHECK_THROW_VKCMD(vkCreateImageView(m_vkDevice, &depthViewInfo, nullptr, &depthView)); - XRC_CHECK_THROW_VKCMD(namer.SetName(VK_OBJECT_TYPE_IMAGE_VIEW, (uint64_t)depthView, "CTS depth image view")); + + const bool isDepth = depthOrStencilImageAspect & VK_IMAGE_ASPECT_DEPTH_BIT; + const bool isStencil = depthOrStencilImageAspect & VK_IMAGE_ASPECT_STENCIL_BIT; + if (isDepth && isStencil) { + XRC_CHECK_THROW_VKCMD(namer.SetName(VK_OBJECT_TYPE_IMAGE_VIEW, (uint64_t)depthView, "CTS depth/stencil image view")); + } + else if (isDepth) { + XRC_CHECK_THROW_VKCMD(namer.SetName(VK_OBJECT_TYPE_IMAGE_VIEW, (uint64_t)depthView, "CTS depth image view")); + } + else if (isStencil) { + XRC_CHECK_THROW_VKCMD(namer.SetName(VK_OBJECT_TYPE_IMAGE_VIEW, (uint64_t)depthView, "CTS stencil image view")); + } + attachments[attachmentCount++] = depthView; } diff --git a/src/loader/.gitignore b/src/loader/.gitignore index 5e8e0ba3..be1be1af 100644 --- a/src/loader/.gitignore +++ b/src/loader/.gitignore @@ -1,4 +1,4 @@ -# Copyright (c) 2020 The Khronos Group Inc. +# Copyright (c) 2020-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 diff --git a/src/loader/AndroidManifest.xml b/src/loader/AndroidManifest.xml index 2817b34a..0c151c63 100644 --- a/src/loader/AndroidManifest.xml +++ b/src/loader/AndroidManifest.xml @@ -2,7 +2,7 @@ package="org.khronos.openxr.openxr_loader_for_android"> diff --git a/src/loader/AndroidManifest.xml.in b/src/loader/AndroidManifest.xml.in index 9bd8a460..291a1e2d 100644 --- a/src/loader/AndroidManifest.xml.in +++ b/src/loader/AndroidManifest.xml.in @@ -4,7 +4,7 @@ android:versionName="${OPENXR_FULL_VERSION}${OPENXR_ANDROID_VERSION_SUFFIX}"> diff --git a/src/loader/CMakeLists.txt b/src/loader/CMakeLists.txt index 22047fbc..a98d3808 100644 --- a/src/loader/CMakeLists.txt +++ b/src/loader/CMakeLists.txt @@ -31,8 +31,8 @@ include(GNUInstallDirs) # List of all files externally generated outside of the loader that the loader # needs to build with. -set(LOADER_EXTERNAL_GEN_FILES ${COMMON_GENERATED_OUTPUT}) -set(LOADER_EXTERNAL_GEN_DEPENDS ${COMMON_GENERATED_DEPENDS}) +set(LOADER_EXTERNAL_GEN_FILES ${LOADER_GENERATED_OUTPUT}) +set(LOADER_EXTERNAL_GEN_DEPENDS ${LOADER_GENERATED_DEPENDS}) run_xr_xml_generate(loader_source_generator.py xr_generated_loader.hpp) run_xr_xml_generate(loader_source_generator.py xr_generated_loader.cpp) diff --git a/src/loader/abi.json.license b/src/loader/abi.json.license index ae931a24..924b9735 100644 --- a/src/loader/abi.json.license +++ b/src/loader/abi.json.license @@ -1,2 +1,3 @@ -Copyright (c) 2020 The Khronos Group Inc. -SPDX-License-Identifier: Apache-2.0 +Copyright (c) 2020-2023, The Khronos Group Inc. + +SPDX-License-Identifier: Apache-2.0 OR MIT diff --git a/src/loader/loader.rc b/src/loader/loader.rc index 8f36b13b..fd646580 100644 --- a/src/loader/loader.rc +++ b/src/loader/loader.rc @@ -8,10 +8,17 @@ // Initial Author: Mark Young // +// The years here should always match the Khronos line above! +#define COPYRIGHT_NOTICE "2017-2023, The Khronos Group Inc. and others" + +// REUSE-IgnoreStart + /////////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////// // Start customize section // Edit this section for your build +// The normal OpenXR CMake build will define the XR_CURRENT_API_*_VERSION +// symbols automatically so no manual changes are needed in that case. /////////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////// @@ -63,7 +70,7 @@ BEGIN BEGIN VALUE "FileDescription", "OpenXR Loader" VALUE "FileVersion", VER_FILE_VERSION_STR - VALUE "LegalCopyright", "Copyright (C) 2015-2020" + VALUE "LegalCopyright", "Copyright (C) " COPYRIGHT_NOTICE VALUE "ProductName", "OpenXR Loader" VALUE "ProductVersion", VER_FILE_VERSION_STR END diff --git a/src/loader/loader_core.cpp b/src/loader/loader_core.cpp index 98d3fa97..06e68700 100644 --- a/src/loader/loader_core.cpp +++ b/src/loader/loader_core.cpp @@ -19,7 +19,7 @@ #include "loader_logger.hpp" #include "loader_platform.hpp" #include "runtime_interface.hpp" -#include "xr_generated_dispatch_table.h" +#include "xr_generated_dispatch_table_core.h" #include "xr_generated_loader.hpp" #include diff --git a/src/loader/loader_instance.cpp b/src/loader/loader_instance.cpp index badd3919..ce5c2055 100644 --- a/src/loader/loader_instance.cpp +++ b/src/loader/loader_instance.cpp @@ -18,7 +18,7 @@ #include "loader_interfaces.h" #include "loader_logger.hpp" #include "runtime_interface.hpp" -#include "xr_generated_dispatch_table.h" +#include "xr_generated_dispatch_table_core.h" #include "xr_generated_loader.hpp" #include diff --git a/src/loader/manifest_file.cpp b/src/loader/manifest_file.cpp index 99f4e841..0683bc16 100644 --- a/src/loader/manifest_file.cpp +++ b/src/loader/manifest_file.cpp @@ -274,16 +274,45 @@ static std::string GetXDGEnvAbsolute(const char *name, const char *fallback_path return fallback_paths; } +/// @param rt_dir_prefix Directory prefix with a trailing slash +static bool FindEitherActiveRuntimeFilename(const char *prefix_desc, const std::string &rt_dir_prefix, uint16_t major_version, + std::string &out) { + { + std::ostringstream oss; + oss << "Looking for active_runtime." XR_ARCH_ABI ".json or active_runtime.json in "; + oss << prefix_desc; + oss << ": "; + oss << rt_dir_prefix; + + LoaderLogger::LogInfoMessage("", oss.str()); + } + { + auto decorated_path = rt_dir_prefix + std::to_string(major_version) + "/active_runtime." XR_ARCH_ABI ".json"; + + if (FileSysUtilsPathExists(decorated_path)) { + out = decorated_path; + return true; + } + } + { + auto undecorated_path = rt_dir_prefix + std::to_string(major_version) + "/active_runtime.json"; + + if (FileSysUtilsPathExists(undecorated_path)) { + out = undecorated_path; + return true; + } + } + return false; +} // Return the first instance of relative_path occurring in an XDG config dir according to standard // precedence order. -static bool FindXDGConfigFile(const std::string &relative_path, std::string &out) { - out = GetXDGEnvHome("XDG_CONFIG_HOME", ".config"); - if (!out.empty()) { - out += "/"; - out += relative_path; - - LoaderLogger::LogInfoMessage("", "Looking for " + relative_path + " in XDG_CONFIG_HOME: " + out); - if (FileSysUtilsPathExists(out)) { +static bool FindXDGConfigFile(const char *relative_dir, uint16_t major_version, std::string &out) { + const std::string message{"Looking for active_runtime." XR_ARCH_ABI ".json or active_runtime.json"}; + std::string dir_prefix = GetXDGEnvHome("XDG_CONFIG_HOME", ".config"); + if (!dir_prefix.empty()) { + dir_prefix += "/"; + dir_prefix += relative_dir; + if (FindEitherActiveRuntimeFilename("XDG_CONFIG_HOME", dir_prefix, major_version, out)) { return true; } } @@ -294,29 +323,26 @@ static bool FindXDGConfigFile(const std::string &relative_path, std::string &out if (path.empty()) { continue; } - out = path; - out += "/"; - out += relative_path; - LoaderLogger::LogInfoMessage("", "Looking for " + relative_path + " in an entry of XDG_CONFIG_DIRS: " + out); - if (FileSysUtilsPathExists(out)) { + dir_prefix = std::move(path); + dir_prefix += "/"; + dir_prefix += relative_dir; + if (FindEitherActiveRuntimeFilename("an entry of XDG_CONFIG_DIRS", dir_prefix, major_version, out)) { return true; } } - out = SYSCONFDIR; - out += "/"; - out += relative_path; - LoaderLogger::LogInfoMessage("", "Looking for " + relative_path + " in compiled-in SYSCONFDIR: " + out); - if (FileSysUtilsPathExists(out)) { + dir_prefix = SYSCONFDIR; + dir_prefix += "/"; + dir_prefix += relative_dir; + if (FindEitherActiveRuntimeFilename("compiled-in SYSCONFDIR", dir_prefix, major_version, out)) { return true; } #if defined(EXTRASYSCONFDIR) - out = EXTRASYSCONFDIR; - out += "/"; - out += relative_path; - LoaderLogger::LogInfoMessage("", "Looking for " + relative_path + " in compiled-in EXTRASYSCONFDIR: " + out); - if (FileSysUtilsPathExists(out)) { + dir_prefix = EXTRASYSCONFDIR; + dir_prefix += "/"; + dir_prefix += relative_dir; + if (FindEitherActiveRuntimeFilename("compiled-in EXTRASYSCONFDIR", dir_prefix, major_version, out)) { return true; } #endif @@ -632,9 +658,8 @@ XrResult RuntimeManifestFile::FindManifestFiles(std::vector @@ -29,6 +29,10 @@ #include "android_utilities.h" #include #include + +// Needed for the loader init struct +#include +#include #endif // XR_USE_PLATFORM_ANDROID #ifdef XR_KHR_LOADER_INIT_SUPPORT @@ -105,33 +109,22 @@ XrResult LoaderInitData::initialize(const XrLoaderInitInfoBaseHeaderKHR* info) { if (cast_info->applicationContext == nullptr) { return XR_ERROR_VALIDATION_FAILURE; } + + // Copy and store the JVM pointer and Android Context, ensuring the JVM is initialised. _data = *cast_info; - jni::init((jni::JavaVM*)_data.applicationVM); _data.next = nullptr; - JNIEnv* Env; - ((jni::JavaVM*)(cast_info->applicationVM))->AttachCurrentThread(&Env, nullptr); - const jclass contextClass = Env->GetObjectClass((jobject)_data.applicationContext); - - const jmethodID getAssetsMethod = Env->GetMethodID(contextClass, "getAssets", "()Landroid/content/res/AssetManager;"); - const jobject AssetManagerObject = Env->CallObjectMethod((jobject)_data.applicationContext, getAssetsMethod); - _android_asset_manager = AAssetManager_fromJava(Env, AssetManagerObject); - - const jmethodID getApplicationContextMethod = - Env->GetMethodID(contextClass, "getApplicationContext", "()Landroid/content/Context;"); - const jobject contextObject = Env->CallObjectMethod((jobject)_data.applicationContext, getApplicationContextMethod); - const jmethodID getApplicationInfoMethod = - Env->GetMethodID(contextClass, "getApplicationInfo", "()Landroid/content/pm/ApplicationInfo;"); - const jobject applicationInfoObject = Env->CallObjectMethod(contextObject, getApplicationInfoMethod); - const jfieldID nativeLibraryDirField = - Env->GetFieldID(Env->GetObjectClass(applicationInfoObject), "nativeLibraryDir", "Ljava/lang/String;"); - const jobject nativeLibraryDirObject = Env->GetObjectField(applicationInfoObject, nativeLibraryDirField); - const jmethodID getBytesMethod = - Env->GetMethodID(Env->GetObjectClass(nativeLibraryDirObject), "getBytes", "(Ljava/lang/String;)[B"); - const auto bytesObject = - static_cast(Env->CallObjectMethod(nativeLibraryDirObject, getBytesMethod, Env->NewStringUTF("UTF-8"))); - const size_t length = Env->GetArrayLength(bytesObject); - const jbyte* const bytes = Env->GetByteArrayElements(bytesObject, nullptr); - _native_library_path = std::string(reinterpret_cast(bytes), length); + jni::init(static_cast(_data.applicationVM)); + const jni::Object context = jni::Object{static_cast(_data.applicationContext)}; + + // Retrieve a reference to the Android AssetManager. + const auto assetManager = context.call("getAssets()Landroid/content/res/AssetManager;"); + _android_asset_manager = AAssetManager_fromJava(jni::env(), assetManager.getHandle()); + + // Retrieve the path to the native libraries. + const auto applicationContext = context.call("getApplicationContext()Landroid/content/Context;"); + const auto applicationInfo = context.call("getApplicationInfo()Landroid/content/pm/ApplicationInfo;"); + _native_library_path = applicationInfo.get("nativeLibraryDir"); + _initialized = true; return XR_SUCCESS; } diff --git a/src/scripts/generate_api_layer_manifest.py b/src/scripts/generate_api_layer_manifest.py index 04013fdd..fc843f3c 100755 --- a/src/scripts/generate_api_layer_manifest.py +++ b/src/scripts/generate_api_layer_manifest.py @@ -1,6 +1,6 @@ #!/usr/bin/python3 # -# Copyright (c) 2017 The Khronos Group Inc. +# Copyright (c) 2017-2023, The Khronos Group Inc. # # SPDX-License-Identifier: Apache-2.0 # diff --git a/src/scripts/loader_source_generator.py b/src/scripts/loader_source_generator.py index 29eef030..188865c0 100755 --- a/src/scripts/loader_source_generator.py +++ b/src/scripts/loader_source_generator.py @@ -81,6 +81,7 @@ def getProto(self, cur_cmd): # self the LoaderSourceOutputGenerator object def outputGeneratedHeaderWarning(self): + # REUSE-IgnoreStart generated_warning = '' generated_warning += '// Copyright (c) 2017-2023, The Khronos Group Inc.\n' generated_warning += '// Copyright (c) 2017-2019 Valve Corporation\n' @@ -90,6 +91,7 @@ def outputGeneratedHeaderWarning(self): generated_warning += '// *********** THIS FILE IS GENERATED - DO NOT EDIT ***********\n' generated_warning += '// See loader_source_generator.py for modifications\n' generated_warning += '// ************************************************************\n' + # REUSE-IgnoreEnd write(generated_warning, file=self.outFile) # Call the base class to properly begin the file, and then add @@ -121,7 +123,7 @@ def beginFile(self, genOpts): preamble += '#include "loader_logger.hpp"\n' preamble += '#include "loader_platform.hpp"\n' preamble += '#include "runtime_interface.hpp"\n' - preamble += '#include "xr_generated_dispatch_table.h"\n\n' + preamble += '#include "xr_generated_dispatch_table_core.h"\n\n' preamble += '#include "xr_dependencies.h"\n' preamble += '#include \n' diff --git a/src/scripts/src_genxr.py b/src/scripts/src_genxr.py index 6bab4654..922e9d90 100755 --- a/src/scripts/src_genxr.py +++ b/src/scripts/src_genxr.py @@ -96,6 +96,7 @@ def makeGenOpts(args): emitExtensionsPat = makeREstring(emitExtensions, allExtensions) featuresPat = makeREstring(features, allFeatures) + # REUSE-IgnoreStart # Copyright text prefixing all headers (list of strings). prefixStrings = [ '/*', @@ -115,6 +116,7 @@ def makeGenOpts(args): '*/', '' ] + # REUSE-IgnoreEnd # Text specific to OpenXR headers xrPrefixStrings = [ @@ -191,37 +193,29 @@ def makeGenOpts(args): apientryp = 'XRAPI_PTR *',) ] - genOpts['xr_generated_dispatch_table.h'] = [ - UtilitySourceOutputGenerator, - AutomaticSourceGeneratorOptions( - conventions = conventions, - filename = 'xr_generated_dispatch_table.h', - directory = directory, - apiname = 'openxr', - profile = None, - versions = featuresPat, - emitversions = featuresPat, - defaultExtensions = 'openxr', - addExtensions = None, - removeExtensions = None, - emitExtensions = emitExtensionsPat) - ] + DISPATCH_TABLE_FILES = [ + 'xr_generated_dispatch_table.h', + 'xr_generated_dispatch_table.c', + 'xr_generated_dispatch_table_core.h', + 'xr_generated_dispatch_table_core.c', + ] - genOpts['xr_generated_dispatch_table.c'] = [ - UtilitySourceOutputGenerator, - AutomaticSourceGeneratorOptions( - conventions = conventions, - filename = 'xr_generated_dispatch_table.c', - directory = directory, - apiname = 'openxr', - profile = None, - versions = featuresPat, - emitversions = featuresPat, - defaultExtensions = 'openxr', - addExtensions = None, - removeExtensions = None, - emitExtensions = emitExtensionsPat) - ] + for filename in DISPATCH_TABLE_FILES: + genOpts[filename] = [ + UtilitySourceOutputGenerator, + AutomaticSourceGeneratorOptions( + conventions = conventions, + filename = filename, + directory = directory, + apiname = 'openxr', + profile = None, + versions = featuresPat, + emitversions = featuresPat, + defaultExtensions = 'openxr', + addExtensions = None, + removeExtensions = None, + emitExtensions = emitExtensionsPat) + ] genOpts['xr_generated_loader.hpp'] = [ LoaderSourceOutputGenerator, diff --git a/src/scripts/utility_source_generator.py b/src/scripts/utility_source_generator.py index c5728697..9ecb0c6b 100755 --- a/src/scripts/utility_source_generator.py +++ b/src/scripts/utility_source_generator.py @@ -29,15 +29,17 @@ class UtilitySourceOutputGenerator(AutomaticSourceOutputGenerator): # Override the base class header warning so the comment indicates this file. # self the UtilitySourceOutputGenerator object def outputGeneratedHeaderWarning(self): + # REUSE-IgnoreStart generated_warning = '' generated_warning += '// Copyright (c) 2017-2023, The Khronos Group Inc.\n' - generated_warning += '// Copyright (c) 2017-2019 Valve Corporation\n' - generated_warning += '// Copyright (c) 2017-2019 LunarG, Inc.\n' + generated_warning += '// Copyright (c) 2017-2019, Valve Corporation\n' + generated_warning += '// Copyright (c) 2017-2019, LunarG, Inc.\n\n' # Broken string is to avoid confusing the REUSE tool here. - generated_warning += '// SPDX-License-' + 'Identifier: Apache-2.0 OR MIT\n' + generated_warning += '// SPDX-License-' + 'Identifier: Apache-2.0 OR MIT\n\n' generated_warning += '// *********** THIS FILE IS GENERATED - DO NOT EDIT ***********\n' generated_warning += '// See utility_source_generator.py for modifications\n' generated_warning += '// ************************************************************\n' + # REUSE-IgnoreEnd write(generated_warning, file=self.outFile) # Call the base class to properly begin the file, and then add @@ -47,15 +49,24 @@ def outputGeneratedHeaderWarning(self): def beginFile(self, genOpts): AutomaticSourceOutputGenerator.beginFile(self, genOpts) preamble = '' - if self.genOpts.filename == 'xr_generated_dispatch_table.h': - preamble += '#pragma once\n' + if self.genOpts.filename == 'xr_generated_dispatch_table_core.h': + preamble += '#pragma once\n\n' + preamble += '#include \n' + elif self.genOpts.filename == 'xr_generated_dispatch_table.h': + preamble += '#pragma once\n\n' + preamble += '#include "xr_dependencies.h"\n' + preamble += '#include \n' + preamble += '#include \n' + elif self.genOpts.filename == 'xr_generated_dispatch_table_core.c': + preamble += '#include "xr_generated_dispatch_table_core.h"\n' elif self.genOpts.filename == 'xr_generated_dispatch_table.c': preamble += '#include \n' preamble += '#include "xr_generated_dispatch_table.h"\n' + else: + raise RuntimeError("Unknown filename! " + self.genOpts.filename) + + preamble += '\n' - preamble += '#include "xr_dependencies.h"\n' - preamble += '#include \n' - preamble += '#include \n\n' write(preamble, file=self.outFile) # Write out all the information for the appropriate file, @@ -68,9 +79,14 @@ def endFile(self): file_data += 'extern "C" { \n' file_data += '#endif\n' - if self.genOpts.filename == 'xr_generated_dispatch_table.h': + if self.genOpts.filename == 'xr_generated_dispatch_table_core.h': file_data += self.outputDispatchTable() file_data += self.outputDispatchPrototypes() + elif self.genOpts.filename == 'xr_generated_dispatch_table.h': + file_data += self.outputDispatchTable() + file_data += self.outputDispatchPrototypes() + elif self.genOpts.filename == 'xr_generated_dispatch_table_core.c': + file_data += self.outputDispatchTableHelper() elif self.genOpts.filename == 'xr_generated_dispatch_table.c': file_data += self.outputDispatchTableHelper() else: @@ -115,6 +131,16 @@ def outputDispatchTable(self): commands = self.ext_commands for cur_cmd in commands: + if self.genOpts.filename == 'xr_generated_dispatch_table_core.h': + if self.isCoreExtensionName(cur_cmd.ext_name): + pass + # Loader implements XR_EXT_debug_utils + elif cur_cmd.ext_name == 'XR_EXT_debug_utils': + pass + else: + # Skip anything that is not core or XR_EXT_debug_utils in the loader dispatch table + continue + # If we've switched to a new "feature" print out a comment on what it is. Usually, # this is a group of core commands or a group of commands in an extension. if cur_cmd.ext_name != cur_extension_name: @@ -168,6 +194,16 @@ def outputDispatchTableHelper(self): if cur_cmd.name in self.no_trampoline_or_terminator: continue + if self.genOpts.filename == 'xr_generated_dispatch_table_core.c': + if self.isCoreExtensionName(cur_cmd.ext_name): + pass + # Loader implements XR_EXT_debug_utils + elif cur_cmd.ext_name == 'XR_EXT_debug_utils': + pass + else: + # Skip anything that is not core or XR_EXT_debug_utils in the loader dispatch table + continue + # If we've switched to a new "feature" print out a comment on what it is. Usually, # this is a group of core commands or a group of commands in an extension. if cur_cmd.ext_name != cur_extension_name: diff --git a/src/scripts/validation_layer_generator.py b/src/scripts/validation_layer_generator.py index c928838d..c037afcc 100644 --- a/src/scripts/validation_layer_generator.py +++ b/src/scripts/validation_layer_generator.py @@ -1,6 +1,6 @@ #!/usr/bin/python3 -i # -# Copyright (c) 2017 The Khronos Group Inc. +# Copyright (c) 2017-2023, The Khronos Group Inc. # Copyright (c) 2017 Valve Corporation # Copyright (c) 2017 LunarG, Inc. # From 4fa1925a7c2f35a875c7d0af06f3731ef232fbff Mon Sep 17 00:00:00 2001 From: Ryan Pavlik Date: Mon, 18 Sep 2023 16:16:53 -0500 Subject: [PATCH 4/4] OpenXR CTS 1.0.29.0 (2023-09-07) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Conformance Tests - Fix: Use actual acquired image index in swapchain rendering test. (internal MR 2746) - Fix: Do not use Catch2 assertion macros in graphics plugin methods that may be called before the first test case execution begins. (internal MR 2756, internal issue 1387) - Fix: spelling. (internal MR 2766) - Fix: Fix element contents in Android manifest. (internal MR 2840, internal issue 2053) - Fix: Allow building CTS with mingw compiler. (internal MR 2850) - Fix: Do not create an XrInstance during XML writing process, to prevent possible crash if one already exists. (internal MR 2927) - Improvement: Refactor utilities that do not depend on Catch2 into a separate internal library. (internal MR 2669) - Improvement: Refactor and standardize creation of swapchain image format tables, fixing some Vulkan invalid usage. (internal MR 2685, internal issue 1978) - Improvement: Make composition test help/example world locked but based on initial view, for more natural reading. (internal MR 2689) - Improvement: Cleanup and code quality work. (internal MR 2704, internal MR 2717, internal MR 2784, internal MR 2785, internal MR 2808, internal MR 2809) - Improvement: Add separate license file for gradlew and gradlew.bat (internal MR 2725) - Improvement: Optionally poll xrGetSystem before running test cases. (internal MR 2735, OpenXR-CTS issue 53, internal issue 1947) - Improvement: Select the first enumerated environment blend mode by default, rather than always using “opaque” (internal MR 2736, internal issue 1950) - Improvement: Migrate more tests to use the SKIP macro when appropriate. (internal MR 2737, internal issue 1932) - Improvement: Change background color based on selected blend mode: black for additive and transparent for alpha blend. (internal MR 2883, internal issue 1949) - Improvement: Add extra information to errors in case of CTS timeouts. (internal MR 2889) - Improvement: Remove conditional XR_KHR_headless support as the extension is not part of OpenXR 1.0. (internal MR 2901) - Improvement: Remove empty XR_EXT_performance_settings test that was never implemented (internal MR 2902) - Improvement: Fix names of tests to not have spaces, and adjust tags so that the instructions in the README will cause all tests to be executed. (internal MR 2924) - New test: Verify two-call idiom behavior of XR_MSFT_controller_model as well as handling of invalid model keys. (internal MR 2387, internal MR 2858) - New test: Added XR_EXT_plane_detection extension. (internal MR 2510, internal MR 2751, internal MR 2676) - New test: Add non-interactive test for XR_EXT_palm_pose vendor extension. (internal MR 2672) - New test: Add joint query to non-interactive test for XR_EXT_hand_tracking. (internal MR 2729, internal MR 2795, internal MR 2858, internal MR 2916) - New test: Add test for calling xrAcquireSwapchainImage multiple times without calling xrEndFrame. (internal MR 2730) - New test: Add additional tests for XR_EXT_debug_utils based on the test app loader_test. (internal MR 2775) - New test: Add checks for palm position and palm and wrist orientation to XR_EXT_hand_tracking interactive tests. (internal MR 2798) - New test: Add unbound action set to action sets test. (internal MR 2862, internal issue 2043) - New test: Add conformance test for calling xrDestroyInstance from a different thread to xrCreateInstance, and xrDestroySession on a different thread to xrCreateSession. (internal MR 2863) - New test: Add interactive conformance test for infrequently updated swapchains. (internal MR 2873) - New test: Add conformance tests for xrCreateSession failing, then passing (internal MR 2884) - New test: Test xrSyncActions with no active action sets. (internal MR 2903) - New test: Test calling xrLocateSpace with timestamps up to 1s old. (internal MR 2904) GitOrigin-RevId: 2b2967dda2f7cc444561b8547fedf70a5613c6a1 --- .reuse/dep5 | 4 +- CHANGELOG.CTS.md | 104 +++++++++++++++ changes/conformance/mr.2387.gl.md | 4 - changes/conformance/mr.2510.gl.md | 5 - changes/conformance/mr.2669.gl.md | 1 - changes/conformance/mr.2672.gl.md | 1 - changes/conformance/mr.2685.gl.md | 4 - changes/conformance/mr.2689.gl.md | 1 - changes/conformance/mr.2704.gl.md | 7 - changes/conformance/mr.2717.gl.md | 1 - changes/conformance/mr.2725.gl.md | 1 - changes/conformance/mr.2729.gl.md | 5 - changes/conformance/mr.2730.gl.md | 1 - changes/conformance/mr.2735.gl.md | 5 - changes/conformance/mr.2736.gl.md | 4 - changes/conformance/mr.2737.gl.md | 4 - changes/conformance/mr.2746.gl.md | 1 - changes/conformance/mr.2756.gl.md | 4 - changes/conformance/mr.2766.gl.md | 1 - changes/conformance/mr.2775.gl.md | 1 - changes/conformance/mr.2840.gl.md | 4 - changes/conformance/mr.2850.gl.md | 1 - src/conformance/build.gradle | 34 ++--- .../composition_examples/stale_swapchain.png | Bin 0 -> 60387 bytes .../stale_swapchain.png.license | 3 + .../conformance_test/test_FrameSubmission.cpp | 4 +- .../conformance_test/test_HapticInterrupt.cpp | 2 +- .../test_InteractiveThrow.cpp | 2 +- .../test_LayerComposition.cpp | 60 +++++++++ .../conformance_test/test_Swapchains.cpp | 9 +- .../test_XR_EXT_debug_utils.cpp | 14 +- .../test_XR_EXT_eye_gaze_interaction.cpp | 4 +- .../test_XR_EXT_hand_tracking.cpp | 95 +++++++++++--- .../test_XR_EXT_local_floor.cpp | 9 +- .../test_XR_EXT_palm_pose.cpp | 7 +- .../test_XR_EXT_performance_settings.cpp | 28 ---- .../test_XR_KHR_D3D11_enable.cpp | 65 +++++++++- .../test_XR_KHR_D3D12_enable.cpp | 66 +++++++++- .../test_XR_KHR_OpenGL_ES_enable.cpp | 62 ++++++++- .../test_XR_KHR_OpenGL_enable.cpp | 71 ++++++++-- .../test_XR_KHR_composition_layer_cube.cpp | 10 +- ...test_XR_KHR_composition_layer_cylinder.cpp | 11 +- .../test_XR_KHR_composition_layer_depth.cpp | 10 +- ...test_XR_KHR_composition_layer_equirect.cpp | 10 +- .../test_XR_KHR_convert_timespec_time.cpp | 7 +- .../conformance_test/test_XR_KHR_headless.cpp | 72 ----------- .../test_XR_KHR_visibility_mask.cpp | 11 +- .../test_XR_KHR_vulkan_enable.cpp | 64 ++++++++- .../test_XR_KHR_vulkan_enable2.cpp | 64 ++++++++- ...win32_convert_performance_counter_time.cpp | 7 +- .../test_XR_META_performance_metrics.cpp | 9 +- .../conformance_test/test_actions.cpp | 65 ++++++++-- .../conformance_test/test_multithreading.cpp | 7 +- .../test_xrCreateInstance.cpp | 14 ++ .../conformance_test/test_xrCreateSession.cpp | 30 ++++- .../conformance_test/test_xrLocateSpace.cpp | 39 ++++-- .../conformance_test/test_xrLocateViews.cpp | 12 +- src/conformance/framework/composition_utils.h | 2 + .../framework/conformance_framework.cpp | 20 ++- .../framework/conformance_framework.h | 6 +- .../framework/conformance_utils.cpp | 121 +++++++++--------- src/conformance/framework/conformance_utils.h | 24 ++-- src/conformance/framework/graphics_plugin.h | 12 +- .../framework/graphics_plugin_d3d11.cpp | 7 +- .../framework/graphics_plugin_d3d12.cpp | 7 +- .../framework/graphics_plugin_opengl.cpp | 7 +- .../framework/graphics_plugin_opengles.cpp | 7 +- .../framework/graphics_plugin_vulkan.cpp | 13 +- .../framework/mesh_projection_layer.cpp | 2 +- .../framework/xml_test_environment.cpp | 1 - src/conformance/gradle.properties | 10 ++ .../gradle/wrapper/gradle-wrapper.jar | Bin 60756 -> 61608 bytes .../gradle/wrapper/gradle-wrapper.properties | 1 + src/conformance/gradlew | 12 +- src/conformance/gradlew.bat | 1 + .../platform_specific/AndroidManifest.xml | 9 +- src/conformance/settings.gradle | 17 +++ src/external/d3dx12/LICENSE | 34 ++--- src/external/d3dx12/README.md | 59 --------- src/external/d3dx12/d3dx12.h | 79 ++++++++---- 80 files changed, 1012 insertions(+), 580 deletions(-) delete mode 100644 changes/conformance/mr.2387.gl.md delete mode 100644 changes/conformance/mr.2510.gl.md delete mode 100644 changes/conformance/mr.2669.gl.md delete mode 100644 changes/conformance/mr.2672.gl.md delete mode 100644 changes/conformance/mr.2685.gl.md delete mode 100644 changes/conformance/mr.2689.gl.md delete mode 100644 changes/conformance/mr.2704.gl.md delete mode 100644 changes/conformance/mr.2717.gl.md delete mode 100644 changes/conformance/mr.2725.gl.md delete mode 100644 changes/conformance/mr.2729.gl.md delete mode 100644 changes/conformance/mr.2730.gl.md delete mode 100644 changes/conformance/mr.2735.gl.md delete mode 100644 changes/conformance/mr.2736.gl.md delete mode 100644 changes/conformance/mr.2737.gl.md delete mode 100644 changes/conformance/mr.2746.gl.md delete mode 100644 changes/conformance/mr.2756.gl.md delete mode 100644 changes/conformance/mr.2766.gl.md delete mode 100644 changes/conformance/mr.2775.gl.md delete mode 100644 changes/conformance/mr.2840.gl.md delete mode 100644 changes/conformance/mr.2850.gl.md create mode 100644 src/conformance/conformance_test/composition_examples/stale_swapchain.png create mode 100644 src/conformance/conformance_test/composition_examples/stale_swapchain.png.license delete mode 100644 src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp delete mode 100644 src/conformance/conformance_test/test_XR_KHR_headless.cpp create mode 100644 src/conformance/gradle.properties delete mode 100644 src/external/d3dx12/README.md diff --git a/.reuse/dep5 b/.reuse/dep5 index 3df6553a..cedb22dd 100644 --- a/.reuse/dep5 +++ b/.reuse/dep5 @@ -54,8 +54,8 @@ Comment: Unmodified, vendored copy of a subset of the tiny-gltf repo v2.8.9 Files: src/external/d3dx12/* Copyright: Copyright (c) Microsoft Corporation. License: MIT -Comment: Unmodified, vendored copy of DirectX-Headers commit da7aedb - of https://github.com/microsoft/DirectX-Headers filtered to just d3dx12 headers +Comment: Unmodified, vendored copy of d3dx12.h from directx-vs-templates commit 86b9c45 + This repo includes ifdefs to improve compatibility with older Windows SDK verions. Files: src/external/mikktspace/* Copyright: 2011 by Morten S. Mikkelsen diff --git a/CHANGELOG.CTS.md b/CHANGELOG.CTS.md index dd89ef9f..e09d4125 100644 --- a/CHANGELOG.CTS.md +++ b/CHANGELOG.CTS.md @@ -17,6 +17,110 @@ particular, since it is primarily software, pull requests may be integrated as they are accepted even between periodic updates. However, versions that are not signed tags on the `approved` branch are not valid for conformance submission. +## OpenXR CTS 1.0.29.0 (2023-09-07) + +- Conformance Tests + - Fix: Use actual acquired image index in swapchain rendering test. + ([internal MR 2746](https://gitlab.khronos.org/openxr/openxr/merge_requests/2746)) + - Fix: Do not use Catch2 assertion macros in graphics plugin methods that may be + called before the first test case execution begins. + ([internal MR 2756](https://gitlab.khronos.org/openxr/openxr/merge_requests/2756), + [internal issue 1387](https://gitlab.khronos.org/openxr/openxr/issues/1387)) + - Fix: spelling. + ([internal MR 2766](https://gitlab.khronos.org/openxr/openxr/merge_requests/2766)) + - Fix: Fix `` element contents in Android manifest. + ([internal MR 2840](https://gitlab.khronos.org/openxr/openxr/merge_requests/2840), + [internal issue 2053](https://gitlab.khronos.org/openxr/openxr/issues/2053)) + - Fix: Allow building CTS with mingw compiler. + ([internal MR 2850](https://gitlab.khronos.org/openxr/openxr/merge_requests/2850)) + - Fix: Do not create an `XrInstance` during XML writing process, to prevent + possible crash if one already exists. + ([internal MR 2927](https://gitlab.khronos.org/openxr/openxr/merge_requests/2927)) + - Improvement: Refactor utilities that do not depend on Catch2 into a separate + internal library. + ([internal MR 2669](https://gitlab.khronos.org/openxr/openxr/merge_requests/2669)) + - Improvement: Refactor and standardize creation of swapchain image format + tables, fixing some Vulkan invalid usage. + ([internal MR 2685](https://gitlab.khronos.org/openxr/openxr/merge_requests/2685), + [internal issue 1978](https://gitlab.khronos.org/openxr/openxr/issues/1978)) + - Improvement: Make composition test help/example world locked but based on + initial view, for more natural reading. + ([internal MR 2689](https://gitlab.khronos.org/openxr/openxr/merge_requests/2689)) + - Improvement: Cleanup and code quality work. + ([internal MR 2704](https://gitlab.khronos.org/openxr/openxr/merge_requests/2704), + [internal MR 2717](https://gitlab.khronos.org/openxr/openxr/merge_requests/2717), + [internal MR 2784](https://gitlab.khronos.org/openxr/openxr/merge_requests/2784), + [internal MR 2785](https://gitlab.khronos.org/openxr/openxr/merge_requests/2785), + [internal MR 2808](https://gitlab.khronos.org/openxr/openxr/merge_requests/2808), + [internal MR 2809](https://gitlab.khronos.org/openxr/openxr/merge_requests/2809)) + - Improvement: Add separate license file for gradlew and gradlew.bat + ([internal MR 2725](https://gitlab.khronos.org/openxr/openxr/merge_requests/2725)) + - Improvement: Optionally poll `xrGetSystem` before running test cases. + ([internal MR 2735](https://gitlab.khronos.org/openxr/openxr/merge_requests/2735), + [OpenXR-CTS issue 53](https://github.com/KhronosGroup/OpenXR-CTS/issues/53), + [internal issue 1947](https://gitlab.khronos.org/openxr/openxr/issues/1947)) + - Improvement: Select the first enumerated environment blend mode by default, + rather than always using "opaque" + ([internal MR 2736](https://gitlab.khronos.org/openxr/openxr/merge_requests/2736), + [internal issue 1950](https://gitlab.khronos.org/openxr/openxr/issues/1950)) + - Improvement: Migrate more tests to use the `SKIP` macro when appropriate. + ([internal MR 2737](https://gitlab.khronos.org/openxr/openxr/merge_requests/2737), + [internal issue 1932](https://gitlab.khronos.org/openxr/openxr/issues/1932)) + - Improvement: Change background color based on selected blend mode: black for + additive and transparent for alpha blend. + ([internal MR 2883](https://gitlab.khronos.org/openxr/openxr/merge_requests/2883), + [internal issue 1949](https://gitlab.khronos.org/openxr/openxr/issues/1949)) + - Improvement: Add extra information to errors in case of CTS timeouts. + ([internal MR 2889](https://gitlab.khronos.org/openxr/openxr/merge_requests/2889)) + - Improvement: Remove conditional `XR_KHR_headless` support as the extension is + not part of OpenXR 1.0. + ([internal MR 2901](https://gitlab.khronos.org/openxr/openxr/merge_requests/2901)) + - Improvement: Remove empty `XR_EXT_performance_settings` test that was never + implemented + ([internal MR 2902](https://gitlab.khronos.org/openxr/openxr/merge_requests/2902)) + - Improvement: Fix names of tests to not have spaces, and adjust tags so that the + instructions in the README will cause all tests to be executed. + ([internal MR 2924](https://gitlab.khronos.org/openxr/openxr/merge_requests/2924)) + - New test: Verify two-call idiom behavior of `XR_MSFT_controller_model` as well + as handling of invalid model keys. + ([internal MR 2387](https://gitlab.khronos.org/openxr/openxr/merge_requests/2387), + [internal MR 2858](https://gitlab.khronos.org/openxr/openxr/merge_requests/2858)) + - New test: Added `XR_EXT_plane_detection` extension. + ([internal MR 2510](https://gitlab.khronos.org/openxr/openxr/merge_requests/2510), + [internal MR 2751](https://gitlab.khronos.org/openxr/openxr/merge_requests/2751), + [internal MR 2676](https://gitlab.khronos.org/openxr/openxr/merge_requests/2676)) + - New test: Add non-interactive test for `XR_EXT_palm_pose` vendor extension. + ([internal MR 2672](https://gitlab.khronos.org/openxr/openxr/merge_requests/2672)) + - New test: Add joint query to non-interactive test for `XR_EXT_hand_tracking`. + ([internal MR 2729](https://gitlab.khronos.org/openxr/openxr/merge_requests/2729), + [internal MR 2795](https://gitlab.khronos.org/openxr/openxr/merge_requests/2795), + [internal MR 2858](https://gitlab.khronos.org/openxr/openxr/merge_requests/2858), + [internal MR 2916](https://gitlab.khronos.org/openxr/openxr/merge_requests/2916)) + - New test: Add test for calling `xrAcquireSwapchainImage` multiple times without + calling `xrEndFrame`. + ([internal MR 2730](https://gitlab.khronos.org/openxr/openxr/merge_requests/2730)) + - New test: Add additional tests for `XR_EXT_debug_utils` based on the test app + `loader_test`. + ([internal MR 2775](https://gitlab.khronos.org/openxr/openxr/merge_requests/2775)) + - New test: Add checks for palm position and palm and wrist orientation to + `XR_EXT_hand_tracking` interactive tests. + ([internal MR 2798](https://gitlab.khronos.org/openxr/openxr/merge_requests/2798)) + - New test: Add unbound action set to action sets test. + ([internal MR 2862](https://gitlab.khronos.org/openxr/openxr/merge_requests/2862), + [internal issue 2043](https://gitlab.khronos.org/openxr/openxr/issues/2043)) + - New test: Add conformance test for calling `xrDestroyInstance` from a different + thread to `xrCreateInstance`, and `xrDestroySession` on a different thread to + `xrCreateSession`. + ([internal MR 2863](https://gitlab.khronos.org/openxr/openxr/merge_requests/2863)) + - New test: Add interactive conformance test for infrequently updated swapchains. + ([internal MR 2873](https://gitlab.khronos.org/openxr/openxr/merge_requests/2873)) + - New test: Add conformance tests for `xrCreateSession` failing, then passing + ([internal MR 2884](https://gitlab.khronos.org/openxr/openxr/merge_requests/2884)) + - New test: Test `xrSyncActions` with no active action sets. + ([internal MR 2903](https://gitlab.khronos.org/openxr/openxr/merge_requests/2903)) + - New test: Test calling `xrLocateSpace` with timestamps up to 1s old. + ([internal MR 2904](https://gitlab.khronos.org/openxr/openxr/merge_requests/2904)) + ## OpenXR CTS 1.0.27.0 (2023-05-10) This release contains a large number of new or improved tests. It is expected diff --git a/changes/conformance/mr.2387.gl.md b/changes/conformance/mr.2387.gl.md deleted file mode 100644 index 988c6921..00000000 --- a/changes/conformance/mr.2387.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- mr.2858.gl ---- -New test: Verify two-call idiom behavior of `XR_MSFT_controller_model` as well as handling of invalid model keys. diff --git a/changes/conformance/mr.2510.gl.md b/changes/conformance/mr.2510.gl.md deleted file mode 100644 index bd57edb1..00000000 --- a/changes/conformance/mr.2510.gl.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -- mr.2751.gl -- mr.2676.gl ---- -New test: Added `XR_EXT_plane_detection` extension. diff --git a/changes/conformance/mr.2669.gl.md b/changes/conformance/mr.2669.gl.md deleted file mode 100644 index abcfc994..00000000 --- a/changes/conformance/mr.2669.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Refactor utilities that do not depend on Catch2 into a separate internal library. diff --git a/changes/conformance/mr.2672.gl.md b/changes/conformance/mr.2672.gl.md deleted file mode 100644 index a703b406..00000000 --- a/changes/conformance/mr.2672.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Add non-interactive test for XR_EXT_palm_pose vendor extension. diff --git a/changes/conformance/mr.2685.gl.md b/changes/conformance/mr.2685.gl.md deleted file mode 100644 index c77f7c89..00000000 --- a/changes/conformance/mr.2685.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- issue.1978.gl ---- -Improvement: Refactor and standardize creation of swapchain image format tables, fixing some Vulkan invalid usage. diff --git a/changes/conformance/mr.2689.gl.md b/changes/conformance/mr.2689.gl.md deleted file mode 100644 index 07b3e8bd..00000000 --- a/changes/conformance/mr.2689.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Make composition test help/example world locked but based on initial view, for more natural reading. diff --git a/changes/conformance/mr.2704.gl.md b/changes/conformance/mr.2704.gl.md deleted file mode 100644 index a792439d..00000000 --- a/changes/conformance/mr.2704.gl.md +++ /dev/null @@ -1,7 +0,0 @@ ---- -- mr.2784.gl -- mr.2785.gl -- mr.2808.gl -- mr.2809.gl ---- -Improvement: Cleanup and code quality work. diff --git a/changes/conformance/mr.2717.gl.md b/changes/conformance/mr.2717.gl.md deleted file mode 100644 index cb4f77b3..00000000 --- a/changes/conformance/mr.2717.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Fix clang warnings `-Wundef` and `-Wmissing-braces`. diff --git a/changes/conformance/mr.2725.gl.md b/changes/conformance/mr.2725.gl.md deleted file mode 100644 index e33f542f..00000000 --- a/changes/conformance/mr.2725.gl.md +++ /dev/null @@ -1 +0,0 @@ -gradle: Add license for gradlew and gradlew.bat diff --git a/changes/conformance/mr.2729.gl.md b/changes/conformance/mr.2729.gl.md deleted file mode 100644 index 7e44ad1a..00000000 --- a/changes/conformance/mr.2729.gl.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -- mr.2795.gl -- mr.2858.gl ---- -Improvement: Add joint query to non-interactive test for `XR_EXT_hand_tracking`. diff --git a/changes/conformance/mr.2730.gl.md b/changes/conformance/mr.2730.gl.md deleted file mode 100644 index a7720525..00000000 --- a/changes/conformance/mr.2730.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Add test for calling `xrAcquireSwapchainImage` multiple times without calling `xrEndFrame`. diff --git a/changes/conformance/mr.2735.gl.md b/changes/conformance/mr.2735.gl.md deleted file mode 100644 index d7f49116..00000000 --- a/changes/conformance/mr.2735.gl.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -- issue.53.gh.OpenXR-CTS -- issue.1947.gl ---- -Improvement: Optionally poll `xrGetSystem` before running test cases. diff --git a/changes/conformance/mr.2736.gl.md b/changes/conformance/mr.2736.gl.md deleted file mode 100644 index 29a8c04a..00000000 --- a/changes/conformance/mr.2736.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- issue.1950.gl ---- -Improvement: Select the first enumerated environment blend mode by default, rather than always using "opaque" diff --git a/changes/conformance/mr.2737.gl.md b/changes/conformance/mr.2737.gl.md deleted file mode 100644 index 1f289d4d..00000000 --- a/changes/conformance/mr.2737.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- issue.1932.gl ---- -Improvement: Migrate more tests to use the `SKIP` macro when appropriate. diff --git a/changes/conformance/mr.2746.gl.md b/changes/conformance/mr.2746.gl.md deleted file mode 100644 index 34bc4818..00000000 --- a/changes/conformance/mr.2746.gl.md +++ /dev/null @@ -1 +0,0 @@ -Fix: Use actual acquired image index in swapchain rendering test. diff --git a/changes/conformance/mr.2756.gl.md b/changes/conformance/mr.2756.gl.md deleted file mode 100644 index ddfb77dc..00000000 --- a/changes/conformance/mr.2756.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- issue.1387.gl ---- -Fix: Do not use Catch2 assertion macros in graphics plugin methods that may be called before the first test case execution begins. diff --git a/changes/conformance/mr.2766.gl.md b/changes/conformance/mr.2766.gl.md deleted file mode 100644 index 944b119d..00000000 --- a/changes/conformance/mr.2766.gl.md +++ /dev/null @@ -1 +0,0 @@ -Fix: spelling. diff --git a/changes/conformance/mr.2775.gl.md b/changes/conformance/mr.2775.gl.md deleted file mode 100644 index c2164b35..00000000 --- a/changes/conformance/mr.2775.gl.md +++ /dev/null @@ -1 +0,0 @@ -Improvement: Add additional tests for XR_EXT_debug_utils based on the test app loader_test. diff --git a/changes/conformance/mr.2840.gl.md b/changes/conformance/mr.2840.gl.md deleted file mode 100644 index 9860ffb9..00000000 --- a/changes/conformance/mr.2840.gl.md +++ /dev/null @@ -1,4 +0,0 @@ ---- -- issue.2053.gl ---- -- Fix: Fix `` element contents in Android manifest. diff --git a/changes/conformance/mr.2850.gl.md b/changes/conformance/mr.2850.gl.md deleted file mode 100644 index 85701ee8..00000000 --- a/changes/conformance/mr.2850.gl.md +++ /dev/null @@ -1 +0,0 @@ -Fix building CTS with mingw compiler. diff --git a/src/conformance/build.gradle b/src/conformance/build.gradle index 8276908d..4a4c7060 100644 --- a/src/conformance/build.gradle +++ b/src/conformance/build.gradle @@ -4,22 +4,11 @@ // Open this directory in Android Studio, or build with Gradle, // to build the conformance suite. -buildscript { - repositories { - google() - mavenCentral() - } - dependencies { - classpath 'com.android.tools.build:gradle:7.4.2' - } -} -repositories { - google() - mavenCentral() +plugins { + id 'com.android.application' version '7.4.2' } -apply plugin: 'com.android.application' // These next few lines are just to make the version match the OpenXR release. project.ext.repoRoot = file('../../') @@ -27,33 +16,30 @@ apply from: file('../version.gradle') def assetDest = layout.buildDir.dir('intermediates/assets') -task copyFont(type: Copy) { +task copyAssets(type: Copy) { + // fonts from('conformance_test') { include '*.otf' } - into assetDest -} -task copyAssets(type: Copy) { + // sample images from('conformance_test/composition_examples') { include '*.png' } - into assetDest - dependsOn copyFont -} -task copyLayerInfo(type: Copy) { + // layer manifests from('conformance_test/android_assets') { include '**/*.json' } + into assetDest - dependsOn copyFont } android { compileSdk 29 ndkVersion "21.4.7075529" buildToolsVersion = "30.0.3" + namespace 'org.khronos.openxr.cts' defaultConfig { applicationId "org.khronos.openxr.cts" @@ -65,8 +51,7 @@ android { externalNativeBuild { cmake { - arguments '-DANDROID_STL=c++_shared', - '-DBUILD_TESTS=OFF', + arguments '-DBUILD_TESTS=OFF', '-DBUILD_LOADER=ON', '-DBUILD_CONFORMANCE_TESTS=ON', '-DBUILD_ALL_EXTENSIONS=ON' @@ -74,7 +59,6 @@ android { } } preBuild.dependsOn(copyAssets) - preBuild.dependsOn(copyLayerInfo) } sourceSets { main { diff --git a/src/conformance/conformance_test/composition_examples/stale_swapchain.png b/src/conformance/conformance_test/composition_examples/stale_swapchain.png new file mode 100644 index 0000000000000000000000000000000000000000..91afdfb844a5abe8d530b47a547f488654902fba GIT binary patch literal 60387 zcmeFZcT`j9+b)cw=!gT1gA_qz{HZ8ifq)W<4UjG+^p3QIfbCDAzf1LQu zqxZ$XlwWeGY2+B#^K&EXL@D!xS>fyo2}R!01f5#y*uCrLjL0!79&ajyS}vTt7ytXO z&maBkk9eumZ(hAWdE%Ez_D0JpzQk`*gRjo|W!XX|Lnd)Umdq_sjMYn1veyKzg*!Kx<8t@Hm`Z>G?TSm#fEjm zfdH3;dCkuR4e`5c4V(B8hm3z8u$Z56wA^>JtaCJ@BRJWYX8jw}C&JU&7QZkh(Wb2d zjXtI017!8MPs8R!;93kf_>lX?W#5mVj&oZDt%K9m?o379`1f()uB$COxuM1ftMzM5 zQW);UxD!m)hlYPnxK#GYOa^Ug$&BQ=&Cf=(zr4-F#P=S?LJ(MQo5EC2?Dfb_25t4L z3IzRZF|zcZm2KYb$vefwRDxysY=*>H`;7l zc$l8EYP9&RHExUZWPA4B^*s9@SAwTdNk&$Uiv<}+b8bg-^Ge6NQ|(bVt}`)N2jhBV zMlbn}xmz7=q$q9I_5n*<{+EmC*109lVdXPok8RlaI_lPZ*m_RlU_n)BZ%($^pQx*< zrFa@#(lB^99{tMPhwWAe?TI>yyQ@|6*`Y^flY8WuY1kE>lT2klOi9)UN0ad17|Z0- zjrRgvCmG*;k7#GSFzwg?+kvn3Hru~n56XD{epM1Z&3OGY#Q5(6pZ~YRKC?%iVEq2) zzYi-YkjXqajzBga(zz|X`tP6NuL5i0aBrwNWTW%OTGRe&dla|Q;nrZ{AK5PFxI~A8 zHhX+0gSQXqqidf|vX6xxA8uP@HEvG?4q5n)J7geb#=Qn7H8U=;BUD9@5&S+*KX8&A zEMxxjnblx*DhAAF`ho@2sO$2otmAUvCGemvtZ23!{Jtd!?-u1#pDrOAY2=k~)l2T5 z3zZHRSMZ~yz7ya>m$1!8D0C?=^PKiA%eyV9v_!B%+fb+NlteB0!ShwKslIa5?WpJp zY|J09@DVwo;k?Cm=lgSy%`cz9W1u6jjWTyHlVeNAa-N;5F#B{R;?h9lhcHvYknM5b zNx$is(pj}ynz2qs{R-ZU+sKFvn+p)XK zQyKC{^a*QYi4fxk>z0>#)9MD+uwygV2vPfumX1>WYFWl!TgZ)5T>s3WbLYojKC9t@ zLdco@wdT+_c?wOBHmGQ<;%2v0qOkSK!o?Dt?{ay^<T705k>4!-W0UIS|0c~R)sR)P~AU1IxHqiHuhndWQ{cWvDJ-A{ezEPihN+otxd z)6Z@AkIQO;(c|blcS0&Z#H^&+#hO$(=~@MjXCU3Chgk6AKBIS!*3&3u)3MOi+W9PR z7w}-m!6s+8g_q79$&ediiYl3PuB(-OinJxL>ChQqV_7$5m4oMJ;D9&Ni;3Z6TR_M|06wwHXe}3Pv@uqJ zKD-IMk%-mqSAoNRf#7Mr$(?E70*?bJ%{$!3Q`yIxp+_`q=pp;@Pxj`Wc8M%^!o~c~ zuiUY+BTwynZvLLqi>8-U_SXGuJ=dqWzl=Yy^&Oqb``l4_Nh!e<>Sa@1b$fcmF*|s7 zHf6u+_6S&f)x)+?O)_^x)~(g)1~8>wZSf1Y?v}g8bMg~4E&GGaz%yN1!)7mEIc)}PeBiZXSe> zM?>xkt(S{%H1AQ-Itn!6Np|^XU^9)$bgfiR9{p@%Pf3~#Si*R3?DLo1UqUVJ1vpat zZf_z36ok$&Er~F$?Ki}OTc@ovQ-RUlBoE_EFZq3c5yd4YM)+GkG;2PCQ@{4(iEIs^93CDHyU^%To4sYzpQKhnXHAnWIpJcnKvV8;$ zi}0EU_GCDT(4)<4EBAKJ$^B2;&BuoVM?WKcK>!2K(@M~i)EvUL10LM?)mes11Y1Ua zd(0E8{|ymr!vM5-kAw|iS;&a%Wwd({vx0)Z}8LoAUDa{A757`wAESHObt;(O7|JZQ7 zUt80VqZEhVoy$nUR8NNNv#Ck!`5au#QF{|N4&vCT;vsN&#A0wQ8>#|3>v5q$z%ej( z?{xKzr?$p0;GhDGo^Tsg$ChGU`G>4RgSX#8S8JwQ&Ip)SJnhW8#8i@UYFZ5hEt{CL z1f~*>&7W;k!1V-!U8k_{;T7B6nYWfM!27v1tP||w6nwz8vGgXvly+L!oA$@oEJ08S z-LDP}Z09sOSawtrxu~?akPB9@OZmo7^O5&Rxw_$oY)VWs9QC{Toy(zZ#9O<5{KX?~ zO}lfReG1!PxsgXcves+IOmdxnZ3FSz-XUQ_yn*-g#nv2bbajm9a$ld$DnE?sV=9?H zH60&$6&M&eIO7}1*`nOy=6)bYVC-W?Bt~PK0tU5YgXf`>wP5M30q3Ch#zgz>?7PH5 zTFyz0NFDk0&y0Lwvqx5G>)S)d+KiHD+M$oS8IeE$)^(ZyPMP7KK|H9H9sRLdw^X`R zTEBXgwQsDsV(K2uam+0ZCKN*XS9)5^6)Ht3!`;YL`M_SVqs@@X+2c%f+xIZT-RhV1Bfd z!&#h*@Zc17E>g^FK2lmTGe5q#*OItVE{Xma0G}w@%}=pJRcc_mOXEx5F&^+=9NJDo zhE1xtWGe&~U4tlu^ZljkI16n!i^^J$pb#XEG>57l(AEz*?=}jiG(IDE$qte53Fito za5h`+1+NnO3Y4_m4E=Nap^%cJ0(eJ^txyv7nf`mhB42{KQw>2K5i99=J|=dkRk(!G zPps`b&$QHR;WIM3HK=n;HEQ15?F$XTHqxn?GP58foSBM}xXg&xT+#DI=lNn>;Jj)r zet+NARURF-{j#q^r(1ZQ;~ZhH*5wb%Y7Tq}7agn42!QRI*|a$hvIM2Q{+1>|~z9GH_2Z{UFMd)l|ugho+>0 z3GI<^)29-HDc6$a`XdcGAb_$lpi+;TjNAQwP4Mq|=Uf2)DQco1;s(f`}HTVwPAts3bQoCVkrL^xE zfuLt1DE;bgxdY%UT<)2kz4{hXNScglht} zRs|s7!OK5QClXw3q-(v)F0LIm#lj45w0yZGZD!L^OtypLh+GQ#GKnI*>=mOE7kd># zj#9+nV&+q~I|@k&BO*b{X?_P&dPzo}B$;1wdv$FrN~cNW1=5Bt$s^_jHbJH0$_mZz zcV2Gon8nD9+Gp^YNb&c*mYeCgCUQ%;wP3bbE}Nl_ryRAUC6V2sIGg~Nb}9MGupdOT zu`h9Paz^JqoQ^AYizRRx;G(!CN~d8^uHTsWB49nAMXvCiBY0lpk)3Xw94e1IWd{}l ztp#;@dfTq0f+8tVw~!<0r2D~Z0Iz!*Iq!tdc$YfMNTg=oKBP~E9`_Cz4iy+*Cidr7hfHNepIhG(g1s zr!X;g;a09~d&huJJ0WXGQ5Jf;%G1}nQ6vJVN?yJiduQs))wpHEo=j$yCk~c!e^d*K zlH$66dS+1ZN0OV8B5>=9v|=4nbQ6DWzon&)#*IV|5!h9}$Q+g6V*Tu+twMy&D|yOD z*@Y7wPSq`OPxa3sEFFA(km&g7iysKmuDG8?xv_G~C+)7O{({bA4ZNRM;A0)5S!wbj z%V!`e?dIT1A=CT>vF7AQS4Yg_M9mbE($b<-YEDAk;xt(eIE-G{KJj;Y=gb4uwTC}_ zI8;lxmCWDqn7%M@$0*cs%`4_9t5c6}vU}u5os7X)byZvlHN+lnxfXX;oi?;iEx8Wy zbEq|R6!gY?;^`xxN=!0yi@h;e2o4@tnZBfPQm+j08@dup%GT{P9=2nRTVFO^+S3wS z{CS!R5|-i(YDsIj{iB)l9O$wruYeH6s<-Ji;dS_!sM3$FzoDSgl+~XBMpAV%h|VqLhD*Uje<)m@#YZbf-2q)0)15jcSacWp+Vm_9eY+SL z^OSuhx5dsqHonA)7SO%l5yX6=%n*mhzp1jpso~bPwCe{zffq`G8O8kKc>^pa0kvDD z(bi+nrN=>4xs(B8C2ao8Xt&yFWEsSXYN`sRga$*a3 zlX@E2%&m4d8RWztY^MiN!Ua`UK9MSgHt^%iYG_w3D^vp><{06X>@(>I!WO6}fGY|N zS&y>no*fcUQ87=n`cQNNmuGOhLL6ibYUf;0$3-8Dp;B1`mFsL{CFgvj8(8s^dwdsQ3|N!EU5Zu4gYYBwNBndpe} zwb0{zv2Dwtytz8wNOLhvlh{epmx6jbgiY@FW4t0?)rNi|>9JwxCds;KvX5OFBepT9 z4aKlPoyDBvx4#TENZ-F|fm7@_1#{FW8u15ZMF1D7*&Aj5?dv5WAHBs0+@EI2n7;U7 zm6}T!qPiao@;2v!0$W%T^y2ycE9r(6z_^tQI^M1sk=FB`I?0 z)z&UmQNX^&8}0*NKzR}^Rd6{*^`*O6ai&&B%`=r0N2@y!Y^>7$a%E;UinnuiA^@(E z;{TAc=PcimudwTKk%nerq*vl?;pO5wimCaX$efH1u8+JM2vq~FAUjO9ZFDurC{X#p z5mSu4r?qmvFSxKutj=KX0@gWdz) zHkcyfYrqAy2=3JMVyeI~tigSOIGg%Fv(PEl^cv|&9o+OBob<$4bQJbMirmVEjVI}s zyC;*?FzH#;Z3wMVN-6WLpmffQEnhFuKlj*=mP9V9;TAZv9XJ%@A6%iEmjRILhT3E5F&|* zgs%>2E6kzuVTr&rdguU)a^AFZf~h-!YEvTBe?8r*lj5)*4wA+sIQgD~+)$sqpv>2zr@rqSWxUZGj=+65`a@JW#C zI~r7MyezjKE6nXBKcj8OTg%r^duxG$3j?<@%eR(iR}ytyqG}?=l;6|wY|e=H{J&Bp zMw3PPyq|l(?~!JENy#S76etu=kDJlerVjRhwsKDIufN;kmMzbA0|5!?EH;;hJE#<7` zZDY8Yx5Op#y=i^fbGga`4Ch1A)8IXxSbJMXRxb);eSC6aKW4^Q%Dc9A;HD3Dy7_GN}lGLO6}nb=ym6LuA1)#48aUxVp(1zhE0j|{_eH&#EqQL>B%tt zO_tA@IkHfeg!+Sua1tO_pzgG?)4SS{ z`@qq~VD$SwzGd-{zCRoLGe{`tojh2@mpnD2@>*QtNnh4)Tyje_e6{3p)o`D$T(Byf zRwv6hqtX<=_$j9wV9b`?RPRirA78$7M&<%QEmkhHr-$2KYuJNjdN*@74yuZ6@#>I554MwvH`q!-)tc=Hl?2d_Yy^kw^Fex$BW>CR^0xY`yxMEu`7 z$?rhElT1$YtD)g&O!_{>glg)ktANRtR&6ZZS1x69zA1@X@$$x~aJ>eQ#f)IEyfv|2 zd;G1TjTS#-``WE!uL#-p%pAG;d%UGhGNKyxQ|C47|AJ~@FvXncEW`c7?a4`;eqddl z>ZGcEy(X$Tc3anR@dE$wCGQmj8;I;2tI<4Fm@o7jb`JIn3ip{&^rDlx|AIuzeuR5)FiMU>QNPfs_rt4@!Fzm!A8m*{8a?A6;z zN8@S>3?Dv}r963YI|El~OKaQ^d#={V8yp{_^P6nIp)=BWF-ny)- z4l1}4#%|t$R={B?DT!XLKN`Ltn91g1Rl>Yd5E$~ak88W`)VHWsbpcojCYVmwXLZ&Q z=1V%O^3El(^jSpJ3ND6oAKv+Eg?&6rH;>ueV*_L$vW=uvy!Yx@Z(J=<_Ydmu`LI!; zKm}Vb{^i_WW3({Yso)@=@XiaO6Y=$N*M>5R*J5Uhd?u1&O!=5UbD3-ogi_Ift?h9|pnlyt=KY~*YRoiZ0kvEx_JY+gC6z&!6$jrOaD+%R@lO9?2qRQ!& zr>9#QgsvIDVv5%=*(|F*_L4lZ~%AS0c!^R=W_bZQJX!UOj4J6H6l@1UTCl;nPvPiBYld(7m4HLpvMvtWKpQ@N^Jkv^l|UNP z1yS&~PXXx0@6$&%?ty0IxsA`3vD2eXcnE&B3PO$*cG;IuN^%=e9NOnBGG5_Rg~22y zQjfSfDG8&y1euXo;}klmo7ZKfag}t34EKHWiAi~-5;DG=6AIrg^0^5g`a&UJ9HlMk zMliq_=BZxTu1u>JoyE)g?)Ssy3-CSmQ{H|;d5LtKMUW4!632?Q6cm>u;y@+FC~Jh6 z$&=cq8edN^(7!$aRq8#4NQ=hk^ep9`pW_`h9sX(9G~r!GDD)c8>} z&)HvJYZ<2Ha*@O5%J1l1)wrgmD&iR6a9IK(6VB=vgS}_5I_#ly*TJENH=nJzbY2sxePj+$Q1__ZCjF-TC0}5^Si}RdFwCOiFyTNk(#9nwKCq@s^x-8l^)1uHI{YCl{}h%Q>Kmb1Zo~QujK*vUzVJlye-{f zw7hfBS8?3O#s>a6Oyf{y^?j$`27met(ARiPJmA+v^j`}7NgkQfF!O8VD|S6-sq%N5 zafv}*9h8Q6WmXfo)fjmH`nf+9{&=^nAq5H}-!|AN^+9~q_1i~F!Q;iV{FbML^2j}8 zCmL~%;GcIu_-Mj&e7)ggQNkZEp~BjmZlsQ0Dpso4?|D zLpKt$-DCZ0_WSwd1rc z-os-!xt4}BpO#RBowb(?VzNzEijN4GpF`dBx^zSkN#+R*DxRZK>`B6*P#ZZmh#CaO z{$McssEW;py8fx`-RX}nR^I;0c^1HflwaH<;t+Y{*c+v6*p1uFti21~i$RZeb#<6m zL1F=Sx(0RQ)^0?wN*^ikdjF74uz=EX`j?!a3SKfA?G77M&21$PI#k}qUY9c3(}U}@ zOWt+pB`nLD10t&K(EdP9Gq?X@M=N@|$4HAm*JzXKH)x=*s=!pnOR1&EGEb8P0bKBM zz2bi!c@CiQ3I01a?v%eMrWE_pJEM8n`qjFZU)V!;+C)lgazyoP{q`0MNAoFHM?JpE zT`vS&@%+nk;tL3DN4DrbPgNJbo|`$c75R2H&Zp%SyJ8KGFYr~zWVb_dO4C+;=$#ec zka^!;P>Uz%kDwh=oGC(xptORTTg9$N2LApz&&Wb>y~$*4IP&D^MZO$RQm<^WFJ%H2 z0G|rGhA~cxjyB#@Pwo<|^*2bkSj#hx22DH-rw$@q2>6p=wz1P9Q;GLM*30M z+@5i0ml>`4F8uesD!)Sjta61Do_kqeTsSO@T(nAedP7dl8ST809RzCnBf8tM&&t8L zGlMf$yOBvuR4GI($Z0@)lO6JWRbDv~H>hqtRE-;$bILO#{g~|~-N+5^TFD*<*3x1a z6v|7V^ZT7@Q?RW{EIk*7=N(&8t2L3+iaz1q?fRM?R6m3yNDzP_2sm z@d~)wAJm@oen0%Lj*{*u_+K>f_vuE>rp{VVxO+_;GikMb!KLTs(N>4>tGh|qp!{=% z2$VW+=lQ6Pu)!fUZ-f;G(VD!zE=}^&R!lYYC2&|YYEOTIPREzT_GSIaWXy_u_;NkH z$fKnufLdcM%7d*Hd<^Qer0zcUD95T3)%FvB9R-lnVUm%hiR95pUkJlD#waZq@Q967 z9bRB#QOHlE69Li`LYH`eg zKmOui@PF$9Rb3D5tkynB_6(0pK}EWwgt5KndB19A7WvS70g$kl))!RJ5zdH8%r*8& z%Ibt^9XhFuzR}+bFX~lWLEJxqzDH+pBn)wxS|6}cRG%D z1df+N53$EHI!Aff(4B}-;}8nqQdF*sy?Uu}x1`9_XAyy2dhqH=M1(=z?!l|jpX^7o%?Bv}ra#^frD2aq3|f}9 zvOAxxbXnLm5EEfI{bhtS`HkkS>ASXNOT??gb?#RRCN)xrQm5ccOUo>PlExhD4X7`@ z4KV%UD!=6NP+F+Ej_{}K^WdJDadtj|7;|H&RE5i{TSB)RA0cc`lByUi&Khc0_IPg6 z>&JzcfeG@{(NY0N&J*luM~QLD2kt+-3_9qx1#7wZxR&KutP_uF+DEsPQFkU|j+g83 zq7CYWXn@}MHFdA-x z;WDcyxm)26FQPH)fRq*Qd8a`!gwhZ5ZZ+D+c1sJtm&l##HFLc=L96qnWB_GJN4WjL z$FeMeWbcKK5tqHb-3R2l|LFf#DJmn0=EO(RkTRu zcg5V3p$tkJutC*WyQ;u>u#HQtXw@hg zbABg0_`NDb&S=tS;QYNVvi(;SHM4@opyIl3cf1FI)2?)|Vf#1oH~wdzP5^pF)!vjd zR~3@0t|G4viT0}s7*yCWl+v3Hye;T;YKr>)Lb{^H;P}3GrKFZs&qBPN{FUx51H;Z+ zyCMGYA9>%-l>2S8Y)!emHnmfb_sW-ku4OxYSN#1t(}+EZqy0+@Byg@Ztr#h0aYsb9 zPD*oPuWSN4T`7^)Pg&<>BAfHYST~?ldO$uY%F~zIpJ}r}j-Z9`b4Q=xnLAeEjJ^4- z$ym-G1m~!BM&t%Kui}wR)xQi(YQK$@np|)aK|T~4UICg0=67lUUof+n7e33GprXhi zYa=uAs~^B@78pn^5Mx9xGqwbv*j{>g{NC@<6hQEsx4!W+|r5W_h_i zx2g2;^ub0+=x;WiTg?xi++6;tyGi->oQM*1Kj{2w4j4p*0|vJ^J=s*GmLx>uGTY8P zg097awCu7nFC9dbLQ&XT3Q&EhAel$>tgR59@#U#z6Uf^JAGoh_Y%eYtVFMijHz}6C z0-_|R6Q_4+c#aNV46mU=d(FSR9q$5>3Do(1P=AiHSS4jpNO&KzyU+wwN1=Oz*&*BYlZWW$J#_OySM&1H z@t)Q3(%RALT4b;tBKy7DiVKxw^&9n(@@9NQKw*ne0XHNPVc(PFWG!`+6g~A=7kd(o z)Z3@%Ue(C%ap^%kXx?sIi@9Gv6)6%zugr#IX}O;?GsYU%qRu1$Jw=&0EQjix_k^VX zMBfU=ys+WQp_`&QA;uN=bZm&w(6r@zt{{Q6zaYbvvftwXF(|lBe8IUeD7pSCjx@H} z=cu@u#)#t=CEjtzBUEk#IPD3%clJ1D2+Kw%kLanZmOv8M6JrMR-dUJVUogj2GAEl3 zLwj73w!M808(psjdNxezDj!@PkjrZWiX1Kxs`tv;hWv6_TOE9WlUTN0HTXof`TBQ< zW4B|v8+b~7M}5b;O7(>OqQZByYvhxk_C8v(8n>- z+%DWDkip>s^52oFd)^1A&fHiS3|x%TnxVohF~>7lqBohj%WUn&)M7KHNxZ zo>c5T`GKDckF}6CMo4ty9HhCjuz*ZQy3w8Og+~BT6oPwXR+Z?nq}{j>u(+6mD3mt$ zg(LeS73k9AF+Xa2(ihUsIUWQMYfMjkPgJ>ON$$R~)?wRX{I}r`W!N)ZI}2uXkM02R z4+VNo0ScaE#r-mc4S-Li?q1amd;Uv{<|z}kXUm_|lLJ-8gr2=?)nM-uTuUlC0!e$I zx@y^oS+`lz z8S0axJ?!xw_K1pY+RR+rs(Wp&hxO$j{$6bz;6p`V0}D?Mrf#1asJ3tMuo0l)8b-ft z>87W6<){>Vzyp>nt~nhe$!f&Rf@4T|9={)TsnlX7JTwh+v>jtqRcl~7vl?7!5`^Fx zG=Gt&7Yd7pkL1EhKqNU3FzURQ{i|E841fDxFATcWpEtHu`DExp&+1@!c>;e8@Xtdy z_vOfRq6hrvQlNxzWsWVPRFITVn!bKcg4H0Al!z2r37p)Y0C8X<%oE5IX-pv41XR&^ zPckhjwfL!!3Qn^(>R%uHxD%Y)T_N}AmTNbI*I^Gjle6|?8p`U@Ykk7WWJC&i9%8XXgMx|z$8rKvTUri!Y z`X`g5aSw%?pLQm!Lw~XFP|4j;e*5zA*osX!Epy&{!SW+A zmLkK^qS|?Z9C!>ABd{1Ow^%=$vwZza_yE}8BSqq*JSx@-R~zqkAToBRgzz9US6f#6 z{Tw=0x6QuDRYzM#?a`j8S$5TyOI98J+_V&-j+5#>$E3 zxTmyGV{<{L3{f8b;H`cJ3+t$~Kf=hO8N_fCfDjp_3?0~DqRH}m(eTXk;$nJNZb8zW zQAG_I=8geP_d?BEl3y#-P1|1^6&znPolN$S!!JB>5pKY9#Mz1%y;U#dlc1E1E|1ox zuy8b*^xm0&?-}At`4J(Y_<%Gs9Ve-+`w9VXsY=%{RZf1XD`A-Mhg{j1p|!9K5Hi{| zF7&V+5DOK(LE6XACl_V~z!wV-Wa|?slW&1es+_at`^$5@n_v6f?~xxp^!gFTeUuxz z71y-e<91XTO79Cf-VSX(4+Xl2POM?#q z=_4M-Z3Fuf4L={;Em0mA%JGDe)?XT#J|Nwdba!v~InU7Iqrba6IUOP(@=4TrlborgWwzI-ef+AHkpOoAK@Ri}Kw#+JjmX0t_ zddxh2o0~=J6n9+#Q9NKq&L`(G+&4t(bI>;6~<|(c`*uQZ+lP!IOYZzIR ze6O&v>FRLHQKiCOW)dNM~T-J#o!OI-v{uZM)Q@hY}f&hH0xbzY#j2h*#vqsLub4)%N+L7!^ZXC z%*2#O=LfNayhLr`a@thtV(7!%*6S4d&_>IS8>KxApsM2CMfeLM=!QW~NeW^#D^T}e zqVF#G&%{qZvVRREN44D7->?Z?PAXD4O?|t5;{?=gy~xFyF2YwIw4U;m;YwMQnN^F4Lc{IPvJS%Z5(2C zzTU-QzaFc#wuM~8zzbFPUbB^i;VOISxNl>>GNhUOmIb2sfp1vyA9@F|jqz`TgqRel zxxHfulfuhIjWu|?Eu)8$)QTG&;<(6VhPpy!Euz21a>9IEZlW9y>hAla?>o)#RX=zd zwg$3Xh9pXo^+7w~yFmfXB+P#GD|L4$-N`=f&t0~L{z$sR6 z!}9<%)swLHE}ooy4Yhx^QFVlC!F6d*`XVMAQ#|BZ^O``ekad0>4q3Hq*Lb5_tU5! z@XRfThZd9Qj~^GDe}D)@ZQ$hTGt~g-zM86c-Ph_6VUt(&ml5!!EnWVS46$~@onJd3 z756Wr%jv{2IZHHe1bMgS88y&1E(-=a0$xzQh3VS~I%d8SC&1CDUb8c1OthSEh*07m zm7oaGsxC>VENPFcFTgPpZFys&Jr17IbB!S+Jbd-S_yT-0?{_A?5`0|iJu?{8@5BdR z$~U0-+ygZs{6cCyK7`?Ql0i)Z;3SvdDT;vKc!U&$~}q@@dhwa9y~t z@)@~8n&g^LMUDa)r7hpVjsz8glU%{Bq{L{X*%3`LMwiha%V4bejl}hMCW^`657r0! zOR5f2y?V`&L2cATaJsE;K?jH`nf*M6^=it2`XOqL>MLo2} zD`E?Ix|lGSo<83|Io6keLS@lu1wq9`h!xcuksE*MM0tyV0!9Mj*4?go)iJzk*7%1? zZd=Ayt5#wNNaSg@(N8-J+fpZ~0#XwL!c!?3BJxK~wZbKZT3VGd$@e+QBgytS+@p*S z>qYu#VZ%1wy7SU}eRGYzazm-Bx1VFusLKr`8A7wo^7;G=xE#%~C3mXn9Vtj|FH1LZ zWIwa|sUb|6zI`0LI_iozBo;502U%f|&fay&wmp1OiUm#XMclb_$vr%MlyBCCq31p% z8`R(-;h4_TD7Cdb&a4j2F`@MI+jFox=|gX0&lOe@2i?7g)-4TcBr&_EE3Z(giEn`D zm7+*dn9q7+^x)QNk9I#josBy>LhdHdcEEenTdSfpFY|Hl25wHEIr{CB+z`Eg_3pNH z!yaJs_k(=*B(xB!&Xq14>Sh}5A33-s1C)^OSKvq^WH$0Aa?=j&x`a$bQ-Erf2BgUL zX*CKZ+c$811KN`&>Sx=(^2+@2u+Yh9RrIRtJHiCJ?cnkGCP`c#f2qUJ_cSkKb0-M_ z=$@ah1}aa*hxo-a7-KBpZEzxIXW-YoWyuuH_k_I4YLPgyi~tTl=$(^Jd7}cq1Pwlf zS4#LtE7TWLGi@JDe|v1GQ;hI|cKF2>u(2-h@W0@df2o@h(37!9J3zs3g@ic3?wcB~ zQP)zVRIoZ3@q2d?%&icPVSH(6Y7%o3Kt%c7{@(7V~Uny>e6x6yJL`*o}gBx31PMN5A zwpVo9`7?EH;W599$=Ai=jf3Yuxhf5LB`zqeo23MtUP@cZb6eY`rM9EYb6T*rTq^jYm* zVZ%|>!#%xF)!1SfX(S^*hTf%TWHm}zYIPc|kkfDk5WK)~OmLM?^urre(O`;d7)qt& z3V(dQ9ba8_0cQeSMM7m7+Muq&>Rnl2s3UzpB#>!eAdQ_!=I%N>pBLjT|ZY6M*J%KoraK) z0D(q90pdMyzCrk#5N}Mc8S5;*&YUWI!7#*N81R8iy_S(uV((4FdLF11&B{s!bmd-I zd8<7hZQwZAIHoz)u%)X2KZVM>yV4Wgi3!)XXk*z@{=uae)gW#}s@T+P4ye>OavU#K z;2+($OCrw+yF5iT$ZX?ciMT8-0ybG%5?m*(5yeYaDSmF|x;jiP31g*w`8+%BJ}~02 zM>0Xx-Z5U)oQ2}?`GSxxzTyk?%x_e$`zm`Kuy4khWRP*i{3^9KZ??KfmENa!( z?U5>U?Lbz#9E)7>j;t`qaB}lmF3PKn#xw@ky_R8fnbwp*A@g3O&5eT=Iey8Fj$boz zcy1b6CK@P!&m*|qQZU9hcux)p$=S^VU?f7@U{lTrli7I2*TB7UgFj=$4r&UWD8AQI zbUVgTL9B&48g9ucqw4*Ph?gO@QofBKJbQSg6ix5&nrsGm7Q}0P?~ z`+}ugbMLX~^k{$uH$mR#v8QehDqInMKV>t$MFr;>BbmFO1f;6dUcqa}16mk(9Ri8v z8@S!Irb9n7Xu-$Zm~1so_A=6$9R!?jW@h90(xBvMlz`_u3zm_WZA^?$k`7H%~ z-V)5eXAbjj-jLF8H=zZHCc&My2)spc!I{oPLJ7lo6 zhiTIy36hewtl90l%OpG%8w8O}%?&!_Jy#(J`xEC`P?>83@#r>+kT$U(LW)sv9cJDc z;pI$+GQvhkS+A;-v9)80ifvNY6Hztivxsu8t)5XQY4i{eFiAvW4jv z7;aXLT+>dui6MvPkQhlOml{I|2lg-ERj{wYe{uBaaMELjK<66}I?o>O5S)p|s_H-P7jQ-ZjFbNg9|u;^<4-xe&<^V%kh#htw*ev6ZIy|WFQq+ z`0rxp#2^xwi2S+ylO3T4ZjsgRk2<7TU_D=66N6VWze@W~M`xDGNVAcYP+Gy+rEWRp zyrF+o#+OiTpTl9|>V)As)02Ps_(`a@5|*Uw`tci1N2;z;j5M=yDBvsVtEuL?Q-ISU#uSzgOC2*NI_wn7;o~W+>@xC zNdR3leiu@qEq>Bi9p>rTKNze9SVAV-yP$7&bF*#g!V`)SB~30*bl`{*JK-=`eU+oS zrwWR{i+e?{TdBQQX&__zhT7L1Du^7!Y;4tS8tRx?LwqBf@CZ7R@26f;oC?0Z_f(DU z!?hF-mAbE*n}o$ZN;&7ve|hD25Q>o3QDv5KO7YSOT8@`)pb|{)*lIa0kJ^u>={}q` zxz2Bho;@)wF3;e4yhZ(AOh=a#F$$0pMe<0xsSe^P zZ<`vYK+!X*+Zh=ssDJnLWTB{d0XxeyZKD_3vV!@&h6QpWmCL5Q-gsU!G1hUvqJ`l| z?=L)!6tb7EpIdDI>Ra6p`hC&!6=^%5VS?d;O6#a71_vsjZ$6ueen^lLwTA63xFtmg z7?z&yGk=6h=9(|U8`5+kvRR}NWoh#60uAJ@@S+1vRo?7Kd!OrNlK}uMl)!{KBZCe(Cth!=(}Vz_Anrf%icuY#JKr_LrLU( z+m6r1NDvRFHrxuc=O~axDzXt*!6kZX@~vqqW@Wg4nb~14fnP_AR>WKa?^qV9XbASC z^op%djd<6yidhR^+I5=a7 zOTZyVRARw9SpD4Vi4Iz3`Qc{?m0H_wm@RImxh~+=wFHlwxf+hMy9$r*kJ0`gv7Q9+e}YgoitsPU@8$otpB^C+~p?dAvbL4c4LX=4si#K^E!e6 z1eH+-o#X;5fN2Z?0-F?3lZ~C}IEgGekXb4KnLV~?kILxK5BlX-0sv?>+iZ87fWThm z$_}H;8*rtJ3YibQ16mM5f3}>-2Ee7-SUTgwOwUEM|Lp(znW0h!eSjvsO`wNA z6mT%(9)Al&Cljt~2!gKr?;Q8QI*Q2<;A&G11l@s=&FrA3Mlf-a0kfHMmK zgby^)hvw34t+&*xF1B@Fy4{ngrKF{>u+wIQ@9v|EMmvGJ) zX22LB04xABBm$tvHIspU|Hm)`832I;Db{x|8lVA0Fa~ln;lJL_{a@WZV>H1KSWj_B zHtUR=1kF#3l8bo?^vgOjZjo#R+VO98a7D2L{s6QZ7W>Zq=R}2Mpp9ZoQ~>ZOoxxoK z-q$kp(EA1Bob8NZ2*6`95C?Ex|H;x__9$jMCMKCj=nK(GdlBuy`x9pfg4AyK<<8vYXrrCBDW&60rR>hnj5!c|X*)^>5yybt?FCk6k`WYP zCv*Yi9FU#cXF`9TVYD@a54`Jt4*b#>tOWuC_6NEe+l|Bbpc9tKMbq$dbjoqt6f?Lb z^O|SifYjYXFr8!g*&!I4z#xTyrT*9X(fnZU!#`jDSl}NE{9}QCEbxy7{;|M67Wlu~ z0>*`4n3uqJJNzz@xTd9*DG$NXW@2Xj5O8W%aj{Z%P)B;56P(&zT%37hQmn(Rpx|(o$npJoweImR7Jo93Iih*R@U|5^?(aq0+mt5ZbY( z)?o{g*xKpawN57zE%fz|*NDWVL|G;#Dy%n-yM}`(FPTL|!5R4?8G3Yp z;8B-m_WxrTXTP!kEz6LZaoZ>{aWI;Q(c#6NSPe3-S2V!x90gHk_V0pGbWFpkCGtEH z#}VzIWn=z&in#}PaRoz9|NXaM3d7+@_TfnAp3brJe+R!n?}%lk{K4xq-=yk&M6n5N$Rv4zvu64g-cfafb5_ zXrbU1GX*U-XTXd*!weVObufvw{CxC@=aBNnnPVjv#*G2SAd-zST=0L;_0~aAzwaCG zM+79LLlBVeE=lR`?(SH+L+P%iBt^Qr)1bR!38fpQ1w_uX-`~ucIe(nlfngj_+4mE7 zT-WP{YYiW-m*5Qz;G(+O*~Y?`LL&?^u2nZnvnrZD$&_ll`vRWjdRvZjt3Cx>16DLX zWC>hj>%itA`go{H6rBVV#ekR9y)@<^50p}W|6We@fub=0-@4EZeTOYQy6*Ci^XiH; zsdD=Hcxg)#jloN)9pBnkN~1=4$)-|4EA)(^RiU!UQ0iLAVa+CoX${y3sxB>i?W>)y zJ$L6sdK=yB_%b`M2A9&yzX+SuRQH3Y9kc^@L_nv&&$Ox_mDu_B1j#{&eCJd>pDuPyZOeVe*d$6m5Tl zwJq}W_YzK0g)6$j5*-2CTva|}qu;k;HrD5RtKV!#LaSTk;-w2@sh1fVgU<|-QoUh7d^7(+$&KN1CXA6DvM zm1VLMfLK}M*zM4@psP*LdojukR{Rx!PBv5$`Fio8tn)O2^?cLmbL38{Ol0ZR3m&uc z>(p7&Mw^Zq(y-RGN4Kkc&=+Za2SLm}W9!+PZokU@VbjWNEq%EPwKI=bY_VjX^Dhr1 zs=lu0gCm3cZUg#!0QHvu%Ovo8fa?@6PlfNCK+#+jYKGy#5BUgGt*^iOQqOs!4<0vl zyx3m&Ykm&4eJe2I!5c7OkKVQd&Gx);ue`F)6rmLN`qI%M-#u6}?*NA~bQrlN&JV26 zA7D`%($s`BTPODYswVxqyqclpSc_gj{aW>E!xZ2}x>G=(KfiXX@<~OZiPG-Yp>TBysHf`*7lA^&_vzn* z5Bsvs{nVJ}k6&!>&VGS~?{Irt3f~XC;JC=Sf{Scb=?Q9(;7uTS_X>`G1-A=W>^lIH zRc;<+c*Ddn->oV~_YdgEdH2eHtJifMzz=*tJodggzyCZ@z$gTy=8EnUDvo=Ao7oKj zir}YH5+H?3X1%Bz;yi|jXnN6hrd;|DBb4I={>5q0a(cq?KJ5%p$an-pxATrmf6;P%s zOG}zu+TJxS03`K*P}ZW$=Lw!JhfSa`Qo>^ckMPY<{9}V#ewxkWBJG@~kNqdW7IK!H za8kV%`3dhJzQ6T(?*rO}d}FDFb1*1cKls{2*HOOHh50vdMtaBn+r)amjOki&QQeUQ z3K*g1^vP!3&v4?GnBn^J)sJppeYe5Ge}ebCE`PDGnR_}qzB^Vr?Om5t99#kIwZS+~ zQx%F9W}-%L75pgeBU;DoYV#I4$mpTm_$Ata*6W-M9MkSZmsz&?E8A*I?^g{CZO}^} zzJcD@Qe3a|tE*rS<2xM*Lx~)_#iNpqLZ+5B3lIR@wH{scPwT9cRbsXidbj!Srt8A` z;CD*7N*+ z;%%S`r?0kthr}BbDXz12S#d!=D(hqAZnseybS_MH8sGg&*Q!2*WI9^{-4JiSF3=i5 z_Ak_6REJw4z*qpMx?UJ5JD1gAYA?)tdg&Sx`HuQ+lTh5=sLb@r9kFmP*)=ap#cne&uErZ^=bXi?=Z4tz0 z8~0dd-QgG7(Ota--0kf1?j2jMkFoW7#NLH&Q;k#`%h!)3D1)TfUwwP1{^#}|RNydq z@=BlJB(B@TitCDKLCIdK{mXU@i4M9SY(BGi#Xm$P$6Ua&aDR{tX`cvUZv~iRQPz|{ zsXnB!dkuL`z*c$c?0o8+I}yMwl<%B91c22IIlh{vn~UFY zt<_YbGCkvQD`|tHwKXg()#c#+Y#6WX6~)s?Gjrjw#Tv(hH~1*WUcqc~ny1YeW{y=2_dGE6#~&tu~vJ4>aaPZiWa0B{{jF+ww){P8<)-=Oe{3i>GX&8q`@Tz>DW>sn_9sJDg56IB1`w9BdU9&H`x>wv7SPXjfBI~-dNLKgT9t6KI5u)xQwY$vUo zHr>YFe`UWx?$_x6B$Lb%htL}ylQ7%*frtaTK>M{uB?jXGSO)rtmF7g*KLlQ=R71+Y@|#WhS_{m|3_zz=|vHUw^`Pf0m4y!;3~ z9>6$xTWp}j!q6N=K*rREQym2ny{>x67jN0wbnp@;wMZO@OB4%(LbL+r!ys))cT6 znpUOS{w4jL@~_5%<)illi3WuKnC;N!FWf4|e|6IYIXm5*2w1i1g8gy%$=-9Nk?HS0 zlLKsjzltSr0qx5@33@_jP{Vk!bth!QzXu^K+q;jq%tVN?bVmQvLSGmA;Ndn z>7$@RyZ&vicowVq+3j05U`d4&=GQX<}`EDHauB!vqnICOZ#CobBtH`-`R-3P& zi)}USZmWeObD5m3rlVD+M?xK^@ucz3np(IHCFBKpxe|TQ^@1)vugVitI-nnrj)FjB zk@Qr*wxB!G8gu@8&3ZY0seYq+GQVM!NG4Bv;nhQVN!M4A3BHr-+i2=UJ2}!3*bljX zKVQ5nAG3U)n{`$@3fw)O!l=z0Ou4mj$bA5shP;D!&Cy4Bw4wdwbQY4y`0N%X?OZ)| zxe|ieck|V>rv37Y!tRtwu`NR9fniM=MrH5U9B_{m>&JgwIkNIR#G0a?S~3pBujGG4{Uhpq>u$_s@g)@9_n@(Bt!br5ZBSJl3z@ zRX`wn*0D?8borSwY>qY?x-P5)(L)*CAQea}T%1xens>BhU_1lT7o^gbl$fR2lj_Qq ztoD3w&9g{-9x$APxr?VWZ4k2we|g(_{g-^BpUXh0dL_|1hhpbC>w}%f;fq)`+Ioa@ zz&u$a>pO%*+@iqpI%HQ8S<47HNtX#&|BT`f!#F_7+YWew+J+CFL{6C_|8umye|o6N zC0Ri-@jXRgpH1KKXF3}+Y1pV(S-y=1Cup9rmIs?inoHfZGVvOYuTsZ&NyrQ z4Y0ak@^Dba)|~I+Y3M2LS*OW@lCYrYoz1`Thu1Y&iwp4Z*Sy(*oPAb(6Q=i6CB8*Zv=b!Io9KJ#yU$;8X5h&c6EIJ%^nJ>^>cMT>1k{PT(fG z>4D7JvFK60-Uu9~1D!Z8UTYp~Q+$jAXZF76Z3mnnn~5j9z<#wMeD4xbv5f?kS&$Q16t^W`+~y z0e;D)1M!10|1oTBa;8Ch*0^_x!jHbPwKccWLAI`CHSo6!MG~5$WD!4GjQYM2p?{Rm zJJt`269Bn_Y{fEtaU zU$}^DbF12Y^749`*(X$TtfQSTPNJPhT=?U^RPRX{|5HQLu+!sHUZu0P41%u7*S4Wl0#=aOkp zH??t)u9;n)G*&*Mn|>^(ok6Qhkz{r9&d$#8mp^;)?vHo^3$gOcoRWxn1xGVla8G^N86P(!SmDk{siQ8<=?MMW7cs30Fs-RgVj>I(o#2H&@u)KJ~$UVnj89V7&p z?lDbxfF5tn?9bp3Z6E=bq<7mDNHN=LdT4z>s_R`6_^|rpY-8g;mDu0{NrMv0+^(er zq5YjEtmfl3<-lEL(+Z*q7fTcE_xEM*VM}}dE9Z5SwU6f?v@)w)*={$l)taWi)oz25 zfw-g=QTEx>Uoaa)?rzMxTd@e*cPZ32C+xk;2RltVU0E+hldrjv?>PJ+Uvbf}-Q2or zntgCKIHCK=+q&h?kh2xbKb!X~{}fs|6pSadek8&&G&U>Agq*sj8u{B2PmR{6_T4>> zzhpqP7hrA+{59(b_vMzhH$!YAB#R;*(p@)BE%jq%g0kpU9#$VjD{~gqsJq%L>ycX5 z66~E1a2b|8?c2t#-e^51sTIuGYasmvr&G_ris-`nyYcJM#H~E$%*Jq&WF?Tm7(mFf zpZ@)L3Yf~+%~JQ330pDZxhCUnk_E|7V$xsDx-P56lZ9*tyY}ojPu~}bIp$6Jckb%C zL5?jY%gKC`6Y2|m;z;#$J)@cHBo1OxR~*xg72ig6+k*gNChbrCC?M~w_ugprn)t}& zz>52y`g(-WTdPFp!hS59jSEi|4bY%c^1MNmc++oZw}LF>>80&c##B*DlUFyJKDOr7 zQ8~un;qIkT4E3Zpg-#jwOLAwuqTzW%Ns0OlA;yo78^vneZ2WdEI8*18+d)xq|3J<^ zKW+f#__*}&e*aegAva70bK@r9uk5F&XS_~MR+s+O@5}OUiaoa^ydR^OW$L2cQ{8V z%tts9l772yZ6|a6i%DA<5WN8}wQiNNz6#l1a^p{lSm6g17aZOL>! z-i@^bgb;+&_7YRieq07BtCuirqhjF(a-2rzn*hnGkANdASq>KhFX3ciKg?QcA+cyMS9QDNd)xAVOLVYUMZ3rd5(OQ2j<{ z3LoO)X_t;i>DcdORuUZdV@w9Yx$^t6?vuUFRm8#SKv$tLqSmq4vQC{zLI*GN4PCTx zv|F^Bhb2ci*eY|}{q54x4Z!H*a>9vSif!5YL5IR(4QnTyIO&zl-h0W-jAT7VCY_h# z_9@huu|f`h0~YBSUwGmKv7Jovc^wz84(Cvp@WVrt7QDP(z!Tj+BTT$b$AlL${tNt% zN0m5B&phfk;WIu!1&BJtb^9}MSUp~#~dbvMEpP*;2 zK*8%mJxhZYqUA1qw5m9WT{VDRGJ~bi!dkklt!q4RX{Dpvh@*~}p@jP<3aiRvB`x2A z#?9J_+RjrCg})!#4K8L@>F!O5#svBa-F1ksKu<5@IBC_GqbYoDyf(M%;5IZ zp48jxtn{wU9j?Buk{O*YXvKazyB6jt6CeKqN&eDiru7Y$Ja(p|;75@FZ?`QpwH%2; zO-7n78k);HTLxNM=+#F)XE=t5j=QdX3++*&&_82&l(t+7*Rf5`Tk$(np;6Wdee5o% zclUrixr)Y^az(Ry$cl6XkK%6nI@mQuakpcO_a>V4T81#!QgBx7(c(7-qsU!y3Gc%S z?f0x7c8f%TRV7UF%3;uT#(i-bu{!wZI=E0Ca17)My}M>+{0x68(Z$}%k*19&yk+=; z*um<|#Hj?4xAVnrB#a@cfwbz`BKO|k@nd&4n_Irz?SC^4ip}H)3 zeeG2$KSe(3=n@+}oW(VnIKsfd;r;tuaeLfGorncnDsH#Tmr?JWYpnYjN%eXKn-_pm z8(qKL)n*1AGvg3$A(}qX4_u4F8lJeS|K&a$lK=)h3TL#~;jVk3kAtt=k>lZ7TKtr>ebT=fq?6^ofLb$To|7FF> zA$0VBo-v9R$1I4Z?1}uZg^cu-sLO`%45$CwCTQ^jY9~oghO%t&6AXn2ijiUz?nwnjGBHO=y3~QbbBve5*rmE8Ua8!+amwD)WJ5 z;nOLzxA1>-?oAf+Y7x|Pdi_V{|6mYe4vzkh1g$m(eK0J}8+G+4;<40LZe6qpVl1Rd zRL+=8=zvXhy>GzHAWyzT9R%1084{Sb3pLAjNt<;|n z>6$ej=Rt88CRUajBregeQo<$2J)o;Hs6ia((Cct7@wLPA!YMBkE>0Vcc~O*B$o5nG zyeJZ_89f9llU)2}8AiYrGpo?a{3#OSlaLQYJ0K!sWb{L2<0mE66jP&c1_Wn?RIW8A zM=P%oFAY52VpdZ7VKEX1^L`B|m-kBBWs%l^D&{npo`cJ+2dhRKkLs+WwI7J4euN{E>@qL`E>VdSrC^lUOHck9eY77Vx{$}${j^vWLhEX zne3(}JJmQ#n;r^-O?orA0v@?-6Uvox^mbip6SC;d7KYOa2#>H{Ll}j(ow?zYNy# z(Nd@V@NaBOS#l2y&Qb?nJg|NsG9%dAka7e{wYTsvQHAMQc!8}~if_-Deo zYP{^@n8o?Fks|{llm2pa5aT>SX=B(gwc4~T>LBUVMbKw)MDfi=$9)yQ@H6b?PmPu{*eOn&eV2q zJ5?C}94VZ5h>;7mU-v+gnUQ%ax+R`r)7GUg-YqWw`uMpKO__7q z`m~xq&hbP@L;k{~o~}|N<#_WrtIqH|lutqJ-Dvx;$WuAtU5$k`F+INO0)pLu%kIE{n^!Y8vQ|w z)%yOXIpL)I5d5>)c6?mOlCwG^Vhh>TLkw+JZp=`9(9Oxp&I1ExHA{npXTThGP zo+30`yI4NJadawM@tS3C7&}{z!%z$Miuc;O)6eJ9{xzs{T!GyaFCZufs`OtGqw-W4-)++oh z+TPj3f^4qT4&ANvX7x}jjwt_~{#hCE5wyF-+Dv4OTR9&#w`?|-EcqhyPO9l7KvIe& zU?v{prVF7|4ow=rVtHFJ)b;h#>%v0kntmp|z}=!uw!*ba5+a3TKc^hwuz(tR<+(RQ z0?9^P>3tqtr&UxauTL4`U;U|(RbdJJ=8V~BQ7|9aY{IQx|F!`S;-vF5lVQp+Kgz;A z_D6YS40Cc`&e%s?Q00pdoEc-2?9p*@t{+%|A~? zGJ6~vEsxX@havT0C7u;Wnsgqsm)2HAK3cM1lHAH7vVMXb!a^gfokVDV)t#Kj6648o z^zj!~RO%5gGoBAG4pb?)R`2xYW0ejk=wCAgI6I@Oh?%u!RS7+Z{Mzfs=NNF+5tIwQ zzr=lFHn)69px#7+v#xhkt2OgqheW1j;&0ArFa407dKk6Ai;?P|&Z=I6+ZpfPTat~i zDd4Wg#E~J%=`7EWww5}5COt?tTSIBKt8~l`D9&Z)v2sw*DpBp289kb^nq5&~$VzV; z?%cC!z&JmFwK%@gK)(+ulA0c#Q<92$Ens8X0jnnFjc;R*yGwnlIsCl^-*U(q@bnFx zqUggNjNfoux_*HXnyOp9GxBcnV7Ny-7XETcU>Cp z_3l_EyJ4%6&ItWjcp$9VUZMn=r^jetT=z}W2HK^JFp6FlOq04}N6n8-ua8srlTtlWC#x z7F_&X;S?}sS**X+X81WMOt@%5Og3(tXcdEOBgRgVlYf;&wELHXZwQCYxeUdkUqLus z;;N@vj62h~S+`${r%fCb)jCsihZVKU1;OS`!bT|=mxDm!iCeT$VbstFQ(IK>u+3ivoNEN4=$#Hrhlb*Q)aPx9$;%Hm@40bn9$-+k{f%jTJs*BCcCp z_^omUMHS(DqwXO70dqI>Dg5H|myA3n0I9Kz;VmIOVHbQf6r5+yYl)TKTO79Mv46UR z-?14krNjoBUjDMy+B>5j(fw7 zHbSiMA!e~q+7f{T?iuvUnc+-iqu4xf4p;1~O4GH^_9}($;GnaoX4l)OaB>ou=3#n% zrefA=9B0W9@~IV^U8@YcWB3|gXMI}v5mic__?w8GrhG|l0+Y7Bvzo%3?ANWp!CFH4 zQ^vXn`TF3`b?sBtY%N)-gHpGt5jW7SUp6(;_08j$08Xe@oF2feEt$gyjwcI zNehXRVC@@XN~O812kI}h)tsSDvu?E9*1?h7y2~mEfddN~rs$Ba(N~$5{3J(2=i@__ zI^-O$YEEun>im3v(US&!v%({9w6=Bg1f`>r+4y^R$*Z&v)#*P}6>X_dcMmoBvZ{U9 zecJuAhzEu9hO@t&bcdtkdz>T)G(ly>=ou?*|J(ldd&6i>Etu;|=Klqtq|`UoZmSal ziA6zL@*n10Ymd?~ajyW|dQYFXz>;qY3z?CPAg$-i@yor=dRC)$HLt@QXKJ0l zYWt4`Gt00KDGa=x1LOh#G5zPotHniVN|F08;#e~=wI*(Wk;pN1bIrz12EHv)4GkI@ zk6Yx=E?o3GhL8d#9)Hap&?VFSi`Zz;=1`%8&Q&qwWpzQEH5wykrEBO!=!G0Fp|LZ* zluj(I{TPiEb(lkq)Wtm~AMsr%(!s|B+mjWVdGNfGNiF`Llf94)|Hp#&DMdSfQc?dp z1^xqLt-a^39kQG#U!YV>GuQ|n5iyw-;TN2#CBCJp8LT-}`?xvYP~?;T33LLR9SFTN zz(Bdl9Qv^S|gK|}0vGnvQRbwaIvjvVQ8>8mgyc5J3s z=r<6BeX>Sq%LRRWF7Yb$H|f$2e}SL|@$wX6>PV_6b*|4f?`qk9kDUPrjL; z+RHL7E$Z@jQx1z5XVRf5_#G*d^LVDGMg!$~N~M$TGrihJ2g7K|kp(Y2;WF%Vb9%$7}WQHh6!IdxeE-N%gg__)EQ zlzWIXd(^Y!37@=)fugFdh(Bk(M{VtC)+6_vUaFoCvSw%YxI_QWUdOZG_UPRibZ;9z zo3tZ@Wg(g-CV4?&g?W*eA}@7>&`(3T8ijLO`P2N6lWjpxQgwY?3cZ)3GP=YtD6BBO z$ab9szVe7%FP~ss9>dBD?i;#QXJ>tnfa*7m*6oOU=1#cF46DkgOxk-@ydy5X(M8u2 z-wi35JzKN5ZT*XLE!!EK&ma4qZl6F_{9Hm+uiKY1M0AwF1gdp2yXFb~{MHRq6Ycli zY>8rzarswo&OZGD8$T!=fns{d_`>Hl3_oX_5kg=1TyaFXwryG9W& zadjT1!o$jK?4W<*BV>`pmub)511Tmvih-Z4pxYDxghs4-^*>*b!I{t3)Fg>}KCIf$ z6pvdokRt}DLQEK>V<5xwD4Vx#Ij0))3rT_B90_WEACnZaRmr(-7%L(P9Az-22WqHE z_^RPHd!RGLkNhI*>2%Q#-~?_8P&RU1|2Uscn(iyXO>OTojD{v(latQKFU;oAtD0kY-^4FgGNFH>hK2ff}gp|+! zuDJ$Yy6&oJu>F=35Wv~4`;VAO_;8KjIf2`bOdTPphqnK!)UE=hhR$6OJ=i{- zM7{g_ITbXiLi2}>Rw-XZ)IL0UiIvhRm8c3mF|oB|eRzrM1>~FQn+%yM67mVnrazXI zzoPw{kq4D9Xe{r?nD>7H<>v9%d((dX({&|s2QCv`o@bVFE}2u`l>|9i5omh-6Y~Rs zM(FDF2}`pO8g$;$!?`afj4|abj{g zLm)nXZlxkb@i4)aXDY9AOM0a}VPR)2m>Ihu9e1-}m~q^)K{E17`2af!MCxOLE1Cw|MYf)6jYI1!{LBEK^I!f2fZtrf`)gA;=O zBb_P1;FZN|@De8ZxdD&*&argDiQ6t4ez2f}ItRHfan_S`l+r#t244k5{DHcZb^>=Y5W$kj zH^e*L>oxV7cf&h|{?Nk;Cfsb->I4QIuJy|J>hPO1OKOX1Ge;{4lu85N*`Zk}GKAg3 zkq4-%Dq`jM+nY_)b5{$kVig;9&UQ{- z*vALHg=y7Q=tN8_4g|mN;=xXra9@tG^OySd8$wOb%=7rOkl21Po!n`kBIXvzk`>pX z@_2c*|0Y$^>}yjNOa%|1-^=^0I{H^7X*EI;xS~b^nN#mZhj~u_cpE*!BA`1|nlyv1UHd#zPq^#ySCU5o z!x)8hu^|kh_*11h%Wo`8Yijyq``5f$gz_AsGrt88bJdC(6Jd>tK2Wo?OiK-8?9+ie z9l@7Oi_(vC2!X~-gd-3l&G-hT@^6&-T}^SkqvjuH$=k{Ww!9P*98sK}^(b0hi)=#n zi;9?+q1L(M(&Zc`j-r(jB!cVH<~-QbDZ<~PIw@ZE*v(K`!aS=eie2ym*_dr6vUzPM zUsqoA^)$bEP{4B02QoHuM{fa~TeQw;V#H|;RXVMwvixAhYT?^7t*aD0Spog+c*Iw` zILWB#7-8RX(}{0b@0l;<>7sb<=L6GHUt7=$AYNebOeJvSpz*TLrfI{ggjIWa0J|+&S0wi?&+aP9_-#1Jw^7p9 z%$yhpw1!V#R>C0XlppJHKS1{i3UWj}Ypq!i%(5(KQYQo1B;%P*55O#u3SKHjRRO10 zZDS(qtI@A3Lp;*F%PcipY&h-7UtQ%Cq)2Q3Sh0L~V%gh?)s1e5o7Gq|c3Y=bDyJf= z{5`U>`hRr|@YxdDBGjdaXR#)?hDIjy{2AE@bzu2J1aTJ|y#P(Wo}OQrC9O=fTU=$Y zN%Xl~u>J{S$fjtm=lr%2j+c)JvxI-=7S4Y^(J5Zou%a3gyG7vz=-eE9wh?$P3`T%y z3ar10@-K`ios*T?%{}t^##SM~fT)Lz;=0>l{$XPHIVa?OC3`KtCE53y^OP*8hRr4N zZ;U?&g(q6&i(~)IMrxQO3+v^DDoxTD;VYUw{v1eAP0OtrX^yZ+kT9RvPW1IWuf7#vPmvb;jZtUkc`iw1!Bl zS5&0>p92UfGi-z?CVHaqF1eX1Mn=&X>9g~LfYSW@TL^3Ko7XxPOqCTSG487DxDhrp z3}X1j;o(awm85FlrR5Pw@O8EBJmUI`$|5aG6>;;T?@Xb=erW>7hupD zQkSsD|Hl~AcFV~lNxD(qly+mu*HQJ3|B7Pl>bHmRyaSeqtGJ~<>8!P3Y zbcjU!3zA80gIB|TUD0yUZm28+JKaI7RZdvo24cHXiI?3Qv(R-aDi^Z}+dDv`I6S!s zOKLZHZ`$EbYk$00lS(N4`Chy!`Nf;gSp(v;`8%n&Wkj~?sEHM)B_%T3H$^5H1(j!79-bCG zY6kk^`&7>*{|#&T5O4(OA1UC)e0{^bJo(!J55N8)5|dPELVc_MVjrC3x9vaCV&hn# z>Io8(UrY$jvw1Yi^;}|TCrGkNC09b{JP`@L5+>9wo?t8w?|ZLo3p5z;3COAWpov@E zbsShWS^TFJ>vxFrO3yjOgNHCjq4Z^wV)-LVV{^#Kq*G+2n22&qvQTk zov`TF70gx!$+7>T)k~ltHaG;nseHgG{7>1@25R3^b}~S3MYxdh1>+oLmc{E-57(QB zL;SEMS5Ka+e&ZF>+7|}Y0%rHYq!zi!vPKAY%vBK8>SrzYkJvlvHvbsOY)R=a*;-p9p9@cYx5(^fBEIbQg1}72= zVf+(qoc(Zx&)C}`ATZvE#kO>nNd14${cRjspnkCGW${)_+Sc#S%IBxz?lTzy_7SvrevJCiBpn^eOc`<_m##5-#^HDP*jTzKZ663UY|U0Y3J4|5lt-cR zwxi8l3#96Zm44rh-Qwu|Mw$`|8jzfk8!Ere5G0$(SUI!ZPlERmPy0J>Q(Klqb+SESoLA5M0=jt&>#FtJ z)u8!~15M3oznYa-)2T0&Ho!ubDCeVZY?QDP>9Uz18L}BLtFJKs{=}!9l>7ZM8!F|X z6OSPEdADCfPP4Uwrh22$P>@80q?DP5Q&(*>O7J(`O46|w`D8XPI(>!;hI#KX*Bsld zwg+T07OA_Y{|fEq!eizP+Nyk@OI4VlshTjRQCo@|a*<1yC@rdG;q_r}gtDMxCS+kP z@uh$4_)+W*wwz(QO0V}<^=9Y%HRXbI#-0p3523yo#uqy6;u$*a0-yJaqDfQ!##dv! z#*Tofj4KL${QB7;&0FZRe~9fN9c9ccTV;paKRzK2l;rDBxhV(8dY>32r>9M97DA%N zI>BCAV7|E3LP0jGpraIt!yxSlp&ldN{e(XSZTc~yf*JI5vTD9SX+joZ^$vxX!dREY z-Cb*++7_m6?_o0*GeBgo?thfx#bYiMp>E!P3!mJz#aM$iL6(NM2F&Sn!p^8Z9`Ld6kB+ zExnXqhBRYxLaeTV0s)b3<*Zu)b96osgR_=(GzxtGv=AqG7JJ<3SEg64@M%Uy?bzqq zt8kthFfN^q0#Dt(0#bl4A;N_oM0`P*-7#8-W0D}B$Jst44f<>$FhYkTG6d-!J++0@ zyTUJBHC4v5ObU}JWa-cwE z-ekcc7k1wg2h&oD`<*~b-&!$P$Pb@07DvRsNX}~)bO9WVokm|({cub_n^I+BX7 zm$YElsxUVThPsk;Kppb2{{3`@l5FdAD^qz*-a1ksxUeH+R2JX8Zw7Jqrb(k!Oj{n> zEl?&|8Xd`9s9Eq(MSh5^2>H-XEnlcchOD`==X4s0&()0Vv_!z&A#S4h#5iy!8cz3Kf z`j_rM$s$b5CtX9s-atRbkGCNPLTI=r zOoV?#FPErYc4E~VOEXGT>U`gQb+*MYwgwgW*nqf@w=ysSn#7DQw+Qa7)8m+=S}m|e zdBu>BnTTtuieI$wZ)xS8x{|53!J ziWkyqG%=LS^y!jw-H{R2Q?eh!)#&<%;%-BwoVe0SN2Mn zvI!r4RF_58RePwEjbF}A<*g!9wqI$!hV@1EnMNF zs1kV$i;N9T51ABk5QM{ zA~)wat-{(&IqLMaeTe<65g7srUC1C>=NHAnSgLd^1Ggp%6ggk+sW4%dqH(&8O+jXF z8{$)rx5Z#_9)Z~MT;OPt1x6WzbuKO|fLnqXH%O$skNm>DN%w!dyeE67>OpLuGMF61AfMI>N%&519i zwllnucmjU!BVcM;L>PBRA5Qs^Uqe%6EBn0liyM&Q#WL4*yWd|OR$6~N4U*e9_Ya!?w?6J>FAO5^o}^`8OEs z&ti#?NVFp($ZIq`%dk_AGbne8@1YIB)|Ft+%dK`?miE?E` zInzY1Hv5kkP`w?RS;l^&8!i*_-@B#kP0DBU*+*}}vtm;`=)C{UxIP0dqp#8`i?0Z> zSp#<{c&pl_I`nzYDAPI-@vP!LqLu&U36B@MoIRmHDgKJUH;SgRM>|!?D-GKVJGLTm z$uteV>-M|!9cldRu%@Zzv?MzPWXSxQZ#Bt0 z-{%(&5Q}+~Q?Y1FYShR}`4628{LMw#Nqbx-)+}2;eDm?fmBuQ+0&Sd&M}opN%P<-j z4m)>S?J#DRjK^iq9(C;<$nLX}`q@l-*Vy>dNy)G98+MkOEK$8@CLF&2OPv>JI(%Gb z)~(&S+`GY^CO&4yUF1yN%M}1+JCg<0vAgfzF?OBfNMnk^vblJ69L{n5arTe)yotb$ zL}_lFs~Zn%kcvJZrs`k5$KJ*Xk453!c!pvPGJjzLi|!uKkRc5xCX#aX0lgUAro3Y@d{hByz}tQNBj@+DQ*u@_ z_LXk)@5q{HDK#<7LNzICxCoucnvXezT;$uip!aJZ-sM?EzxkWKwc51r-wg0l0YjFl z-!4J87@I~NqK4De1x5%Ei)MfI2D8-v{szs1joV%yz#y2_%)ol>zFP?NoG1$(lMen& zwsyjD(gZ0yoEh@5* z9V&@=R<4=4)0b!yNUWu0Y;@K6TzrV@A2Z3Fhkm^{gG&6R{(*e<4OK#(ZnIJ13SAg( z3X9|YfkYnj9ep^*T$9Q2(Wl1UWe6pP?lqU}U0mYM=BqsMH=xHC{hH@^P^2A2Ao(di z-Pma8FOkp_M^FA`znSaUSQuMz!Z6#oq#zCz?VntwT52`JE+A@McnfcWL6vJ3Oi>pW zS*?1nXS-_LuBG29!~CY60D~aIllBN$(&zbkk^$mZ?a05|G?3AZ`W2guV0}1yRa79>nLKNB(T5{e2|5``FWn{J zch-7TaDBmthid-@_uEsZv3FK{V>?%|{>)vFq@RY1o2uwt@m)FI(eqiQwGcf!9iM|V z?6KFEDdPYdqPn-Y$B+Uo(=j?;7OHTUl()X+M1z`Fol>Hsif zHcM$0Ebw|U*s+tDXrn~KWo5s&lPSjiXOe=&vH|TX7f3wGID0~7tHOe8tSbu`OEDxE-Ex*=;@;DnfGt5w>L?dKuBeHz??ePCF_7+f8wq3Wdlys+bcS{Qr z(%qe#5Re8bK^p1qPH8D=klJ)e3R2P~Ehr%IU;BC9_nh;eG0qs@02xx7z3+QnE9aWi z9?g_6&p0Kq=u0ssdR0p3$&&wu@Xo~) z7B0QH@{qzW$lhh}lua2N)u`6Ru6_?JU-y%7g^^tVA`u)U7cW=`w>mRw{rFa@ zl(rjt7>h!L`&wPKraW`dBr`9O**vg1=-%SJ=txB z7v<=T+BUyKqCb*yMz~i*Cevvs7vzW`WM)AabTeN#x(UzGyA8Al3-@Sp0njBYX5HYX_?ob5~w%BAtsmVnz7+9s9L}H1Y_(D(VGcM1yMaj;SwCueRlb zuyWWVDnowB^r8&ux*AB@5nVmiDZr4_5q7zQPt*SVne2_%SR%x8hA2m)XN;-Q&w90=#caVG?%Kt;RL7GafL-S_KaM-gd_&9Gi64D&q-D#Wj;pVeKl}j8d|768yksWTOCc(F>qozD1 zDsNDvXJh+;;`*6b*zK!g;j|uVN^;Rt-6cH%v@{?SqqoX1t0?w^EDnc_ie(#3J<_T& z2~~Ab^M)mT`}qcaIhQq%o@p8TrD5;nH$dirPYlN@YZTzP$N6fz5KHR^+;o{?Y%0bc z1KMBP*gyC&-+AvY?&u?3U}kiidg@!}QNWoRUWJ`4UYj z_ZE}X3YgpXqi&~NOVU@5hly^=Lce`7`3tsqJdtg!G;-sFwIL|4Sy|iLCY_N4G_kSQ zq@KPY)y(KuF-r~%mxZtLXV2W3seBE$iM%V{dH5!0gfMOgGf!hrgf3tv)Jle-kzrYY zMaHq$MEZMQ2&wJz&M~d#XNG)KzFtD*^eC)hxM&kDv8w__8cOOqod(VF#X5qbN_Q4% zg_^Q8M6`juZ9F7aj_;3V&R?mNJbYFRn=2{3Wq!pb=Q1%UC$9z-8@^eI@{I9C1%?@ow|e!q;)R%NZ$48TF2UptPx#(iO>!hV=Sa7C{@vFDs0 z;;<6BGkPY!Cp4LFJV`didj1ImDz-M92pm%i^6~1>VLKvPS7%awXGvKmSd^+YYjHB^ zTGbc{S^EOiOIT@a-X0V?<7^r?CmcWtE&D)|saf=^I9{MTOzB>4i$h0_Xmsd=9rDuJ zVQC`rLvShqWeR>Yx+BHTb8&Q)SX8E>{vdwp!~JtN#{Qtk8!gm`ICy8XB)Jg$5Pnkj z)Zyxx#zc?Gh=^cT^(8l11Y%yV?>hdbCVOL>l(-P~V$sQZhJ1-akLgq#%LmY`%Oiq} z#}e4&x^O2W?$Hb)SajfQ)rVqiKbyqh`CX0gBIPr+=3o6u$vlFm7&P}Nb+{#GeoMvt zIpd06wqfll2@;vkBE+MLOmJCJGG+BXCd-kBkHdd%SD7xX8G>U+KYrp1-QpLhl(riq z6xA(?vJAr8VOY*IZ7&Q8qR0ws=q4zRA8$7O4LT#Gn(#^=j}0>=3Z^udPwtG}{bly3 zZkdCM;j}v2<43E%uLjpi?4?t^GOrzBrmIiLpgon8VYQH=&C*CT1DYG9Hy^*9JBh*{^X) zxg3b5)c5KKOV6xT+Ypo$!0l3hR+RKkuU(Z!vj6HI_L^n8?3NOd(HK){IZ_ev5OxKXDU! z*#41b!)i$ejYlPr9y`Are!8$M!dCrC$q0icAO5DHRW6i7sS;10*KuJKV85D;cF$qM z^>wgUfU{*EnfI0f+A8NQ%@Im+N_$$Fwd`BDR^=+)QNFp|>^Golbmt3#AB zcR2jWtPTOZ0ak5|7P@Fe&6lVFr73BwcdAkNA>|!ad%^fKd><8O1>~&T=C=K*9BJpiYzS8ITm+IY%o27X^AZ`CH(=Ud@ z($4YQ)K?7NN9kgmZ+NxlUbMFVmHegN6MzbS5A;sBvG*MPiAp}^?J!$JQG5PvjH;B{ zVxA@WwB{i)A;m>Rfs-q{f>Yqm@4p0pBk{^!Z@4Eioov?7oET(-I-BM5ij}f@TnNvK6NojzF4H$+EbHq)ZyIH zhxiap9TB91%g+VW8q#s#Wy(ScGU+N6J6g?4OLMBlHJ3Br&(rO{3fjcC9*A>=#M42y z>mrI%Nbav>O$Kh%N1o`^>6E1;0$j_nyvDkOlWQqn!#AU?MA}*2x;!M|(e?tn3?^aqJho9u_e4AP2g#V47{ zJZ_6&=JQjIX>o*$%$7sqOixcrleTfa!n7u0Cw-+bso&|iSE-onS^ZU{v@SOR&B;u{ zo!jzntSgiUdGY*X$_i96Dguhwl7c&6jfXRnr_FOcsp9l|J*|>=Q-ssi_)u0`=ud~= zNiC^QOy|dk05ge`yYdl+gB-qTQOuK6ch4mJ4%*o0E`qB~Ww^?u>P#bwq=c+^jJ;=U zVnqWQfj115lzMHt!M8hKaW|lM>=HV2c$yxDm`Hk4VpX3RLU7n#b1aZ%>5IM;KIcxT zq8W610d4IGy2w|nL}NxRqPry3CbYv8Vv34;7~l8>Gy4t6uV^B}%)T?tLXJ)T>!D}~ z9WWT=y#x=x=6UK9-b~1L9UIg;8wUN#4L!E)Q%b=ta8r$_Uq@?yo%-8JI>zg448l*B zK=|~FO*RS1%!{v)K3XJJ8FdwKk+d@MYa3=b3Hp%81|_a`jd?;`ye(SHRD0FXwAi%B zgi&ceiY79@a(2!9hFF{#55b8D-&A}f#pYk)1avCHg@pUAQr9^A+P{{%wwYH0t)%6K z=wp?WlwLI_tGB22XwK02<_|;-jpOLmr1ep^JqO&&vCN_B+MM;xfki!`YsIs`HUlXByRisNmGo3ea; zIg#1!8N4f8kIahi=V#oKOtG)aCph<@eMI@53ZAM$VMy91(*{Mp6SmSZs2SfZRJv!& zt}_+KPxs)c3`U8P!+ldqSmXI=n|)_YOj&K}1}49x;c>c{dN1|$Zbu}IFj#3@*x06i z1a6KlOxPG_0>N-T%e0{E=aILt_A=FFT>8_ke+kWoQwPBhxvbx=uytRercS^{( zeT%`<^aWQCahNZB!sIlz&!otqOarIy`Dz8PKmoQmdNRw1L|cs^A(viNlg%_%P^Koz zGB=e15hJfKT8sJHM%#BiU93tmty#rZbOFtU6=;>TZZb@X^z$p7pXc^*cXU&RkTe*l zMiV6}XunV*(^XTvuIXuRocBxl3ekfvMEgNHWA(zNAey1kgD$yW>Eln{Ohh+nGg<|v zz2)(u+@P8;G770qm$FT=&XPw#wx=QLMIzKG9-O2#FI-K|G0p9C7+k8Q-Rebr-neCf8Ms%?LqU z9b*aJt2#&WT@<`4I#TNxFIvJbT0fa25a`LplO5gz1`mK(!5&RuU@ECt- z{?qn1Cl#ZCwi1VuU?XP2P87rJ+e~FC*tZ0IVGA*1mE^9!CIrn~2M9-x0M*R(fm(o~ zbH(?SlRNBz%ReKz;>ufGKD8}TLEBX+j^7-`FZm%%Dp?+edq1A)SZ-E!)E64m-{a8W znDfE`u-C6{QTKuBBsPLw#wHb?5xvXLAH(WK76j^tag|th3`}>Kr^VWT>ABsli{x_D znSR1yj@@qe6&OQWb5Q4QC@CQvP7#yyc#j_h(M+&{H_Eg(bxKrFh)-8mNs~~-n*{2Y zzmVdJl2SDFBy>~@nmG#?>a)lgVqOk^Xvf2hy#o!-?>-X?vg&5s`LS2C+|wG|QV0%Q zU{?*AwlQj|B7&Ce-E?e8vM3csrU6L*;$r5*z4brV4{t@absENwYwzk#rO z5SzRcI7wmp@b*F)S{0Uh6Ku-DM{vH$XTzH&0U7a7WlwKzPF8>qnxjiTrge%gv}#V! zc1TfGeA^d_O3volJliHDE$ywdn_@Xaz&=s`_k*P+LLn~JE_y7Yiz($g`E*4%n*_O$ zDIv8jlWQ-ecJ{FTeat~96GPef_8{wErF5^3(29Ikp1{*EggKneXU}bHohE(qXuJb8 zHaZRF5TF{b_m~T<8lX5$_MH<^G}N1wO%J*Yjee7!^`Cw~H!R)=MLB62QjWT3YJDG$ zy)F5isG-A^x{a1VR)u~uJkq2mIDE;TIinUa5GIi|{Az(gc)+Sbl)kY;M`UKoa~T82 z2=tu2-fF#gY>vb_l8mY`>F@-&$d9dk4zS2HpuhIb#fld{bQ!}YMR_eVWvjh$Qe@0l zk3}SYyW(beCYMJmBPE_-N1{+KMCWDm`kK8HVR3Vt0;SO#id9!)6@|@;(ISZWJd?s~ zTvQw0J_|g9i&5%uCQ(`GpX_)E#;ffb9!WvSO^`Bil}=@tW`l^&TP66&w==bRF+%$E zEAr8;b$aUhCDkzHb%4t#=!S@bg`?0JIs5FFU}cZwhbJ~ar?ojQ!VZ_now|;}^P(*J z?nG)4A=!!Cta!(UbD9MW&3bL*sT+RH8U&si{?&EO?1o{wW}D*u=c8|2x}O2MpF%RD z?Rb+6#wo{I+m25*JDZ>FX6M2dQPXG{v`R<~il5$ZfY!JnAW=?X(P~91yFZT)SXue` zjAl^4I?*!16&;vs){J;iWsbx#j-{JElbUqrJ5TKpG#*MyDLV3TY@Rhn04oay_%fQo zQmeNYhiNo}oXsPHAB>41ohp#d5Hq6aP~+_gTFxV@yltDup3pVUgz-Y&ys>M;4)&3# zGJ|^jsdI%i4?IagZ22tg7%+&6dvP>u$HpLer zO1(|ijDKg8nR-j|daQjI5#E3M%)%;8NM8>#&WR7P@>?gnAkom$=8!5vsRh@ePY#(9 zZv@uOp{`tc7E@Z!1^wCa*(5oN6T!WGbzHM0`bLrR$hXos{+R5c+u+g^cCnwfpY`NE zO`DRYHJ6}%KU3tL+FFq(O@7f3UoFeE?@;}jeN|B?Q|yZ;o~niJuh-{fy2(c(Bb|yC zmBFlZE1aAH*zv&7AvJlbC0!}yIeTz$tYSe)pDFg?WEC>-Hc^WD@7(B6$rt{9!L*lG;VYKR=(mJF7k5Itf*Ghj%egw}VqiXeelhp%1;tUESj4*)EbSdW zqv4UtbH$5h^!XfoB)N5Ez2?kqdhOdF45n{{Nct7b&^hYLq^ZbOLF_+IzR$Rw`0g+( zLi=U@43WfaTB*q5U6Ade2xf6Azv54$oL*(~u3gJycoKhX#oMx@RalE3*Hql06He&$ zLS%aRgPjwhD_&@J!l8p|iS7$3CF%)^fm{g=4-z)}mou0i^1QvMO>b_!bDUDe5l0=V zypj?w;Ri?C?O#-XDo}rgC&2a;Vdx^#*D5SNDUE7sNV2Q)a?!L2TVONva9eSv(f<2Z z!K;fB<)3mGOV`2M+dO2h1XIlhb76+gT-k2CNFHR-c0w+)2{tPJ$9=SFI!k)B<#&iL zE#7u9;;OAzAMY44suD- zl^>-UarDE)W5#ije<*S&Q@6RL7vECbW8ojDx3w10e7Dd&z<`P}zXIG$;Op+w@L!}Rus8HDb zt!l3v-raHj$hjLeY^k&!gWnG;*jZh1t9W z#O*q@1sM>J2Ftf%nk3m=(Oz#*@--`OsV$mw{)7%?Rb@VI#fkAAq}-9q3tlJv^K~z6Yd2@KY=3!PF?Q_f9gSTe#D2T>Bd*L4(oK{|JauwbA3AMd@UaefBm|GzIu z;%tv?{`7vQ~F=e;(zo=5?|P{{_|@8 z`&a)bj|4xQKJb4%amoM5BR#ok5c=me{r8UrQzp{Tg=pu_*0?toy!2b*zem#wkc zw^OuKg#n2ibQ2~q3fN|?-n+m1ibQ6;3TWB_0GfQXACTks4t4dO!Jfh*58x2OlZqL_ zhIP%mnLtYmlJo{>Y!BkqZG3Nh zO$YU^OGf8xeM0sUEL5+G9F!#EM8Nb=VQK_m!R7%>Qu(lvm;1nLoTvvlqkLZLC=#Eh z^)7&&ZE2&3A;c0rlW=APCTPkaV9#6%xR=dtFwB6&X`@p-0$Jj^LNX(d-|l!ONFlM@ zV({PK0!n2cdtvbONyNG9xYbq;7A@oNfi8l}5_pcSZ41xA3xfSa{qKu%ja!R69}X}8 zbVz;|BVb>t3r5ic$hjRz?8>RG@|S{=0ZEwpM39&nydcjfSJUaJXpI%oZEscC`t_$cVU)6^0=Q% z5`Xs?4o@S%=-j#y4rAY!i8vQ94q}y~css{4MDD&B$pq<7W>m@IwH^g*r3YxbEy$0b zeW0#@gVnb-L&%jD*{xWQIL`3|$ucYiOJpC|goe$P5+Y_r@u|}3u zzG(XR1jbtnZdgHEi#v5DAJ>!7J0N9i)kK;v;7Gdc^XjeO&woz{52Zh_7GxOG33h7V z;mKZ)L4LV`1$Xzo$u7Wif;mY#%68wrRwhr{32f+mPZw(gMq7vWx#P=t-z7ODCz_!AR%LXsd@`AynOplEv z(M*H^hjrH54ipawoxQL5k6=4502r$xo|f~;G5Pi~*m+WY0Ji?sG&}(C4GwmddS zk3)~wR3v9$8SD`-S^;k2O`tDvqxN_pe!u$TewFL}c{J>ofRux69(M35B9NQ-w9p1C z++slzJE-`ycVF~>e8v9BvSwSl&5re*5`I*xt_puX`wI)6X$@xMg$u` zcf_>KZ3j@>hx3__0J_lBpRxzhTwfx@gl@nOUA$T_Zwaw_4`f%;0J$?v3&C}W0tPOI z(dq$*u?AcSY|~?LqTM7E$UJBNxc%1h8)jv{m%j?f%;j6X2UfTCmI~+_0Gicfz&kvD zPjS~C0?;1>&z-&@fh*l>8jrJO&ILl+KS8aI$s)9y{pHzFf{Ib=;Zxt$DH4Sqv;1Q_wi9b>0BK zXVU3i?-dwuxDIfyhV32)pbaUoW9Iiyri1lrAhlsos@$e7B!Q5%Af8&i+4*Ui#KG ze=1{_)J`u=s#2 ze&;e@sk68k@YngVhLQ@+nvn@7npJ1`FraeY(T?lk(t#`s59w3O3^*HOZVmCT2E5cg zkh`ez@!=tCCuYE3aQpf~L+I;LAf+*Kiuihle+-{;w#$6sr>OZ`Vjaoqe>kds7SRQ@ z^%nfC72n~#_U%&#twfXF&l$j0B~QtRtyJ)uE$HVcOi>5i_pjN4r!RioVC6+sR6${c z;#czGQ_Y%HFH8xjlX+>S%uUD%cEs?^J7`Voah%ASDzHjY=Y6r#v9^@W-GnE$h!*HP z&ZY3`Ve(mu#Y7&(zl_wiU)hGG=r<=`SR(UhCs*E(Y1MdKT_U~3;H-HT#x_7aN!mf24Xi)qGrYd|ZbVnjPRWVu)3wEQo zKjbxfIFd$ySKf*j-pzpNTexr>TCz6Y11|x%)CKo%0NVk!qMR%d#t=riE0AuO?=Dvo z(_7gro%t}*XftsOa8e)eXWNKXjY|solPz(>im2}tczNlST6e&aWvpw}_~l|Yg%e4m zXxoFTAv3VTYIk`DBpbYABp|tTg3H?qDY%;;c`13ZznWAp`%K@pj4AJRvE-E%J57uj zpnaE4hR@l>RLR3(MqX(cz?CDnu)dLWVNqZl0nS!rf`ikJcMk-)#666~MsUw-- zoQiG^N2g09;SyXWu@2|8&P?B)WQsBY)0$R2N;kSHLi7!UAIKy-6^hP#B@DpII+&SFn$LB!3;2E&`N=**@#Zs`%||Q&QP)jrcdTwn zb{Kw39dAaonJ}C$2o`PumB&eE+Y+3xe4+54Tr6de5X{);XT!BV3iy4P>oAEY;D!8w z9}O>(L%IvBg!xXugzb>|y&EzR1t=k>>%0a`?8zQ6dh=ofQu<1yb*FBD8w$?z8?(l=hMKXy8 z`++`3$sqOzhE@HyEqD4Hz^abVs7##y2so+!M8=u(c z9R!!NP+bA6HRKKVbo(dMk1E-+@Ha^=B>gNEQ57w|SN3@Nb1Nk9eA4;`OS@Lt@_fWo z0?XD|Q?V|n+_tgk2`$o-R=jd7jz62wlVns{eR?<;T}Ve7(!D+Pndh4+UMPg;)} z#yGq0+>B?TGQKB(Z}Qc&eDidPLUKYGZGpbsSmgIN6a5s65cQWl!Rzh56Foh)Vax<_ zL$ND6vbP15t%(tU4OjQtf<%=aXE8iACsKe1pQb9GNpCocA;I_oD*CO~>}#of3ETZ0 z3)<%*%7n=7nFMB1G(o#>dLB5WP?TR1wm4q^ODRfAEUBdbhJhif$TAyIb(L8GbDv?7 z2CZ+DRMj3hi!!aPg2Bzf3E+&H0cACPirN_qKmd$Fo!7>@3uRs@OHGBqk8Hy}xKLCv^kTFXhWRdxAl%5Bpa~7=K?B#!zHQgkKQdEUB zn5?i#%%tj1YBTOAaM_JJXR-|+wt?(6Up(KAvVqQIsOtik%!PXG1xQ>Iw1in!0-JfilWAlgebUF*@80dP zALPBMAELyU(|h{u8av>+AF#2+J$E&$B?4hQ-G)KG=<9RMXqWe&yfAJIPD!9q4dyUv zD-j)AQ0MmMMg7M{{@vYb=+ZCh`fst5D|rfSy?M;NLioL9G1rUZ|NHxEz|gJnd{M~G z$?M@gW>7XjxnjD|Y25+X?^OzqF?~Pj)rPui|GB0 zPqa!n2GkLkH7JEY<%@=VR}PY0VeauP$FuJIV6OC;fA0G{sVdu{1|IfPrmMVlPDVY4 zio_D-8|MH*YjKzHQ2xFhdQvZHRc=D~q?_Gev~89nH0#&VDr`X=8?f-MWHFA0eDgTMq()-q zP$sOWQkty`VGf;A=sdAJZq#1_&dX=!%~Q%B4UW8$naQVbUDcP5kyNYy6nJUlrFbfk zppZot2=Y~MMgvj=h7}JYbk9hB))m9mU8{MjX28uzs6-S5EP`u;FQWcIL2upy*Q2^a_T?y z(}V%sZvIPukxFSS8t1MnB3=7ZIy=TsCNY9P5A0k>Gj$u1w~MUrT@?4Qp~_w-XVjoO zK|M?GwI6jq5QF9mK_ikdGwve6LP=RP2huZFb;@IN{&u^Bg|Aha{Z)%P&*Dl`H&%?K zq0(L_qXjt`kC`!S=9J~#+CbZn4&%LSVq_t#*$}79sEg=o_I_PWA`ePY>YjUtC(kfr z>kTVB0V9}4bB~eAz&SYJWlX286n}iDOyLn}A?cuw*JM`GMmGqx)r>!m^A`9h<-So~ z@Xi_@tz~yMS$TA7iZJP53hxiRlgE->84s0ka}qCNyL`a=b&<>%t*DBjUQ_8H{JfN7 ztBU{yI_Y2qvRfs*#iU*%NvsVIPTe8TrHgRu3|)nNWJY&tJZ0XZk!2~m=eamCRSj9s zQR`Xi$#^$ws`^N z%WC*N&v#ePSLBSem-NR+Aky+OmGqYg6VqlIY02xQGU~?L%xJ?@@3(wJ5_)70RD`Mw zm=JY&pl5qO($64R*Y}+6*B48p*2JdRonGbz|s+m{ammO zS}Xu10ZYop@D&6Qb_H{#r_M6}sf@Vucn-~ovcC)#N=C?LSi$6aZn4Jc5yoA3GnH$b z-YmQX{s|ydK6P+#mjL$l(GbcOnGrtfbB;?}z}q(&%bC5Di-U#(A$~<{A|WkmUGnUm z41xVww$GrE!cSRO2s>1sF-?v{pCFUwT>P(9M=eK#OOY!Yhq zJSSQ|mXIf*2PiScbX5<=XsCro{z3U1V5wR%H|<4dUu4$TaGc!u#a9tjE%D z89wr{=c%|`T(NU91b=d8$Nby%$vjhE@BaM|sF3$}5KnP+zkZ{;v?5qOI6xDN4m8|~ z5N(rZ|Ao8*Mf9C5U^Oa18@e5~43%dUafiRrO!Tu)K9&E-CS<^toY(wnPG9f)GG@FB z`$l|%ibr6ueUBV{_3+C51s*4ny|G)R>vW=2XzLiF?s92^37UdZy)C_`dQPh;cQp&C zqqPpsh+AY3LE59Sp3h|I8XpOpG6BgvJT^#3JZT1Z#{jVwt~AU-{%a+qpnN%9e{A$- zzB~T$aD26379aXdiLwoZ!vKJN1kBQ_6UT)X(khv}JFt#QEr3gg%(C;9#^~QYQQ}^f zf90fXR-9tNZ@|OaICUn#h}jKeHheAqH&b{ap1yDiCM)9A{nw;%R(3IjznPm_gYDN0 z{UCP_xi#YsE;0MmnkKb{rVf#%?qxj~e>dHs=0EiP$J$j_BrK07?M$=nwUi5oFZ z2cRlqwkwHSMTKrSb)FB3|7{eq8 zAo>op2^D=WdH~W*fyx!CdvL&fYpZ*#swRBEFS}1MO}bm)s{7kgK9C-!p@A2PpCGmC z!ZMIE_csLDRa!e9XM`H@P`cGFnnt7bkBgbx2*br8DDrn+DJB73CtB36BP9Bb=!ngvuW?>RhceUlsT@E)o&BfAqeGHLPwr>DXb-=vni?xL%m$gpxRO!O|Jn8>}gDy>J}v029E>H>U-ku>o9{iqQs{^I(?pxJgw7+b&I8>d0LI)#aHv}J553mC#2RO@Va2GTWOD zm|Oy5eKR%DJE7eSOBWV;1dcb>@RT7dpY>X^{KgKyHmtCkcKF>~z(~a|-(P^`IJ7;9 zq^FvKM%vI!HI$JEbUdJ61bHr)K@dB)?5#ACYIfO>I_OO>fne8QSn&!gRG&XzCRW@w~7K==uW~e!`&z29qS4aF-cr;POS%AXaVJ%~i3~$1{ zi_17E3VH+ysFTvX0v>@pXLUbUC9pc}%x&(}osqoh1KPuC()6=^@6UvsdOr*!dh&Xm zth@obE1+$k&to|V3Ic;|l2Xpm3;VrvEpjC(acWO`^W!zOYGPbAy=wiSjLm(prHVs0 z=>vMbGAd5gKcs%Fkilysog~RnO17qR7==47u<171E(Bqb_}s@*YBIO3a$(Ehv<+}f zdlteyf~Zq4Sh^2hw0^Dq zVOS_Y`l9e@m^ycw43}Q$qh~9cpK6d#9$Oufp-G4HvB9pox!*rS0~q5c2+^>X@gNCT zf6wRVhf{Q&IM@>12~Qd^GvDZ9^nh3zvOVwpeEtk!i)f;cB#STv?&~#ZX+*#00F4e1 zIVpdHU80<_o3{RV|2OkHGfVP8I3+(@GyWaulPmADlWK^Pad`5&ZbHQ$w!}>V{Nm4T zUn-1Q?@wJQZFB_j;dnBA$|BHiq(b%w+Cb`Pl&=&Sl`~wvTc_Ux1+(jiKiz6aAQS`~ zH!W-q0lHlRq!J(RN=Y$aUMd!r23lsJgjTIqgU_?&E|zzu`QAFRu1B00UgA@s$JhCm z{?*`)UeMowbO}+zgLOVE93iX%wuzYtHH6wtB1&2$>em9iO!<%CdI^qUwB6^%@C#L0 zdS>4NwL5yAw&Yc|&_wB%^;eLt1=m)GKfE7L`XXU|cE?o}@e(1zg-!Jiy z{NnH_CF_3{#*ncx5p@g&jk9X!3AahRs&P6#|$Xo&)2E$_vIcp&F-QDu4*6V!9?Qm z4gg`K5<%U0DD1uqYp!rqogl|+CEYZe3#s0%y*rJ(1}z_f;G9&eO2yPz3J|Nv23~s6 zsq3fFCQ_q)I?G@V;hxVr$8Mv?@3f8VxaO69_O13ghEB*asSFL9F^x3u>G;@_Ole`= zxWa+3b~4yBcMB2Lp8~=?wEvw~A;hF}MaTN|HwcmAnctC_T=m}>a^vjBeGfmMQ@#gd zvLFU(D1#L_S*qfES)sn0QA8WB!JXhQ=){ zdcmMeiy&<;=w%J5swvny{QXGX44fwzOu(7ZP^r*70xk4xQJ)LI*4b&^8ys*0^vlg2 zugt*52L{^r!0=QuxGkvPSzZv$+}Aj_$q4m_%}jrph)R4if{4eq`T4Db(~)kuc5TEp z^Sn>|56|NKJ1w1Ro$}8PfJ!U#+wP)-zV0c)_@DaF;bV&T&(?cRs{iTNpGV?YVl-l4 z^Em0oAj&f)EaGgLME^TW%CRJ$&_1eEwTE2swmki07lwfpKWEJ6^k(YBkmQ$<_Zzum z5tg~tTx2Dlcv#mCG+-}DxiP@K65~Pzg20R?=vRMPyzNIeT2!(HU^9+IZdE^Lh5T zRml0%iSB+ncJ-4JQik|gO2b!*zt*m5Q;pCp*$!{9%%c}P1-t*L7HO6&^en5=H~3v9 z1eiX`Md^i-{CO2|-YV1WXO85u!0h92IYp|O_2*MS&59<8k;a186Qj=NjvEfx;PM|s zI8OzN_I=aB_D#T%fKkE^(uckN6R-pR1vF*4i{p2V?_1yg*%VX$*i3h?a(jDob{38u ze?gD4ocUk|nzXQC2~5kJ=qfObg59(e#&yBfc?NH|GJNb!BEzWx6Xpdy<%cn|hnf1} zbFec3w5xgUuEE!u{{rvn4dlqwa)i!d%eqA&03ug~I~T;=)J>oszM!_?HTYW$FGs7} zp-=cK8bNCOZH}mRrQQnqL@AbMp>u`d&cQxibmicO*DX3*)*bOR356K{1UNASnLYB{ z4iHEBTJC^UknkC*1xSW=|3TTwd z^ByDuR(La@Ak6{V>#K;vCX%(?6;?-JV#Hn$ zm*Lt-yB~-xsXL4r8MNz`^@lt`0EAk7buO~2q8+rBkOmQt#ehv4ZAmJ7rMIrcPH!ie zH$LJGWXbq%AocvU$mC+n=;7q&Y(cdn`e(3~7x5qOI!zP)-?sKXq|?vf4*WrLjgPr3 zPNtA{m{9q%*Et)DfG8Eg4NNu5a@gtR@!!0@PHeH(~CW?S0HMVvgS^MyjA|pLqDb-@=>44zu6DAINi~ zFx!X{D6lTB`o3vCh!R79-Ci(-9QqIyJU`6P(hcgGq7aBcbUQVl-%tM#OP~^l_mwTV zI8MY~&nM=k>3lhd_(9-S)HU#E&+TUDBd0-3s)ZZPziC+TMbealO)W9({~hk~{L|~@ zp(Ob4N}6y)%hIlCrBj z5g_{4{VVr$oeIUe@iW(vsd3}Jjr&`+1iUEVw|xMRVb_nX<2&J|wOP(z;CG>AVB6xZ z2Lan*r4=`_9)s+~^?FFAZ1Qf9_$^)iU|IB{fhAlrSL-@_tw_JRV0g!yQtL&{F0A7( z=8R{axf1Th2vlvWKM^g}L=NaV`&`>n@_+dLcl4K9Bre0~_An1!?~nAZ6sg zS*fj)$2tIq{+lETkmTsaip7Y0z6QRAUnq7QY(wK@q4qcoOyl~G4a-I+v zPKG%bB!~cNx*|n@F$Tuu&uFTK$O=0U_G4DKsmkQox}H$`fng)+EI{I53>NuU@8X2+ zf*wK}XsTZ$iVbjmC>qembp}%jbSNZ;-dr2;f~k8$e?Kw$H!I$FeaU&{LJaU!0yxcv!(nM~%1EjJYA)O7^DEsA-pSKVjhtG4+ISPwWhO{D@$ z(|jy1@pe|#^60)8F9F^n)B-dY2x2=*Id9@~(W6kS>|m248Ft^%uCyjxDp&K*r{I7A zV8e-wJk;)FLQymF3JeHxxQ$cncI2tPGkgP8Gawr}aLC- zbqb;^W%BwEo*?nHP1wyQ@~RGtg8E7ogilZ`yaUM-AnE3~@EBz&)3-SzOJmcm_6BJ> z=n>4zLWtS>xh72Tj)H?Ss{y~#Gmmo;1Qjdxa48 zKP}lz{RK>n%vI^?5w=IRI^`Ho6AwbMadCO4-#{o#gu#?lgMo_@z?izh5E~|0j4|E_ z^1#7%)_wYdLKv`P_iKJUru)Y-#n)DGIAm}qv=S$pspM=N5{*{yYu&zAhi`v=L-nv5 zcDaof8W0_}vWjp5^cv&?@>LPZr%@=QY_5B74*FRd8OSdmt}S=RW^~z8WNiI?Q2G4; zB!!+P<{EoVL&ry_LMKg;gKXZpmM{7?SU7K!;jlh^zH=fItXx69@L%%?nu3YCvMeSC zH5Hd1Kz>pFTnACoi5}~zD)o;<qQ+vY{yGw^A6O7qMZQu1H(_9$`cfezV>}*++;rw!W7O= zSqSn>45>q6X?df*`}AlkTlTWQg&AADOSsmP7RF?r|Y`<8HYdN^_ zc^~3l`8NGR^vdY0l^%X84<#bY*vkm95+`GGhZ}g59Kdsz#cKs zf}!NsM~A|LiK!HVBklx!Ya!|m;m*L7PJ^)SiWDoqp{kg$&my~FhJVlf<{P^%HO>{a z(c%kcCQpj6h5`~#3S4+qx;i8~ywcFGpuZCtU5@Gx43h41t316*!t;`6`P9<7G`W3n z-z@V>nJR;t4Ot{q*NB1=9B5 z1fE#mawb+AbTmOJFE&8G$@SC$SBt7Gamk4$Y|W4Ju&2%HtRUiy-gN(X6O3cGyYPgV zNldxOdP;Y!6fcyKqV=_5-JH4sSi4bCW0~^%2A0d_@A)t%?Iq2%TrcsWxmK?yLI4<7 zVo;huy2myNWXRP_eGeG;u`W8P-e5DqQr7TZr(0_P^n;NC!QNnJlP<#KxAi#?n25%w zdqpXU<+}&9#U#}rD16v|s;u%V_nFK~6v0kR%SKG3-YCjq-khW-pzjil2GL*@GujsPM~V`lu@Y3c9t4?Ef{20h8F`wYfYLE(!rKRs}~4o8&f zm6`q7X~H(44GRiwC~B3W848??lXTZCmWNPD8^jp@%V|4ZE)6;GU?Azotjg7F-kk(!CDX^~;s^0>b z*eMXsc`hm$$*-~rJlQdg`oAv^X1Of!u4D)HC=`||rTf<4jmoa*u0!Y6m78W?Fg2&I zUgfxK0_zqd>-Y_8pi%^KhmoWwh2d-^{lmHr>&DZ2eor3&lSh76Or~F77X=2y*0xfR zmYX>ebpBGc~;|+Z`g=lQ==&iy#&&d%HKh{ zvLfW5dt^Tq^&vBU-}1vcqBAHE`|W>qMBoMhFPbddYNURILo7*5YG9q;=`apKD9 zIFJLnKOK zx@8Q;IS*%84;cV$(ugU7E=rUfN;0}kc&O(5R*ee{U4An&!F|nV$^+bUAnfx34U)}@ zbV+fF>vVfoA}18|L!bU~q6jy5u8-wd<`qytigTi?#h01|Z1zs7?9~{1?%)4{+&Wt7 zCV5_U)M}C|*=MFIOd{Hw?u~82VCH(xb95cWrm)sogSysXNFfspcvujiKtf47>A9^F zjmtZpN_de_+283sr;T{qJ&)1`jPEGx;~+8Bl&|w$FhdYvo($5WT*O+GWuIhj7WV^S$O>)5&Q6- zVZTuYf9*K3gyg=Z3svMGR~@#EP+yARVAn(fjk5h+C?$dPq3a97aeXxn%@}=? ziW8>+6ZMjI{@4P?Ud8%#kr~$Fc7Ha_l*_eX+a;-r;AF=Kd)dNP{^i_cy0;8Zm#wpo z?I+aDvkLSNBuA(0l{7tP8g$0tXBLh4PcS{1>lZl~Ile(zk;8g$+U8bH&}*MvMbo3y zub6*HU90-AaSls9oMKJ>!r--7A#eZfS^SHuK`HWrh+cP}+Z+Dy`!%6w>>7sZ55n)= zWtlktu3V0q1HLVl=}U{gP*@`(!!nNNtBZ#gej87qii%d7@{FEp^=3%Zn4X-5Sv5rSlEm6@q-It`;=+Ya&zK1_ z`PHd9eu>aIRGDh7?+AdffeZ#{UH1aCQxM80 zgGSj@*!$3?l`=lzF3A-t-T5+Y6Jp)c2)=*xt;gk1A;>=Z^v zFwAU8SjYPfnUUhtnAGDnpPaeVKv*BAH_ifZufaM8!AR&Il`}6uX1MWL{NX%4`yC*O z^VXyJ$DIFzNoE0L+e26B1B@-tBHJ@JTXCC1G5kT}i?5eH=dSQH-{6?1=L6PBLA7jf z;@CuPKv?F;i*b~j>BlRBaJkJyPQjDH;?SGfDsyO-Y zspe$hmYz?QQ@s=0{!V4-s#+-zw|XlW4s54;|y5;(LCTFP4k925lZim1PQ=LT>UC;LGSV9CFe*q5~fozAOZ$;SmK^$Cn;JU1dJXBo*XxbID;I~H`q+BBV&qLrMkyXw2 z%O4~!+zFbm1)*y8f(PJsl7d7yXh(vCp#hq zVOkg=YjP80lz^Kr4$ffWtOFhw1{~%t1P!eK5B&iyKe-MJ_?W`IIQ2XN4qWN@)c^-H zMbzVVm8{eTEic)u1oI#hFc?5-%)$k>99N4?K}V&HQtZ=v}^0(jQQ1ZO?q&WfL(pMys7fGPMFaGUh6 z$MXLnTRT7l_ds-47MQNng3ZGUKB=DYDz zzZ)`@2+}D6)Ty{%=il!Azjp&|0j-8UyA*2wjogiym(~7!%m4o^fA)D`SqR;)b7QiB zjiKD1(~p1y1drDL|Fs^tk>%gp{QBKHL2EfdXt%Ce{=GlIfyqzu|3Av3>fLY{W)3uy lz>9++&baaWgM0m#|IFGB+NVRC%TF-?fv2mV%Q~loCIB)u@;(3n literal 0 HcmV?d00001 diff --git a/src/conformance/conformance_test/composition_examples/stale_swapchain.png.license b/src/conformance/conformance_test/composition_examples/stale_swapchain.png.license new file mode 100644 index 00000000..6d36aba0 --- /dev/null +++ b/src/conformance/conformance_test/composition_examples/stale_swapchain.png.license @@ -0,0 +1,3 @@ +SPDX-FileCopyrightText: 2020-2023, The Khronos Group Inc. + +SPDX-License-Identifier: Apache-2.0 diff --git a/src/conformance/conformance_test/test_FrameSubmission.cpp b/src/conformance/conformance_test/test_FrameSubmission.cpp index 1a343852..77662b8d 100644 --- a/src/conformance/conformance_test/test_FrameSubmission.cpp +++ b/src/conformance/conformance_test/test_FrameSubmission.cpp @@ -31,6 +31,7 @@ #define ENUM_LIST(name, val) name, constexpr XrEnvironmentBlendMode SupportedBlendModes[] = {XR_LIST_ENUM_XrEnvironmentBlendMode(ENUM_LIST)}; +#undef ENUM_LIST namespace Conformance { @@ -204,7 +205,8 @@ namespace Conformance CHECK(XR_SUCCESS == xrRequestExitSession(session)); - REQUIRE(FrameIterator::RunResult::Success == FrameIterator(&session).RunToSessionState(XR_SESSION_STATE_STOPPING, 5s)); + FrameIterator frameIterator(&session); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); CHECK(XR_SUCCESS == xrEndSession(session)); diff --git a/src/conformance/conformance_test/test_HapticInterrupt.cpp b/src/conformance/conformance_test/test_HapticInterrupt.cpp index 89422738..c5ad5d63 100644 --- a/src/conformance/conformance_test/test_HapticInterrupt.cpp +++ b/src/conformance/conformance_test/test_HapticInterrupt.cpp @@ -39,7 +39,7 @@ namespace Conformance { constexpr XrVector3f Up{0, 1, 0}; - TEST_CASE("Haptic Interrupt", "[scenario][interactive][no_auto]") + TEST_CASE("HapticInterrupt", "[scenario][interactive][no_auto]") { const char* instructions = "Press the select button on either hand to begin a 2 second haptic output. " diff --git a/src/conformance/conformance_test/test_InteractiveThrow.cpp b/src/conformance/conformance_test/test_InteractiveThrow.cpp index c3fdd4f8..b70c86b2 100644 --- a/src/conformance/conformance_test/test_InteractiveThrow.cpp +++ b/src/conformance/conformance_test/test_InteractiveThrow.cpp @@ -37,7 +37,7 @@ namespace Conformance // Purpose: Verify behavior of action timing and action space linear/angular velocity through throwing // 1. Use action state changed timestamp to query velocities // 2. Use action space velocities at various rigid offsets to verify "lever arm" effect is computed by runtime. - TEST_CASE("Interactive Throw", "[scenario][interactive][no_auto]") + TEST_CASE("InteractiveThrow", "[scenario][interactive][no_auto]") { const char* instructions = "Press and hold 'select' to spawn three rigidly-attached cubes to that controller. " diff --git a/src/conformance/conformance_test/test_LayerComposition.cpp b/src/conformance/conformance_test/test_LayerComposition.cpp index d3cf88c8..049d9340 100644 --- a/src/conformance/conformance_test/test_LayerComposition.cpp +++ b/src/conformance/conformance_test/test_LayerComposition.cpp @@ -653,4 +653,64 @@ namespace Conformance RenderLoop(compositionHelper.GetSession(), updateLayers).Loop(); } + + TEST_CASE("StaleSwapchain", "[composition][interactive][no_auto]") + { + CompositionHelper compositionHelper("Stale swapchain"); + InteractiveLayerManager interactiveLayerManager(compositionHelper, "stale_swapchain.png", + "Updates swapchain of each square at 1Hz. " + "Square on left should be constantly green, and square on right " + "should switch between green and blue every second. " + "If there is any flicker on the green square, " + "likely at the same time as the other square changes color, " + "that is a failure."); + compositionHelper.GetInteractionManager().AttachActionSets(); + compositionHelper.BeginSession(); + + const XrSpace viewSpace = compositionHelper.CreateReferenceSpace(XR_REFERENCE_SPACE_TYPE_VIEW, XrPosef{Quat::Identity, {0, 0, -1}}); + + constexpr int ImageSize = 1; + + // Create an array swapchain + auto swapchainCreateInfo = + compositionHelper.DefaultColorSwapchainCreateInfo(ImageSize, ImageSize, 0, GetGlobalData().graphicsPlugin->GetSRGBA8Format()); + swapchainCreateInfo.usageFlags |= XR_SWAPCHAIN_USAGE_TRANSFER_DST_BIT; + const XrSwapchain constantColorSwapchain = compositionHelper.CreateSwapchain(swapchainCreateInfo); + const XrSwapchain alternatingColorSwapchain = compositionHelper.CreateSwapchain(swapchainCreateInfo); + + RGBAImage images[2] = {RGBAImage(ImageSize, ImageSize), RGBAImage(ImageSize, ImageSize)}; + images[0].DrawRect(0, 0, ImageSize, ImageSize, Colors::Green); + images[1].DrawRect(0, 0, ImageSize, ImageSize, Colors::Blue); + images[0].ConvertToSRGB(); + images[1].ConvertToSRGB(); + + XrCompositionLayerQuad* const constantQuad = + compositionHelper.CreateQuadLayer(constantColorSwapchain, viewSpace, 0.02f, XrPosef{Quat::Identity, {-0.1f, 0, -1}}); + interactiveLayerManager.AddLayer(constantQuad); + + XrCompositionLayerQuad* const alternatingQuad = + compositionHelper.CreateQuadLayer(alternatingColorSwapchain, viewSpace, 0.02f, XrPosef{Quat::Identity, {0.1f, 0, -1}}); + interactiveLayerManager.AddLayer(alternatingQuad); + + XrTime lastUpdate = 0; + bool alternatingIndex = false; + RenderLoop(compositionHelper.GetSession(), [&](const XrFrameState& frameState) { + // Failing this test may create a flashing image. 1Hz is well outside the + // documented normal range for photosensitive epilepsy (rarely as low as 3Hz). + // Regardless, failures may e.g. create a black flash every second, so we use a + // small square to minimise any effects of the failure condition. + if (lastUpdate == 0 || (frameState.predictedDisplayTime - lastUpdate) >= 1_xrSeconds) { + lastUpdate = frameState.predictedDisplayTime; + compositionHelper.AcquireWaitReleaseImage(constantColorSwapchain, [&](const XrSwapchainImageBaseHeader* swapchainImage) { + GetGlobalData().graphicsPlugin->CopyRGBAImage(swapchainImage, 0, images[0]); + }); + compositionHelper.AcquireWaitReleaseImage(alternatingColorSwapchain, [&](const XrSwapchainImageBaseHeader* swapchainImage) { + GetGlobalData().graphicsPlugin->CopyRGBAImage(swapchainImage, 0, images[(uint32_t)alternatingIndex]); + alternatingIndex = !alternatingIndex; + }); + } + return interactiveLayerManager.EndFrame(frameState); + }).Loop(); + } + } // namespace Conformance diff --git a/src/conformance/conformance_test/test_Swapchains.cpp b/src/conformance/conformance_test/test_Swapchains.cpp index db0b5a93..42eddf4e 100644 --- a/src/conformance/conformance_test/test_Swapchains.cpp +++ b/src/conformance/conformance_test/test_Swapchains.cpp @@ -628,13 +628,8 @@ namespace Conformance auto graphicsPlugin = globalData.GetGraphicsPlugin(); - // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); XrSwapchain swapchain{XR_NULL_HANDLE}; XrExtent2Di widthHeight{0, 0}; // 0,0 means Use defaults. @@ -673,7 +668,7 @@ namespace Conformance } if ((j % 2) == 0) { - runResult = frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); } } diff --git a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp index 2b6fda21..751defe6 100644 --- a/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_debug_utils.cpp @@ -796,9 +796,6 @@ namespace Conformance AutoBasicSession::createSession | AutoBasicSession::createSpaces | AutoBasicSession::createSwapchains, instance); FrameIterator frameIterator(&session); - auto timeout = (globalData.options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Create a label struct for initial testing XrDebugUtilsLabelEXT first_label = {XR_TYPE_DEBUG_UTILS_LABEL_EXT}; first_label.labelName = first_individual_label_name; @@ -862,8 +859,7 @@ namespace Conformance // Begin the session now. { - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_READY); XrSessionBeginInfo session_begin_info = {XR_TYPE_SESSION_BEGIN_INFO}; session_begin_info.primaryViewConfigurationType = GetGlobalData().GetOptions().viewConfigurationValue; @@ -942,8 +938,7 @@ namespace Conformance { REQUIRE_RESULT(XR_SUCCESS, xrRequestExitSession(session)); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); REQUIRE_RESULT(XR_SUCCESS, xrEndSession(session)); } @@ -1062,11 +1057,8 @@ namespace Conformance AutoBasicInstance instance({XR_EXT_DEBUG_UTILS_EXTENSION_NAME}); AutoBasicSession session(AutoBasicSession::createSession, instance); - // Wait until the runtime is ready for us to begin a session - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_READY); // Must call extension functions through a function pointer: diff --git a/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp b/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp index e62710a1..19b20bbf 100644 --- a/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_eye_gaze_interaction.cpp @@ -43,7 +43,7 @@ namespace Conformance MakeSystemPropertiesBoolChecker(XrSystemEyeGazeInteractionPropertiesEXT{XR_TYPE_SYSTEM_EYE_GAZE_INTERACTION_PROPERTIES_EXT}, &XrSystemEyeGazeInteractionPropertiesEXT::supportsEyeGazeInteraction); - TEST_CASE("XR_EXT_eye_gaze_interaction", "[XR_EXT_eye_gaze_interaction][interactive][no_auto]") + TEST_CASE("XR_EXT_eye_gaze_interaction", "[scenario][interactive][no_auto][XR_EXT_eye_gaze_interaction]") { GlobalData& globalData = GetGlobalData(); if (!globalData.IsInstanceExtensionSupported(XR_EXT_EYE_GAZE_INTERACTION_EXTENSION_NAME)) { @@ -373,7 +373,7 @@ namespace Conformance } } - TEST_CASE("XR_EXT_eye_gaze_interaction-interactive_gaze_only", "[XR_EXT_eye_gaze_interaction][scenario][interactive][no_auto]") + TEST_CASE("XR_EXT_eye_gaze_interaction-interactive_gaze_only", "[scenario][interactive][no_auto][XR_EXT_eye_gaze_interaction]") { GlobalData& globalData = GetGlobalData(); diff --git a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp index e37130fa..adea1a48 100644 --- a/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_hand_tracking.cpp @@ -20,6 +20,7 @@ #include "utilities/system_properties_helper.h" #include "utilities/utils.h" #include +#include #include #include "common/xr_linear.h" @@ -137,10 +138,8 @@ namespace Conformance REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); // Wait until the runtime is ready for us to begin a session - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_READY); for (auto hand : {LEFT_HAND, RIGHT_HAND}) { std::array, HAND_COUNT> jointLocations; @@ -172,18 +171,12 @@ namespace Conformance // the returned radius must be a positive value. REQUIRE(jointLocations[hand][i].radius > 0); - // If an XrHandJointVelocitiesEXT structure is chained to XrHandJointLocationsEXT::next, - // the returned XrHandJointLocationsEXT::isActive is true, and the velocity is observed - // or can be calculated by the runtime, the runtime must fill in the linear velocity of - // each hand joint within the reference frame of baseSpace and set the - // XR_SPACE_VELOCITY_LINEAR_VALID_BIT. - // Similarly, if an XrHandJointVelocitiesEXT structure is chained to - // XrHandJointLocationsEXT::next, the returned XrHandJointLocationsEXT::isActive is true, - // and the angular velocity is observed or can be calculated by the runtime, the runtime - // must fill in the angular velocity of each joint within the reference frame of baseSpace - // and set the XR_SPACE_VELOCITY_ANGULAR_VALID_BIT. - REQUIRE((jointVelocities[hand][i].velocityFlags & XR_SPACE_VELOCITY_LINEAR_VALID_BIT) != 0); - REQUIRE((jointVelocities[hand][i].velocityFlags & XR_SPACE_VELOCITY_ANGULAR_VALID_BIT) != 0); + // From https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#_locate_hand_joints + // the velocity is observed or can be calculated by the runtime, the runtime must + // fill in the linear velocity of each hand joint within the reference frame of baseSpace + // and set the XR_SPACE_VELOCITY_LINEAR_VALID_BIT. + // So we cannot validate the jointVelocities flags XR_SPACE_VELOCITY_LINEAR_VALID_BIT + // or XR_SPACE_VELOCITY_ANGULAR_VALID_BIT. } else { // If the returned isActive is false, it indicates the hand tracker did not detect the hand @@ -207,10 +200,8 @@ namespace Conformance REQUIRE_RESULT(xrCreateReferenceSpace(session, &localSpaceCreateInfo, &localSpace), XR_SUCCESS); // Wait until the runtime is ready for us to begin a session - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_READY, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_READY); // The application must input jointCount as described by the XrHandJointSetEXT when creating the XrHandTrackerEXT. // Otherwise, the runtime must return XR_ERROR_VALIDATION_FAILURE. @@ -328,6 +319,74 @@ namespace Conformance locateInfo.baseSpace = localSpace; locateInfo.time = frameState.predictedDisplayTime; REQUIRE(XR_SUCCESS == xrLocateHandJointsEXT(handTracker[hand], &locateInfo, &locations)); + + if (locations.isActive) { + const auto& wrist = jointLocations[hand][XR_HAND_JOINT_WRIST_EXT]; + const auto& palm = jointLocations[hand][XR_HAND_JOINT_PALM_EXT]; + const auto& middleMetacarpal = jointLocations[hand][XR_HAND_JOINT_MIDDLE_METACARPAL_EXT]; + const auto& middleProximal = jointLocations[hand][XR_HAND_JOINT_MIDDLE_PROXIMAL_EXT]; + + // Check if the palm joint is located correctly for each hand. + if ((palm.locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT) != 0 && + (middleMetacarpal.locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT) != 0 && + (middleProximal.locationFlags & XR_SPACE_LOCATION_POSITION_VALID_BIT) != 0) { + + // The palm joint is located at the center of the middle finger’s metacarpal bone. + XrPosef expectedPalmPose; + XrVector3f_Lerp(&expectedPalmPose.position, &middleMetacarpal.pose.position, &middleProximal.pose.position, 0.5f); + REQUIRE_THAT(palm.pose.position.x, Catch::Matchers::WithinRel(expectedPalmPose.position.x)); + REQUIRE_THAT(palm.pose.position.y, Catch::Matchers::WithinRel(expectedPalmPose.position.y)); + REQUIRE_THAT(palm.pose.position.z, Catch::Matchers::WithinRel(expectedPalmPose.position.z)); + } + + // Check the palm orientation for each hand if we have valid orientation. + if ((palm.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0 && + (middleMetacarpal.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0) { + + // The backward (+Z) direction is parallel to the middle finger’s metacarpal bone, and points away from the + // fingertips. + XrVector3f zAxis{0, 0, 1.0f}; + XrVector3f palmZAxis, middleMetacarpalZAxis; + XrQuaternionf_RotateVector3f(&palmZAxis, &palm.pose.orientation, &zAxis); + XrQuaternionf_RotateVector3f(&middleMetacarpalZAxis, &middleMetacarpal.pose.orientation, &zAxis); + REQUIRE_THAT(XrVector3f_Dot(&palmZAxis, &middleMetacarpalZAxis), Catch::Matchers::WithinRel(1.0f)); + + // The up (+Y) direction is perpendicular to palm surface and pointing towards the back of the hand. + // We can compare this to the +Y axis of the middle metacarpal bone to check gross direction. + XrVector3f yAxis{0, 1.0f, 0}; + XrVector3f palmYAxis, middleMetacarpalYAxis; + XrQuaternionf_RotateVector3f(&palmYAxis, &palm.pose.orientation, &yAxis); + XrQuaternionf_RotateVector3f(&middleMetacarpalYAxis, &middleMetacarpal.pose.orientation, &yAxis); + REQUIRE_THAT(XrVector3f_Dot(&palmYAxis, &middleMetacarpalYAxis), Catch::Matchers::WithinRel(1.0f, 0.1f)); + + // The X direction is perpendicular to the Y and Z directions and follows the right hand rule. + XrVector3f xAxis{0, 1.0f, 0}; + XrVector3f palmXAxis, middleMetacarpalXAxis; + XrQuaternionf_RotateVector3f(&palmXAxis, &palm.pose.orientation, &xAxis); + XrQuaternionf_RotateVector3f(&middleMetacarpalXAxis, &middleMetacarpal.pose.orientation, &xAxis); + REQUIRE_THAT(XrVector3f_Dot(&palmXAxis, &middleMetacarpalXAxis), Catch::Matchers::WithinRel(1.0f, 0.1f)); + } + + // Check the orientation of the wrist pose is correct for each hand, we can only reliably test the +Z direction + // programmatically. + if ((wrist.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0 && + (middleMetacarpal.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0) { + + // The (wrist) backward (+Z) direction is parallel to the line from wrist joint to middle finger metacarpal joint, + // and points away from the fingertips. + XrVector3f zAxis{0, 0, 1.0f}; + XrVector3f wristZAxis, fromMiddleMetacarpalToWrist; + XrQuaternionf_RotateVector3f(&wristZAxis, &wrist.pose.orientation, &zAxis); + XrVector3f_Sub(&fromMiddleMetacarpalToWrist, &wrist.pose.position, &middleMetacarpal.pose.position); + XrVector3f_Normalize(&fromMiddleMetacarpalToWrist); + // 0.1 here represents 26 degrees variance between these orientations; which is more than can reasonable be + // explained by numerical inaccuracy... + REQUIRE_THAT(XrVector3f_Dot(&wristZAxis, &fromMiddleMetacarpalToWrist), Catch::Matchers::WithinRel(1.0f, 0.1f)); + if (XrVector3f_Dot(&wristZAxis, &fromMiddleMetacarpalToWrist) > 0.03) { + WARN("Variance between wrist z axis orientation and metacarpal greater than 14 degrees!"); + } + } + } } // Check if user has requested to complete the test. diff --git a/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp b/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp index 0207a3cb..3c80ca86 100644 --- a/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_local_floor.cpp @@ -70,17 +70,12 @@ namespace Conformance AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces, instance); - // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Render one frame to get a predicted display time for the xrLocateSpace calls. - runResult = frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); // If stage space is defined, then LOCAL_FLOOR height off the floor must match STAGE diff --git a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp index 3d2cb6c5..7955250a 100644 --- a/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp +++ b/src/conformance/conformance_test/test_XR_EXT_palm_pose.cpp @@ -472,14 +472,9 @@ namespace Conformance AutoBasicSession::createSwapchains, instance); - // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); XrSessionActionSetsAttachInfo attachInfo{XR_TYPE_SESSION_ACTION_SETS_ATTACH_INFO}; attachInfo.actionSets = &actionSet; diff --git a/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp b/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp deleted file mode 100644 index ee44c0dc..00000000 --- a/src/conformance/conformance_test/test_XR_EXT_performance_settings.cpp +++ /dev/null @@ -1,28 +0,0 @@ -// Copyright (c) 2019-2023, The Khronos Group Inc. -// -// SPDX-License-Identifier: Apache-2.0 -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -#include - -namespace Conformance -{ - - TEST_CASE("XR_EXT_performance_settings", "") - { - // XrResult xrPerfSettingsSetPerformanceLevelEXT(XrSession session, XrPerfSettingsDomainEXT domain, - // XrPerfSettingsLevelEXT level); - } - -} // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp index bff5dd31..afe4f16e 100644 --- a/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_D3D11_enable.cpp @@ -43,8 +43,10 @@ namespace Conformance SKIP(XR_KHR_D3D11_ENABLE_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -57,12 +59,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -71,7 +76,7 @@ namespace Conformance SECTION("NULL D3D11 device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingD3D11KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.device = nullptr; @@ -81,10 +86,39 @@ namespace Conformance graphicsPlugin->ShutdownDevice(); } + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingD3D11KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.device = nullptr; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingD3D11KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + } + SECTION("Singlethreaded D3D11 device") { // Verify that the runtime supports D3D11_CREATE_DEVICE_SINGLETHREADED. - graphicsPlugin->InitializeDevice(instance, systemId, true, D3D11_CREATE_DEVICE_SINGLETHREADED); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true, D3D11_CREATE_DEVICE_SINGLETHREADED)); XrGraphicsBindingD3D11KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); @@ -108,11 +142,28 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetD3D11GraphicsRequirementsKHR = + GetInstanceExtensionFunction(instance, "xrGetD3D11GraphicsRequirementsKHR"); + + XrGraphicsRequirementsD3D11KHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_D3D11_KHR}; + REQUIRE(xrGetD3D11GraphicsRequirementsKHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingD3D11KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsD3D11KHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_D3D11_KHR}; + REQUIRE(xrGetD3D11GraphicsRequirementsKHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.adapterLuid.HighPart == graphicsRequirements.adapterLuid.HighPart); + REQUIRE(referenceGraphicsRequirements.adapterLuid.LowPart == graphicsRequirements.adapterLuid.LowPart); + REQUIRE(referenceGraphicsRequirements.minFeatureLevel == graphicsRequirements.minFeatureLevel); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp index 7bd3e69a..b815305c 100644 --- a/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_D3D12_enable.cpp @@ -42,8 +42,10 @@ namespace Conformance SKIP(XR_KHR_D3D12_ENABLE_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -56,12 +58,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -70,8 +75,9 @@ namespace Conformance SECTION("NULL D3D12 device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); - XrGraphicsBindingD3D12KHR graphicsBinding = {XR_TYPE_GRAPHICS_BINDING_D3D12_KHR, nullptr}; + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingD3D12KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.device = nullptr; sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); @@ -81,6 +87,35 @@ namespace Conformance graphicsPlugin->ShutdownDevice(); } + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingD3D12KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.device = nullptr; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingD3D12KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + } + SECTION("Multiple session with same device") { auto createSwapchains = [](std::shared_ptr graphicsPlugin, XrSession session) { @@ -96,11 +131,28 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetD3D12GraphicsRequirementsKHR = + GetInstanceExtensionFunction(instance, "xrGetD3D12GraphicsRequirementsKHR"); + + XrGraphicsRequirementsD3D12KHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_D3D12_KHR}; + REQUIRE(xrGetD3D12GraphicsRequirementsKHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingD3D12KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsD3D12KHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_D3D12_KHR}; + REQUIRE(xrGetD3D12GraphicsRequirementsKHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.adapterLuid.HighPart == graphicsRequirements.adapterLuid.HighPart); + REQUIRE(referenceGraphicsRequirements.adapterLuid.LowPart == graphicsRequirements.adapterLuid.LowPart); + REQUIRE(referenceGraphicsRequirements.minFeatureLevel == graphicsRequirements.minFeatureLevel); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp index 429e460c..2d31abe0 100644 --- a/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_OpenGL_ES_enable.cpp @@ -39,8 +39,10 @@ namespace Conformance SKIP(XR_KHR_OPENGL_ES_ENABLE_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -53,12 +55,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -69,7 +74,7 @@ namespace Conformance // tests related to the graphics binding are OS specific SECTION("NULL context: context is NULL") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLESAndroidKHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.context = EGL_NO_CONTEXT; @@ -78,6 +83,35 @@ namespace Conformance cleanup.Destroy(); graphicsPlugin->ShutdownDevice(); } + + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingOpenGLESAndroidKHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.context = EGL_NO_CONTEXT; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingOpenGLESAndroidKHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + } + } + SECTION("Multiple session with same device") { auto createSwapchains = [](std::shared_ptr graphicsPlugin, XrSession session) { @@ -93,11 +127,27 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetOpenGLESGraphicsRequirementsKHR = + GetInstanceExtensionFunction(instance, "xrGetOpenGLESGraphicsRequirementsKHR"); + + XrGraphicsRequirementsOpenGLESKHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_OPENGL_ES_KHR}; + REQUIRE(xrGetOpenGLESGraphicsRequirementsKHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLESAndroidKHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsOpenGLESKHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_OPENGL_ES_KHR}; + REQUIRE(xrGetOpenGLESGraphicsRequirementsKHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.maxApiVersionSupported == graphicsRequirements.maxApiVersionSupported); + REQUIRE(referenceGraphicsRequirements.minApiVersionSupported == graphicsRequirements.minApiVersionSupported); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp index 490500d6..bad989d0 100644 --- a/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_OpenGL_enable.cpp @@ -40,8 +40,10 @@ namespace Conformance SKIP(XR_KHR_OPENGL_ENABLE_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -54,12 +56,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -70,7 +75,7 @@ namespace Conformance // tests related to the graphics binding are OS specific SECTION("NULL context: both are NULL") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLWin32KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.hDC = nullptr; @@ -80,9 +85,10 @@ namespace Conformance cleanup.Destroy(); graphicsPlugin->ShutdownDevice(); } + SECTION("NULL context: DC is NULL") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLWin32KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.hDC = nullptr; @@ -91,9 +97,10 @@ namespace Conformance cleanup.Destroy(); graphicsPlugin->ShutdownDevice(); } + SECTION("NULL context: GLRC is NULL") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLWin32KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.hGLRC = nullptr; @@ -102,11 +109,41 @@ namespace Conformance cleanup.Destroy(); graphicsPlugin->ShutdownDevice(); } + + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingOpenGLWin32KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.hDC = nullptr; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingOpenGLWin32KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + } + // This test dies in the tear-down wglMakeCurrent in ksGpuContext_Destroy, turn it off for now. #if 0 SECTION("Context for runtime is not current") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = graphicsPlugin->GetGraphicsBinding(); // Exercise presence of unrecognized extensions, which the runtime should ignore. @@ -151,11 +188,27 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetOpenGLGraphicsRequirementsKHR = + GetInstanceExtensionFunction(instance, "xrGetOpenGLGraphicsRequirementsKHR"); + + XrGraphicsRequirementsOpenGLKHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_OPENGL_KHR}; + REQUIRE(xrGetOpenGLGraphicsRequirementsKHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingOpenGLWin32KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsOpenGLKHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_OPENGL_KHR}; + REQUIRE(xrGetOpenGLGraphicsRequirementsKHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.maxApiVersionSupported == graphicsRequirements.maxApiVersionSupported); + REQUIRE(referenceGraphicsRequirements.minApiVersionSupported == graphicsRequirements.minApiVersionSupported); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp index a9ba932f..5cf450ae 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cube.cpp @@ -46,8 +46,6 @@ namespace Conformance } auto graphicsPlugin = globalData.GetGraphicsPlugin(); - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); AutoBasicInstance instance({XR_KHR_COMPOSITION_LAYER_CUBE_EXTENSION_NAME}); AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | @@ -55,8 +53,7 @@ namespace Conformance instance); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // At this point we have a session ready for us to generate custom frames for. // The current XrSessionState is XR_SESSION_STATE_FOCUSED. @@ -104,7 +101,7 @@ namespace Conformance // } XrCompositionLayerCubeKHR; for (const XrQuaternionf& orientation : orientationTestArray) { - runResult = frameIterator.PrepareSubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.PrepareSubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); // Set up our cubemap layer. We always make two, and some of the time we @@ -150,7 +147,6 @@ namespace Conformance result = xrRequestExitSession(session.GetSession()); CHECK(result == XR_SUCCESS); - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp index 624d320c..4013bde9 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_cylinder.cpp @@ -43,17 +43,13 @@ namespace Conformance SKIP("Test run not using graphics plugin"); } - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - AutoBasicInstance instance({XR_KHR_COMPOSITION_LAYER_CYLINDER_EXTENSION_NAME}); AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces, instance); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // At this point we have a session ready for us to generate custom frames for. // The current XrSessionState is XR_SESSION_STATE_FOCUSED. @@ -71,7 +67,7 @@ namespace Conformance std::array radiusTestArray{0, 1.f, INFINITY}; // Spec explicitly supports radius 0 and +infinity for (float radius : radiusTestArray) { - runResult = frameIterator.PrepareSubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.PrepareSubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); // Set up our cylinder layer. We always make two, and some of the time we @@ -119,7 +115,6 @@ namespace Conformance result = xrRequestExitSession(session.GetSession()); CHECK(result == XR_SUCCESS); - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp index 0b1d18ed..445f70b6 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_depth.cpp @@ -42,8 +42,6 @@ namespace Conformance auto graphicsPlugin = globalData.GetGraphicsPlugin(); - auto timeout = (globalData.options.debugMode ? 3600s : 10s); - CAPTURE(timeout); AutoBasicInstance instance({XR_KHR_COMPOSITION_LAYER_DEPTH_EXTENSION_NAME}); AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces, @@ -51,8 +49,7 @@ namespace Conformance REQUIRE(session.IsValidHandle()); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // At this point we have a session ready for us to generate custom frames for. // The current XrSessionState is XR_SESSION_STATE_FOCUSED. @@ -98,7 +95,7 @@ namespace Conformance DepthVaryingInfo{0.0f, 1.0f, std::numeric_limits::max(), minimum_useful_z}}; for (const DepthVaryingInfo& varyingInfo : varyingInfoTestArray) { - runResult = frameIterator.PrepareSubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.PrepareSubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); { @@ -148,8 +145,7 @@ namespace Conformance CHECK(result == XR_SUCCESS); } - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); for (const XrSwapchain& swapchain : depthSwapchains) { XrResult result = xrDestroySwapchain(swapchain); diff --git a/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp b/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp index 64deccc5..3c30ee33 100644 --- a/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_composition_layer_equirect.cpp @@ -46,8 +46,6 @@ namespace Conformance } auto graphicsPlugin = globalData.GetGraphicsPlugin(); - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); AutoBasicInstance instance({XR_KHR_COMPOSITION_LAYER_EQUIRECT_EXTENSION_NAME}); AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | @@ -55,8 +53,7 @@ namespace Conformance instance); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // At this point we have a session ready for us to generate custom frames for. // The current XrSessionState is XR_SESSION_STATE_FOCUSED. @@ -109,7 +106,7 @@ namespace Conformance }; for (const XrQuaternionf& orientation : orientationTestArray) { - runResult = frameIterator.PrepareSubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.PrepareSubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); // Set up our equirect layer. We always make two, and some of the time we @@ -173,7 +170,6 @@ namespace Conformance result = xrRequestExitSession(session.GetSession()); CHECK(result == XR_SUCCESS); - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp b/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp index fa8718c9..b7a2d4de 100644 --- a/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_convert_timespec_time.cpp @@ -134,13 +134,12 @@ namespace Conformance CAPTURE(timespecBefore); // Wait until the runtime is ready for us to begin a session - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Submit a frame and query the time for the next frame - frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); + REQUIRE(runResult == FrameIterator::RunResult::Success); XrTime nextFrameTime = frameIterator.frameState.predictedDisplayTime; // predicted display time is required to be a time in the future so it is fair to assume it is after now. diff --git a/src/conformance/conformance_test/test_XR_KHR_headless.cpp b/src/conformance/conformance_test/test_XR_KHR_headless.cpp deleted file mode 100644 index 2c266c43..00000000 --- a/src/conformance/conformance_test/test_XR_KHR_headless.cpp +++ /dev/null @@ -1,72 +0,0 @@ -// Copyright (c) 2019-2023, The Khronos Group Inc. -// -// SPDX-License-Identifier: Apache-2.0 -// -// Licensed under the Apache License, Version 2.0 (the "License"); -// you may not use this file except in compliance with the License. -// You may obtain a copy of the License at -// -// http://www.apache.org/licenses/LICENSE-2.0 -// -// Unless required by applicable law or agreed to in writing, software -// distributed under the License is distributed on an "AS IS" BASIS, -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -// See the License for the specific language governing permissions and -// limitations under the License. - -#include "utilities/utils.h" -#include "conformance_utils.h" -#include "conformance_framework.h" -#include -#include -#include -#include -#include -#include -#include - -namespace Conformance -{ - TEST_CASE("XR_KHR_headless", "") - { -#ifdef XR_KHR_headless - GlobalData& globalData = GetGlobalData(); - - // When this extension is enabled, the behavior of existing functions that interact with the - // graphics subsystem is altered. When calling the function xrCreateSession with no graphics - // binding structure, the session will be created as headless. - // - // When operating with a headless session, the function xrEnumerateSwapchainFormats must return - // an empty list of formats. Calls to functions xrCreateSwapchain, xrDestroySwapchain, - // xrAcquireSwapchainImage, xrWaitFrame are invalid. All other functions, including those - // related to tracking, input and haptics, are unaffected. - if (!globalData.IsInstanceExtensionEnabled("XR_KHR_headless")) { - return; - } - - AutoBasicSession session(AutoBasicSession::createSession | AutoBasicSession::skipGraphics); - - SECTION("xrEnumerateSwapchainFormats should return XR_SUCCESS but zero formats.") - { - uint32_t countOutput = UINT32_MAX; - REQUIRE(xrEnumerateSwapchainFormats(session, 0, &countOutput, nullptr) == XR_SUCCESS); - REQUIRE(countOutput == 0); - - int64_t formats[100]; - countOutput = UINT32_MAX; - REQUIRE(xrEnumerateSwapchainFormats(session, sizeof(formats) / sizeof(formats[0]), &countOutput, formats) == XR_SUCCESS); - REQUIRE(countOutput == 0); - } - - // Calls to functions xrCreateSwapchain, xrDestroySwapchain, xrAcquireSwapchainImage, - // xrWaitFrame are invalid, but there isn't a specification for what happens when called. - - // We begin a session and call valid session functions. - XrSessionBeginInfo sessionBeginInfo{XR_TYPE_SESSION_BEGIN_INFO, nullptr, globalData.options.viewConfigurationValue}; - REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrBeginSession(session, &sessionBeginInfo)); - - // To do: call input and tracking functions here. - REQUIRE_RESULT_UNQUALIFIED_SUCCESS(xrEndSession(session)); -#endif // XR_KHR_headless - } -} // namespace Conformance diff --git a/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp b/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp index 5a2fe5d5..2fd3a31e 100644 --- a/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_visibility_mask.cpp @@ -278,8 +278,11 @@ namespace Conformance { MeshHandle mesh; bool meshCoversHiddenArea = maskType == XR_VISIBILITY_MASK_TYPE_HIDDEN_TRIANGLE_MESH_KHR; - XrColor4f color = meshCoversHiddenArea ? BrightRed : DarkSlateGrey; - XrColor4f bgColor = meshCoversHiddenArea ? DarkSlateGrey : BrightRed; + + XrColor4f normalBgColor = GetGlobalData().GetClearColorForBackground(); + + XrColor4f color = meshCoversHiddenArea ? BrightRed : normalBgColor; + XrColor4f bgColor = meshCoversHiddenArea ? normalBgColor : BrightRed; XrVisibilityMaskKHR visibilityMask{XR_TYPE_VISIBILITY_MASK_KHR}; @@ -327,7 +330,7 @@ namespace Conformance return {mesh, bgColor}; } - TEST_CASE("XR_KHR_visibility_mask-interactive", "[XR_KHR_visibility_mask][interactive][no_auto]") + TEST_CASE("XR_KHR_visibility_mask-interactive", "[scenario][interactive][no_auto][XR_KHR_visibility_mask]") { // successcodes="XR_SUCCESS,XR_SESSION_LOSS_PENDING" // errorcodes="XR_ERROR_HANDLE_INVALID,XR_ERROR_INSTANCE_LOST,XR_ERROR_RUNTIME_FAILURE,XR_ERROR_VALIDATION_FAILURE, @@ -379,7 +382,7 @@ namespace Conformance SKIP("No vertices returned, so no visibility mask available in this system."); } - std::vector bgColors{meshProjectionLayerHelper.GetViewCount(), DarkSlateGrey}; + std::vector bgColors{meshProjectionLayerHelper.GetViewCount(), Colors::Green}; // should be overwritten before render // const auto maskTypes = {XR_VISIBILITY_MASK_TYPE_HIDDEN_TRIANGLE_MESH_KHR, XR_VISIBILITY_MASK_TYPE_VISIBLE_TRIANGLE_MESH_KHR, // XR_VISIBILITY_MASK_TYPE_LINE_LOOP_KHR}; diff --git a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp index fb12e8cc..3a6cc0f8 100644 --- a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable.cpp @@ -40,8 +40,10 @@ namespace Conformance SKIP(XR_KHR_VULKAN_ENABLE_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -54,12 +56,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -68,7 +73,7 @@ namespace Conformance SECTION("Valid vulkan device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkanKHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); @@ -79,7 +84,7 @@ namespace Conformance SECTION("NULL vulkan device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkanKHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.device = VK_NULL_HANDLE; @@ -89,6 +94,35 @@ namespace Conformance graphicsPlugin->ShutdownDevice(); } + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingVulkanKHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.device = VK_NULL_HANDLE; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingVulkanKHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + } + SECTION("Multiple session with same device") { auto createSwapchains = [](std::shared_ptr graphicsPlugin, XrSession session) { @@ -104,11 +138,27 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetVulkanGraphicsRequirementsKHR = + GetInstanceExtensionFunction(instance, "xrGetVulkanGraphicsRequirementsKHR"); + + XrGraphicsRequirementsVulkanKHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_VULKAN_KHR}; + REQUIRE(xrGetVulkanGraphicsRequirementsKHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkanKHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsVulkanKHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_VULKAN_KHR}; + REQUIRE(xrGetVulkanGraphicsRequirementsKHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.maxApiVersionSupported == graphicsRequirements.maxApiVersionSupported); + REQUIRE(referenceGraphicsRequirements.minApiVersionSupported == graphicsRequirements.minApiVersionSupported); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp index 7c458dd4..6f2b1f62 100644 --- a/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_vulkan_enable2.cpp @@ -40,8 +40,10 @@ namespace Conformance SKIP(XR_KHR_VULKAN_ENABLE2_EXTENSION_NAME " not enabled"); } - AutoBasicInstance instance{AutoBasicInstance::createSystemId}; - XrSystemId systemId = instance.systemId; + AutoBasicInstance instance; + + XrSystemId systemId{XR_NULL_SYSTEM_ID}; + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); // Create the graphics plugin we'll need to exercise session create functionality below. std::shared_ptr graphicsPlugin; @@ -54,12 +56,15 @@ namespace Conformance // We'll use this XrSession and XrSessionCreateInfo for testing below. XrSession session = XR_NULL_HANDLE_CPP; - XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO, nullptr, 0, systemId}; + + XrSessionCreateInfo sessionCreateInfo{XR_TYPE_SESSION_CREATE_INFO}; + sessionCreateInfo.systemId = systemId; + CleanupSessionOnScopeExit cleanup(session); SECTION("No graphics binding") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); sessionCreateInfo.next = nullptr; CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); cleanup.Destroy(); @@ -68,7 +73,7 @@ namespace Conformance SECTION("Valid vulkan device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkan2KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); @@ -79,7 +84,7 @@ namespace Conformance SECTION("NULL vulkan device") { - graphicsPlugin->InitializeDevice(instance, systemId, true); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkan2KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); graphicsBinding.device = nullptr; @@ -89,6 +94,35 @@ namespace Conformance graphicsPlugin->ShutdownDevice(); } + SECTION("Valid session after bad session") + { + // Pass invalid binding the first time + { + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingVulkan2KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + graphicsBinding.device = nullptr; + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_ERROR_GRAPHICS_DEVICE_INVALID); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + + // Using the same instance pass valid binding the second time + { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); + XrGraphicsBindingVulkan2KHR graphicsBinding = + *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); + sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); + cleanup.Destroy(); + graphicsPlugin->ShutdownDevice(); + } + } + SECTION("Multiple session with same device") { auto createSwapchains = [](std::shared_ptr graphicsPlugin, XrSession session) { @@ -104,11 +138,27 @@ namespace Conformance } }; - graphicsPlugin->InitializeDevice(instance, systemId, true); + auto xrGetVulkanGraphicsRequirements2KHR = + GetInstanceExtensionFunction(instance, "xrGetVulkanGraphicsRequirements2KHR"); + + XrGraphicsRequirementsVulkan2KHR referenceGraphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_VULKAN2_KHR}; + REQUIRE(xrGetVulkanGraphicsRequirements2KHR(instance, systemId, &referenceGraphicsRequirements) == XR_SUCCESS); + + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, true)); XrGraphicsBindingVulkan2KHR graphicsBinding = *reinterpret_cast(graphicsPlugin->GetGraphicsBinding()); sessionCreateInfo.next = reinterpret_cast(&graphicsBinding); for (int i = 0; i < 3; ++i) { + REQUIRE(XR_SUCCESS == FindBasicSystem(instance.GetInstance(), &systemId)); + sessionCreateInfo.systemId = systemId; + + XrGraphicsRequirementsVulkan2KHR graphicsRequirements{XR_TYPE_GRAPHICS_REQUIREMENTS_VULKAN2_KHR}; + REQUIRE(xrGetVulkanGraphicsRequirements2KHR(instance, systemId, &graphicsRequirements) == XR_SUCCESS); + + // We expect that the graphics requirements don't change... + REQUIRE(referenceGraphicsRequirements.maxApiVersionSupported == graphicsRequirements.maxApiVersionSupported); + REQUIRE(referenceGraphicsRequirements.minApiVersionSupported == graphicsRequirements.minApiVersionSupported); + CHECK(xrCreateSession(instance, &sessionCreateInfo, &session) == XR_SUCCESS); createSwapchains(graphicsPlugin, session); CHECK(xrDestroySession(session) == XR_SUCCESS); diff --git a/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp b/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp index 78edae70..fb43b0a2 100644 --- a/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp +++ b/src/conformance/conformance_test/test_XR_KHR_win32_convert_performance_counter_time.cpp @@ -121,13 +121,12 @@ namespace Conformance CAPTURE(qpcBefore); // Wait until the runtime is ready for us to begin a session - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Submit a frame and query the time for the next frame - frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); + REQUIRE(runResult == FrameIterator::RunResult::Success); XrTime nextFrameTime = frameIterator.frameState.predictedDisplayTime; // predicted display time is required to be a time in the future so it is fair to assume it is after now. diff --git a/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp b/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp index 25b63a78..585b95ff 100644 --- a/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp +++ b/src/conformance/conformance_test/test_XR_META_performance_metrics.cpp @@ -85,10 +85,6 @@ namespace Conformance SECTION("Query metrics after xrEndFrame") { - // how long the test should wait for the app to get focus: 10 seconds in release, 1hr in debug builds. - auto timeout = (globalData.options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Get a session started. AutoBasicSession session(AutoBasicSession::createInstance | AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces, @@ -101,11 +97,10 @@ namespace Conformance // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Render one frame to some frame stats. - runResult = frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); for (const XrPath& path : paths) { diff --git a/src/conformance/conformance_test/test_actions.cpp b/src/conformance/conformance_test/test_actions.cpp index ce37068f..fdb1132b 100644 --- a/src/conformance/conformance_test/test_actions.cpp +++ b/src/conformance/conformance_test/test_actions.cpp @@ -478,10 +478,7 @@ namespace Conformance CAPTURE(bindingPath); const XrActionType& actionType = inputSourcePathData.Type; -#define ENUM_NAME(e, val) {(XrActionType)val, #e}, - std::map actionTypeToString = {XR_LIST_ENUM_XrActionType(ENUM_NAME)}; -#undef ENUM_NAME - std::string actionTypeStr = actionTypeToString.at(actionType); + std::string actionTypeStr = enum_to_string(actionType); CAPTURE(actionTypeStr); XrAction* actionRef; @@ -1181,18 +1178,41 @@ namespace Conformance compositionHelper.BeginSession(); SECTION("No Focus") { - compositionHelper.GetInteractionManager().AddActionSet(actionSet); - compositionHelper.GetInteractionManager().AttachActionSets(); + SECTION("No active action sets") + { + XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; + syncInfo.activeActionSets = nullptr; + syncInfo.countActiveActionSets = 0; - XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; - XrActiveActionSet activeActionSet{actionSet}; - syncInfo.activeActionSets = &activeActionSet; - syncInfo.countActiveActionSets = 1; + { + INFO("No action sets attached"); + + REQUIRE_RESULT_SUCCEEDED(xrSyncActions(compositionHelper.GetSession(), &syncInfo)); + } + { + INFO("With action sets attached"); + + compositionHelper.GetInteractionManager().AddActionSet(actionSet); + compositionHelper.GetInteractionManager().AttachActionSets(); + + REQUIRE_RESULT_SUCCEEDED(xrSyncActions(compositionHelper.GetSession(), &syncInfo)); + } + } + SECTION("Active action sets") + { + compositionHelper.GetInteractionManager().AddActionSet(actionSet); + compositionHelper.GetInteractionManager().AttachActionSets(); - REQUIRE_RESULT(xrSyncActions(compositionHelper.GetSession(), &syncInfo), XR_SESSION_NOT_FOCUSED); + XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; + XrActiveActionSet activeActionSet{actionSet}; + syncInfo.activeActionSets = &activeActionSet; + syncInfo.countActiveActionSets = 1; - REQUIRE_RESULT(xrGetActionStateBoolean(compositionHelper.GetSession(), &getInfo, &actionStateBoolean), XR_SUCCESS); - REQUIRE_FALSE(actionStateBoolean.isActive); + REQUIRE_RESULT(xrSyncActions(compositionHelper.GetSession(), &syncInfo), XR_SESSION_NOT_FOCUSED); + + REQUIRE_RESULT(xrGetActionStateBoolean(compositionHelper.GetSession(), &getInfo, &actionStateBoolean), XR_SUCCESS); + REQUIRE_FALSE(actionStateBoolean.isActive); + } } SECTION("Focus") { @@ -1631,6 +1651,12 @@ namespace Conformance REQUIRE_RESULT(xrCreateActionSet(compositionHelper.GetInstance(), &actionSetCreateInfo, &subactionPathFreeActionSet), XR_SUCCESS); + XrActionSet unboundActionActionSet{XR_NULL_HANDLE}; + strcpy(actionSetCreateInfo.localizedActionSetName, "test action set localized name 5"); + strcpy(actionSetCreateInfo.actionSetName, "test_action_set_name_5"); + REQUIRE_RESULT(xrCreateActionSet(compositionHelper.GetInstance(), &actionSetCreateInfo, &unboundActionActionSet), + XR_SUCCESS); + XrAction leftHandAction{XR_NULL_HANDLE}; actionCreateInfo.actionType = XR_ACTION_TYPE_BOOLEAN_INPUT; strcpy(actionCreateInfo.localizedActionName, "test select action"); @@ -1645,12 +1671,19 @@ namespace Conformance actionCreateInfo.subactionPaths = &rightHandPath; REQUIRE_RESULT(xrCreateAction(actionSet, &actionCreateInfo, &rightHandAction), XR_SUCCESS); + XrAction unboundAction{XR_NULL_HANDLE}; + strcpy(actionCreateInfo.localizedActionName, "test select action 4"); + strcpy(actionCreateInfo.actionName, "test_select_action_4"); + actionCreateInfo.subactionPaths = &defaultDevicePath; + REQUIRE_RESULT(xrCreateAction(unboundActionActionSet, &actionCreateInfo, &unboundAction), XR_SUCCESS); + compositionHelper.GetInteractionManager().AddActionBindings( simpleControllerInteractionProfile, {{leftHandAction, StringToPath(compositionHelper.GetInstance(), "/user/hand/left/input/select/click")}, {rightHandAction, StringToPath(compositionHelper.GetInstance(), "/user/hand/right/input/select/click")}}); compositionHelper.GetInteractionManager().AddActionSet(actionSet); compositionHelper.GetInteractionManager().AddActionSet(subactionPathFreeActionSet); + compositionHelper.GetInteractionManager().AddActionSet(unboundActionActionSet); compositionHelper.GetInteractionManager().AttachActionSets(); if (globalData.leftHandUnderTest) { @@ -1663,6 +1696,7 @@ namespace Conformance XrActionsSyncInfo syncInfo{XR_TYPE_ACTIONS_SYNC_INFO}; XrActiveActionSet activeActionSet{actionSet}; XrActiveActionSet subactionPathFreeActiveActionSet{subactionPathFreeActionSet}; + XrActiveActionSet unboundActionActiveActionSet{unboundActionActionSet}; syncInfo.activeActionSets = &activeActionSet; syncInfo.countActiveActionSets = 1; @@ -1754,6 +1788,11 @@ namespace Conformance REQUIRE_RESULT(xrCreateActionSet(compositionHelper.GetInstance(), &actionSetCreateInfo, &unattachedActionSet), XR_SUCCESS); + INFO("Unbound action"); + syncInfo.activeActionSets = &unboundActionActiveActionSet; + unboundActionActiveActionSet.subactionPath = defaultDevicePath; + actionLayerManager.SyncActionsUntilFocusWithMessage(syncInfo); + INFO("unattached action set"); XrActiveActionSet activeActionSet2 = {unattachedActionSet}; syncInfo.countActiveActionSets = 1; diff --git a/src/conformance/conformance_test/test_multithreading.cpp b/src/conformance/conformance_test/test_multithreading.cpp index 75559484..77632e72 100644 --- a/src/conformance/conformance_test/test_multithreading.cpp +++ b/src/conformance/conformance_test/test_multithreading.cpp @@ -328,10 +328,6 @@ namespace Conformance // Exercise session multithreading. { - // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - ThreadTestEnvironment env(invocationCount); env.GetAutoBasicSession().Init(AutoBasicSession::beginSession | AutoBasicSession::createActions | AutoBasicSession::createSpaces | AutoBasicSession::createSwapchains); @@ -386,8 +382,7 @@ namespace Conformance // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&env.GetAutoBasicSession()); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); env.lastFrameTime = frameIterator.frameState.predictedDisplayTime; diff --git a/src/conformance/conformance_test/test_xrCreateInstance.cpp b/src/conformance/conformance_test/test_xrCreateInstance.cpp index e7ab8232..6af99a31 100644 --- a/src/conformance/conformance_test/test_xrCreateInstance.cpp +++ b/src/conformance/conformance_test/test_xrCreateInstance.cpp @@ -24,6 +24,7 @@ #include #include +#include namespace Conformance { @@ -254,6 +255,7 @@ namespace Conformance // To do: Enable any layers and extensions available. } } + TEST_CASE("xrDestroyInstance", "") { SECTION("null handle") @@ -263,6 +265,18 @@ namespace Conformance CHECK(xrDestroyInstance(XR_NULL_HANDLE_CPP) == XR_ERROR_HANDLE_INVALID); } + SECTION("destroy on a different thread to create") + { + for (int i = 0; i < 2; ++i) { + CAPTURE(i); + AutoBasicInstance instance; + XrResult destroyResult = XR_ERROR_RUNTIME_FAILURE; + std::thread t([&destroyResult, &instance] { destroyResult = xrDestroyInstance(instance); }); + t.join(); + REQUIRE(destroyResult == XR_SUCCESS); + } + } + OPTIONAL_INVALID_HANDLE_VALIDATION_SECTION { GlobalData& globalData = GetGlobalData(); diff --git a/src/conformance/conformance_test/test_xrCreateSession.cpp b/src/conformance/conformance_test/test_xrCreateSession.cpp index a29d4612..cb6f1314 100644 --- a/src/conformance/conformance_test/test_xrCreateSession.cpp +++ b/src/conformance/conformance_test/test_xrCreateSession.cpp @@ -27,6 +27,7 @@ #include #include #include +#include namespace Conformance { @@ -81,7 +82,7 @@ namespace Conformance // Happens if the application tries to create the session but hasn't queried the graphics requirements (e.g. // xrGetD3D12GraphicsRequirementsKHR). This spec states that applications must call this, but // how we enforce it in conformance testing is problematic because a specific return code isn't specified. - graphicsPlugin->InitializeDevice(instance, systemId, false /* checkGraphicsRequirements */); + REQUIRE(graphicsPlugin->InitializeDevice(instance, systemId, false /* checkGraphicsRequirements */)); sessionCreateInfo.next = graphicsPlugin->GetGraphicsBinding(); XrResult sessionResult = xrCreateSession(instance, &sessionCreateInfo, &session); CHECK_THAT(sessionResult, In({XR_ERROR_VALIDATION_FAILURE, XR_ERROR_GRAPHICS_REQUIREMENTS_CALL_MISSING})); @@ -127,4 +128,31 @@ namespace Conformance } } } + + TEST_CASE("xrDestroySession", "") + { + SECTION("null handle") + { + CHECK(xrDestroySession(XR_NULL_HANDLE_CPP) == XR_ERROR_HANDLE_INVALID); + } + + SECTION("destroy on a different thread to create") + { + for (int i = 0; i < 2; ++i) { + CAPTURE(i); + + AutoBasicInstance instance; + AutoBasicSession session(AutoBasicSession::createSession, instance); + XrResult destroySessionResult = XR_ERROR_RUNTIME_FAILURE; + XrResult destroyInstanceResult = XR_ERROR_RUNTIME_FAILURE; + std::thread t([&destroySessionResult, &destroyInstanceResult, &session, &instance] { + destroySessionResult = xrDestroySession(session); + destroyInstanceResult = xrDestroyInstance(instance); + }); + t.join(); + REQUIRE(destroySessionResult == XR_SUCCESS); + REQUIRE(destroyInstanceResult == XR_SUCCESS); + } + } + } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_xrLocateSpace.cpp b/src/conformance/conformance_test/test_xrLocateSpace.cpp index 99c2c6bc..96b54ffa 100644 --- a/src/conformance/conformance_test/test_xrLocateSpace.cpp +++ b/src/conformance/conformance_test/test_xrLocateSpace.cpp @@ -22,31 +22,28 @@ #include #include +#include #include #include #include #include +#include #include namespace Conformance { TEST_CASE("xrLocateSpace", "") { - // how long the test should wait for the app to get focus: 10 seconds in release, infinite in debug builds. - auto timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Get a session started. AutoBasicSession session(AutoBasicSession::createInstance | AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces); // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Render one frame to get a predicted display time for the xrLocateSpace calls. - runResult = frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); XrResult result; @@ -88,6 +85,31 @@ namespace Conformance XrTime time = frameIterator.frameState.predictedDisplayTime; CHECK(time != 0); + SECTION("valid inputs") + { + // two identical spaces: + result = xrCreateReferenceSpace(session, &spaceCreateInfo, &spaceA); + CHECK(result == XR_SUCCESS); + result = xrCreateReferenceSpace(session, &spaceCreateInfo, &spaceB); + CHECK(result == XR_SUCCESS); + + // Exercise the predicted display time + result = xrLocateSpace(spaceA, spaceB, time, &location); + CHECK(result == XR_SUCCESS); + + // Exercise 40ms ago (or the first valid time, whichever is later) + result = xrLocateSpace(spaceA, spaceB, std::max(time - 40_xrMilliseconds, (XrTime)1), &location); + CHECK(result == XR_SUCCESS); + + // Exercise 1s ago (or the first valid time, whichever is later) + result = xrLocateSpace(spaceA, spaceB, std::max(time - 1_xrSeconds, (XrTime)1), &location); + CHECK(result == XR_SUCCESS); + + // cleanup + CHECK(XR_SUCCESS == xrDestroySpace(spaceA)); + CHECK(XR_SUCCESS == xrDestroySpace(spaceB)); + } + SECTION("wrong inputs") { // two identical spaces: @@ -233,7 +255,6 @@ namespace Conformance result = xrRequestExitSession(session); CHECK(result == XR_SUCCESS); - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); } } // namespace Conformance diff --git a/src/conformance/conformance_test/test_xrLocateViews.cpp b/src/conformance/conformance_test/test_xrLocateViews.cpp index f1917d62..583dc66e 100644 --- a/src/conformance/conformance_test/test_xrLocateViews.cpp +++ b/src/conformance/conformance_test/test_xrLocateViews.cpp @@ -30,21 +30,16 @@ namespace Conformance { GlobalData& globalData = GetGlobalData(); - // how long the test should wait for the app to get focus: 10 seconds in release, 1hr in debug builds. - auto timeout = (globalData.options.debugMode ? 3600s : 10s); - CAPTURE(timeout); - // Get a session started. AutoBasicSession session(AutoBasicSession::createInstance | AutoBasicSession::createSession | AutoBasicSession::beginSession | AutoBasicSession::createSwapchains | AutoBasicSession::createSpaces); // Get frames iterating to the point of app focused state. This will draw frames along the way. FrameIterator frameIterator(&session); - FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeout); - REQUIRE(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); // Render one frame to get a predicted display time for the xrLocateViews calls. - runResult = frameIterator.SubmitFrame(); + FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); REQUIRE(runResult == FrameIterator::RunResult::Success); XrResult result; @@ -170,7 +165,6 @@ namespace Conformance result = xrRequestExitSession(session); CHECK(result == XR_SUCCESS); - runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING, timeout); - CHECK(runResult == FrameIterator::RunResult::Success); + frameIterator.RunToSessionState(XR_SESSION_STATE_STOPPING); } } // namespace Conformance diff --git a/src/conformance/framework/composition_utils.h b/src/conformance/framework/composition_utils.h index f20cdff5..21f835de 100644 --- a/src/conformance/framework/composition_utils.h +++ b/src/conformance/framework/composition_utils.h @@ -348,7 +348,9 @@ namespace Conformance constexpr XrColor4f Blue = {0, 0, 1, 1}; constexpr XrColor4f Yellow = {1, 1, 0, 1}; constexpr XrColor4f Orange = {1, 0.65f, 0, 1}; + constexpr XrColor4f Magenta = {1, 0, 1, 1}; constexpr XrColor4f Transparent = {0, 0, 0, 0}; + constexpr XrColor4f Black = {0, 0, 0, 1}; /// A list of unique colors, not including red which is a "failure color". constexpr std::array UniqueColors{Green, Blue, Yellow, Orange}; diff --git a/src/conformance/framework/conformance_framework.cpp b/src/conformance/framework/conformance_framework.cpp index a4dcb037..9a2fa3c7 100644 --- a/src/conformance/framework/conformance_framework.cpp +++ b/src/conformance/framework/conformance_framework.cpp @@ -16,6 +16,7 @@ #include "conformance_framework.h" +#include "composition_utils.h" // for Colors #include "graphics_plugin.h" #include "platform_plugin.h" #include "report.h" @@ -479,11 +480,6 @@ namespace Conformance if (IsInstanceExtensionEnabled(XR_MND_HEADLESS_EXTENSION_NAME)) { return false; } -#ifdef XR_KHR_headless - if (IsInstanceExtensionEnabled(XR_KHR_HEADLESS_EXTENSION_NAME)) { - return false; - } -#endif return true; } @@ -497,4 +493,18 @@ namespace Conformance std::unique_lock lock(dataMutex); conformanceReport.swapchainFormats.emplace_back(format, name); } + + XrColor4f GlobalData::GetClearColorForBackground() const + { + switch (options.environmentBlendModeValue) { + case XR_ENVIRONMENT_BLEND_MODE_OPAQUE: + return DarkSlateGrey; + case XR_ENVIRONMENT_BLEND_MODE_ADDITIVE: + return Colors::Black; + case XR_ENVIRONMENT_BLEND_MODE_ALPHA_BLEND: + return Colors::Transparent; + default: + XRC_THROW("Encountered unrecognized environment blend mode value while determining background color."); + } + } } // namespace Conformance diff --git a/src/conformance/framework/conformance_framework.h b/src/conformance/framework/conformance_framework.h index 20bc7f7e..7cb29a8b 100644 --- a/src/conformance/framework/conformance_framework.h +++ b/src/conformance/framework/conformance_framework.h @@ -355,7 +355,7 @@ namespace Conformance std::shared_ptr GetGraphicsPlugin(); /// Returns true if under the current test environment we require a graphics plugin. This may - /// be false, for example, if the XR_KHR_headless is enabled. + /// be false, for example, if the XR_MND_headless extension is enabled. bool IsGraphicsPluginRequired() const; /// Returns true if a graphics plugin was supplied, or if IsGraphicsPluginRequired() is true. @@ -364,6 +364,9 @@ namespace Conformance /// Record a swapchain format as being supported and tested. void PushSwapchainFormat(int64_t format, const std::string& name); + /// Calculate the clear color to use for the background based on the XrEnvironmentBlendMode in use. + XrColor4f GetClearColorForBackground() const; + public: /// Guards all member data. mutable std::recursive_mutex dataMutex; @@ -605,6 +608,7 @@ MAKE_ENUM_TO_STRING_FUNC(XrViewConfigurationType); MAKE_ENUM_TO_STRING_FUNC(XrVisibilityMaskTypeKHR); MAKE_ENUM_TO_STRING_FUNC(XrFormFactor); MAKE_ENUM_TO_STRING_FUNC(XrEnvironmentBlendMode); +MAKE_ENUM_TO_STRING_FUNC(XrActionType); namespace Catch { diff --git a/src/conformance/framework/conformance_utils.cpp b/src/conformance/framework/conformance_utils.cpp index 3a90faff..d796083b 100644 --- a/src/conformance/framework/conformance_utils.cpp +++ b/src/conformance/framework/conformance_utils.cpp @@ -392,7 +392,7 @@ namespace Conformance // Normally the testing requires a graphics plugin. However, there's currently one case in // which that's not true: when a headless extension is enabled. In that case the - // runtime supports creating a session without a graphics system. See XR_MND_headless and/or XR_KHR_headless doc. + // runtime supports creating a session without a graphics system. See XR_MND_headless doc. if (graphicsPlugin && enableGraphicsSystem) { // If the following fails then this app has a bug, not the runtime. assert(graphicsPlugin->IsInitialized()); @@ -481,8 +481,8 @@ namespace Conformance // that the session is ready. // timeout in case the runtime will never transition to READY: 10s in release, no practical limit in debug - std::chrono::nanoseconds timeout = (GetGlobalData().options.debugMode ? 3600s : 10s); - CountdownTimer countdownTimer(timeout); + auto timeoutToTransitionToSessionState = (GetGlobalData().options.debugMode ? 3600s : 10s); + CountdownTimer countdownTimer(timeoutToTransitionToSessionState); while ((sessionState != XR_SESSION_STATE_READY) && (!countdownTimer.IsTimeUp())) { XrEventDataBuffer eventData{XR_TYPE_EVENT_DATA_BUFFER}; @@ -503,7 +503,24 @@ namespace Conformance } if (sessionState != XR_SESSION_STATE_READY) { - XRC_THROW("Time out waiting for XR_SESSION_STATE_READY session state change"); + // We have failed this check with the timeout. This is a pretty common place to fail + // so we will offer helpful hints for the most common errors - as well as a generic + // message. + + // https://registry.khronos.org/OpenXR/specs/1.0/html/xrspec.html#sessionstatechanged-description + // If the system supports a user engagement sensor and runtime is in XR_SESSION_STATE_IDLE state, + // the runtime should not transition to the XR_SESSION_STATE_READY state until the user starts + // engaging with the device. + + std::string extraInfo; + if (sessionState == XR_SESSION_STATE_IDLE) { + extraInfo = + " If this system supports a user engagement sensor, the runtime may not transition to XR_SESSION_STATE_READY state until the user starts engaging with the device."; + } + + CAPTURE(timeoutToTransitionToSessionState); + CAPTURE(sessionState); + FAIL("Time out waiting for XR_SESSION_STATE_READY session state change after creating a new session." << extraInfo); } XrSessionBeginInfo sessionBeginInfo{XR_TYPE_SESSION_BEGIN_INFO, @@ -627,17 +644,13 @@ namespace Conformance //////////////////////////////////////////////////////////////////////////////////////////////// FrameIterator::FrameIterator(AutoBasicSession* autoBasicSession_) - : autoBasicSession(autoBasicSession_) - , sessionState(autoBasicSession->GetSessionState()) - , countdownTimer() - , frameState() - , viewVector() - { - } - - void FrameIterator::SetAutoBasicSession(AutoBasicSession* autoBasicSession_) + : autoBasicSession(autoBasicSession_), sessionState(autoBasicSession->GetSessionState()), frameState(), viewVector() { - autoBasicSession = autoBasicSession_; + XRC_CHECK_THROW(autoBasicSession); + XRC_CHECK_THROW(autoBasicSession->instance); + XRC_CHECK_THROW(autoBasicSession->session); + XRC_CHECK_THROW(!autoBasicSession->viewConfigurationTypeVector.empty()); + XRC_CHECK_THROW(!autoBasicSession->environmentBlendModeVector.empty()); } XrSessionState FrameIterator::GetCurrentSessionState() const @@ -647,12 +660,6 @@ namespace Conformance FrameIterator::TickResult FrameIterator::PollEvent() { - // App must have called SetAutoBasicSession and set flags enabling these. - if (!autoBasicSession) - return TickResult::Error; - if (!autoBasicSession->instance) - return TickResult::Error; - XrEventDataBuffer eventData{XR_TYPE_EVENT_DATA_BUFFER}; XrResult result = xrPollEvent(autoBasicSession->instance, &eventData); @@ -693,11 +700,10 @@ namespace Conformance if (!GetGlobalData().IsUsingGraphicsPlugin()) return RunResult::Success; - // App must have called SetAutoBasicSession and set flags enabling these. - if (!autoBasicSession) - return RunResult::Error; - if (autoBasicSession->swapchainVector.empty()) + if (autoBasicSession->swapchainVector.empty()) { + // AutoBasicSession must be created with flags including AutoBasicSession::createSwapchains return RunResult::Error; + } // Call the helper function for this. const XrDuration twoSeconds = 2_xrSeconds; @@ -715,15 +721,10 @@ namespace Conformance FrameIterator::RunResult FrameIterator::WaitAndBeginFrame() { - // App must have called SetAutoBasicSession and set flags enabling these. - if (!autoBasicSession) - return RunResult::Error; - if (!autoBasicSession->session) - return RunResult::Error; - if (autoBasicSession->spaceVector.empty()) - return RunResult::Error; - if (autoBasicSession->viewConfigurationTypeVector.empty()) + if (autoBasicSession->spaceVector.empty()) { + // AutoBasicSession must be created with flags including AutoBasicSession::createSpaces return RunResult::Error; + } XrResult result; @@ -756,13 +757,10 @@ namespace Conformance FrameIterator::RunResult FrameIterator::PrepareFrameEndInfo() { - // App must have called SetAutoBasicSession and set flags enabling these. - if (!autoBasicSession) - return RunResult::Error; - if (autoBasicSession->spaceVector.empty()) - return RunResult::Error; - if (autoBasicSession->environmentBlendModeVector.empty()) + if (autoBasicSession->spaceVector.empty()) { + // AutoBasicSession must be created with flags including AutoBasicSession::createSpaces return RunResult::Error; + } if (GetGlobalData().IsUsingGraphicsPlugin() && autoBasicSession->swapchainVector.empty()) return RunResult::Error; @@ -839,27 +837,28 @@ namespace Conformance // Runs until the given XrSessionState is achieved or timesout before so. // targetSessionState may be any XrSessionState, but some session states may require // special handling in order to get to, such as XR_SESSION_STATE_LOSS_PENDING. - FrameIterator::RunResult FrameIterator::RunToSessionState(XrSessionState targetSessionState, std::chrono::nanoseconds timeout) + void FrameIterator::RunToSessionState(XrSessionState targetSessionState) { - if (!autoBasicSession) // App must have called SetAutoBasicSession. - return RunResult::Error; + auto initialSessionState = sessionState; - countdownTimer.Restart(timeout); + auto timeoutToTransitionToSessionState = (GetGlobalData().options.debugMode ? 3600s : 10s); + CAPTURE(timeoutToTransitionToSessionState); + CountdownTimer countdownTimer(timeoutToTransitionToSessionState); while (!countdownTimer.IsTimeUp()) { TickResult tickResult = PollEvent(); + REQUIRE(tickResult != TickResult::Error); - if (tickResult == TickResult::Error) - return RunResult::Error; - - if (sessionState == targetSessionState) - return RunResult::Success; - - if ((sessionState == XR_SESSION_STATE_LOSS_PENDING) || (sessionState == XR_SESSION_STATE_EXITING) || - (sessionState == XR_SESSION_STATE_STOPPING)) { - return RunResult::Timeout; + if (sessionState == targetSessionState) { + // calling SUCCEED here to flush the CAPTURE / INFO messages from this function + SUCCEED(); + return; } + REQUIRE(sessionState != XR_SESSION_STATE_LOSS_PENDING); + REQUIRE(sessionState != XR_SESSION_STATE_EXITING); + REQUIRE(sessionState != XR_SESSION_STATE_STOPPING); + // At this point sessionState is one of XR_SESSION_STATE_UNKNOWN, // XR_SESSION_STATE_IDLE, XR_SESSION_STATE_READY, XR_SESSION_STATE_SYNCHRONIZED, // XR_SESSION_STATE_VISIBLE, XR_SESSION_STATE_FOCUSED. We proceed based on the @@ -882,9 +881,7 @@ namespace Conformance XrSessionBeginInfo sessionBeginInfo{ XR_TYPE_SESSION_BEGIN_INFO, globalData.GetPlatformPlugin()->PopulateNextFieldForStruct(XR_TYPE_SESSION_BEGIN_INFO), globalData.options.viewConfigurationValue}; - XrResult result = xrBeginSession(autoBasicSession->session, &sessionBeginInfo); - if (XR_FAILED(result)) - return RunResult::Error; + REQUIRE(xrBeginSession(autoBasicSession->session, &sessionBeginInfo) == XR_SUCCESS); } // Fall-through because frames must be submitted to get promoted from READY to SYNCHRONIZED. @@ -894,10 +891,8 @@ namespace Conformance case XR_SESSION_STATE_FOCUSED: { // In these states we need to submit frames. Otherwise the runtime won't // necessarily move us from synchronized to visible or focused. - RunResult runResult = SubmitFrame(); + REQUIRE(SubmitFrame() == RunResult::Success); - if (runResult == RunResult::Error) - return RunResult::Error; // Just keep going. We haven't reached the target state yet. break; } @@ -910,7 +905,17 @@ namespace Conformance } } - return RunResult::Timeout; + // We have failed this check with the timeout. This is a pretty common place to fail + // so we will offer helpful hints for the most common errors - as well as a generic + // message. + + std::string extraInfo; + if (targetSessionState == XR_SESSION_STATE_FOCUSED && initialSessionState == XR_SESSION_STATE_READY && + sessionState == XR_SESSION_STATE_VISIBLE) { + extraInfo = " This might indicate that some other (maybe system) application still has focus for the user."; + } + FAIL("Timeout while waiting for session state transition to: " << enum_to_string(targetSessionState) << " from initial state: " + << enum_to_string(initialSessionState) << "." << extraInfo); } bool WaitUntilPredicateWithTimeout(const std::function& predicate, const std::chrono::nanoseconds timeout, diff --git a/src/conformance/framework/conformance_utils.h b/src/conformance/framework/conformance_utils.h index d770f113..bcb1386a 100644 --- a/src/conformance/framework/conformance_utils.h +++ b/src/conformance/framework/conformance_utils.h @@ -398,11 +398,7 @@ namespace Conformance class CountdownTimer { public: - CountdownTimer() : stopwatch(), timeoutDuration() - { - } - - CountdownTimer(std::chrono::nanoseconds timeout) : stopwatch(), timeoutDuration(timeout) + explicit CountdownTimer(std::chrono::nanoseconds timeout) : stopwatch(), timeoutDuration(timeout) { stopwatch.Restart(); } @@ -717,11 +713,10 @@ namespace Conformance /// /// // Get frames iterating to the point of app focused state. This will draw frames along the way. /// FrameIterator frameIterator(&session); - /// FrameIterator::RunResult runResult = frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED, timeoutMicroseconds); - /// REQUIRE(runResult == FrameIterator::RunResult::Success); + /// frameIterator.RunToSessionState(XR_SESSION_STATE_FOCUSED); /// /// // Let's have the FrameIterator draw one more frame itself. - /// runResult = frameIterator.SubmitFrame(); + /// FrameIterator::RunResult runResult = frameIterator.SubmitFrame(); /// REQUIRE(runResult == FrameIterator::RunResult::Success); /// /// // Now let's draw a frame ourselves. @@ -739,12 +734,9 @@ namespace Conformance class FrameIterator { public: - FrameIterator(AutoBasicSession* autoBasicSession_ = nullptr); + explicit FrameIterator(AutoBasicSession* autoBasicSession_ = nullptr); ~FrameIterator() = default; - /// Must not be called after calling any other member function. - void SetAutoBasicSession(AutoBasicSession* autoBasicSession_); - XrSessionState GetCurrentSessionState() const; enum class TickResult @@ -797,16 +789,16 @@ namespace Conformance /// an example of this. RunResult SubmitFrame(); - /// Runs until the given XrSessionState is achieved or timesout before so. + /// Runs until the given XrSessionState is achieved or times out before so. /// targetSessionState may be any XrSessionState, but some session states may require /// special handling in order to get to, such as XR_SESSION_STATE_LOSS_PENDING. /// Will repeatedly call SubmitFrame if necessary to get to the desired state. - RunResult RunToSessionState(XrSessionState targetSessionState, std::chrono::nanoseconds timeout); + /// Will fail test if targetSessionState is not reached. + void RunToSessionState(XrSessionState targetSessionState); protected: - AutoBasicSession* autoBasicSession; + AutoBasicSession* const autoBasicSession; XrSessionState sessionState; - CountdownTimer countdownTimer; public: XrFrameState frameState; //< xrWaitFrame from WaitAndBeginFrame fills this in. diff --git a/src/conformance/framework/graphics_plugin.h b/src/conformance/framework/graphics_plugin.h index a26b00f4..baf6e864 100644 --- a/src/conformance/framework/graphics_plugin.h +++ b/src/conformance/framework/graphics_plugin.h @@ -18,6 +18,7 @@ #include "platform_plugin.h" #include "conformance_utils.h" +#include "conformance_framework.h" #include "utilities/Geometry.h" #include "RGBAImage.h" @@ -307,8 +308,15 @@ namespace Conformance AllocateSwapchainImageDataWithDepthSwapchain(size_t size, const XrSwapchainCreateInfo& colorSwapchainCreateInfo, XrSwapchain depthSwapchain, const XrSwapchainCreateInfo& depthSwapchainCreateInfo) = 0; - virtual void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex = 0, - XrColor4f bgColor = DarkSlateGrey) = 0; + /// Clears a slice to an arbitrary color + virtual void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) = 0; + + /// Clears to the background color which varies depending on the environment blend mode that is active. + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex = 0) + { + GlobalData& globalData = GetGlobalData(); + ClearImageSlice(colorSwapchainImage, imageArrayIndex, globalData.GetClearColorForBackground()); + } /// Create internal data for a mesh, returning a handle to refer to it. /// This handle expires when the internal data is cleared in Shutdown() and ShutdownDevice(). diff --git a/src/conformance/framework/graphics_plugin_d3d11.cpp b/src/conformance/framework/graphics_plugin_d3d11.cpp index a2dee28e..b5c9b551 100644 --- a/src/conformance/framework/graphics_plugin_d3d11.cpp +++ b/src/conformance/framework/graphics_plugin_d3d11.cpp @@ -200,8 +200,7 @@ namespace Conformance XrSwapchain depthSwapchain, const XrSwapchainCreateInfo& depthSwapchainCreateInfo) override; - void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor = DarkSlateGrey) override; + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) override; MeshHandle MakeSimpleMesh(span idx, span vtx) override; @@ -643,7 +642,7 @@ namespace Conformance } void D3D11GraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor) + XrColor4f color) { D3D11SwapchainImageData* swapchainData; @@ -655,7 +654,7 @@ namespace Conformance // Create RenderTargetView with original swapchain format (swapchain is typeless). ComPtr renderTargetView = CreateRenderTargetView(*swapchainData, imageIndex, imageArrayIndex); // TODO: Do not clear to a color when using a pass-through view configuration. - FLOAT bg[] = {bgColor.r, bgColor.g, bgColor.b, bgColor.a}; + FLOAT bg[] = {color.r, color.g, color.b, color.a}; d3d11DeviceContext->ClearRenderTargetView(renderTargetView.Get(), bg); // Clear depth buffer. diff --git a/src/conformance/framework/graphics_plugin_d3d12.cpp b/src/conformance/framework/graphics_plugin_d3d12.cpp index cd7b438d..0fd9e210 100644 --- a/src/conformance/framework/graphics_plugin_d3d12.cpp +++ b/src/conformance/framework/graphics_plugin_d3d12.cpp @@ -355,8 +355,7 @@ namespace Conformance XrSwapchain depthSwapchain, const XrSwapchainCreateInfo& depthSwapchainCreateInfo) override; - void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor = DarkSlateGrey) override; + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) override; MeshHandle MakeSimpleMesh(span idx, span vtx) override; @@ -937,7 +936,7 @@ namespace Conformance } void D3D12GraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor) + XrColor4f color) { D3D12SwapchainImageData* swapchainData; @@ -957,7 +956,7 @@ namespace Conformance D3D12_CPU_DESCRIPTOR_HANDLE renderTargetView = CreateRenderTargetView(colorTexture, imageArrayIndex, swapchainData->GetCreateInfo().format); // TODO: Do not clear to a color when using a pass-through view configuration. - FLOAT bg[] = {bgColor.r, bgColor.g, bgColor.b, bgColor.a}; + FLOAT bg[] = {color.r, color.g, color.b, color.a}; cmdList->ClearRenderTargetView(renderTargetView, bg, 0, nullptr); // Clear depth buffer. diff --git a/src/conformance/framework/graphics_plugin_opengl.cpp b/src/conformance/framework/graphics_plugin_opengl.cpp index 3ce587ab..b2d06d96 100644 --- a/src/conformance/framework/graphics_plugin_opengl.cpp +++ b/src/conformance/framework/graphics_plugin_opengl.cpp @@ -443,8 +443,7 @@ namespace Conformance XrSwapchain depthSwapchain, const XrSwapchainCreateInfo& depthSwapchainCreateInfo) override; - void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor = DarkSlateGrey) override; + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) override; MeshHandle MakeSimpleMesh(span idx, span vtx) override; @@ -1024,7 +1023,7 @@ namespace Conformance } void OpenGLGraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor) + XrColor4f color) { OpenGLSwapchainImageData* swapchainData; uint32_t imageIndex; @@ -1058,7 +1057,7 @@ namespace Conformance XRC_CHECK_THROW_GLCMD(glEnable(GL_SCISSOR_TEST)); // Clear swapchain and depth buffer. - XRC_CHECK_THROW_GLCMD(glClearColor(bgColor.r, bgColor.g, bgColor.b, bgColor.a)); + XRC_CHECK_THROW_GLCMD(glClearColor(color.r, color.g, color.b, color.a)); XRC_CHECK_THROW_GLCMD(glClearDepth(1.0f)); XRC_CHECK_THROW_GLCMD(glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT)); diff --git a/src/conformance/framework/graphics_plugin_opengles.cpp b/src/conformance/framework/graphics_plugin_opengles.cpp index 5632dd29..6930513e 100644 --- a/src/conformance/framework/graphics_plugin_opengles.cpp +++ b/src/conformance/framework/graphics_plugin_opengles.cpp @@ -291,8 +291,7 @@ namespace Conformance XrSwapchain depthSwapchain, const XrSwapchainCreateInfo& depthSwapchainCreateInfo) override; - void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor = DarkSlateGrey) override; + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) override; MeshHandle MakeSimpleMesh(span idx, span vtx) override; @@ -1088,7 +1087,7 @@ namespace Conformance } void OpenGLESGraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor) + XrColor4f color) { OpenGLESSwapchainImageData* swapchainData; uint32_t imageIndex; @@ -1121,7 +1120,7 @@ namespace Conformance GL(glEnable(GL_SCISSOR_TEST)); // Clear swapchain and depth buffer. - GL(glClearColor(bgColor.r, bgColor.g, bgColor.b, bgColor.a)); + GL(glClearColor(color.r, color.g, color.b, color.a)); GL(glClearDepthf(1.0f)); GL(glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT)); diff --git a/src/conformance/framework/graphics_plugin_vulkan.cpp b/src/conformance/framework/graphics_plugin_vulkan.cpp index 4db260a5..e8f869ef 100644 --- a/src/conformance/framework/graphics_plugin_vulkan.cpp +++ b/src/conformance/framework/graphics_plugin_vulkan.cpp @@ -629,8 +629,7 @@ namespace Conformance void SetViewportAndScissor(const VkRect2D& rect); - void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor = DarkSlateGrey) override; + void ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, XrColor4f color) override; MeshHandle MakeSimpleMesh(span idx, span vtx) override; @@ -1847,7 +1846,7 @@ namespace Conformance } void VulkanGraphicsPlugin::ClearImageSlice(const XrSwapchainImageBaseHeader* colorSwapchainImage, uint32_t imageArrayIndex, - XrColor4f bgColor) + XrColor4f color) { VulkanSwapchainImageData* swapchainData; uint32_t imageIndex; @@ -1882,10 +1881,10 @@ namespace Conformance // Clear the buffers static std::array clearValues; - clearValues[0].color.float32[0] = bgColor.r; - clearValues[0].color.float32[1] = bgColor.g; - clearValues[0].color.float32[2] = bgColor.b; - clearValues[0].color.float32[3] = bgColor.a; + clearValues[0].color.float32[0] = color.r; + clearValues[0].color.float32[1] = color.g; + clearValues[0].color.float32[2] = color.b; + clearValues[0].color.float32[3] = color.a; clearValues[1].depthStencil.depth = 1.0f; clearValues[1].depthStencil.stencil = 0; std::array clearAttachments{{ diff --git a/src/conformance/framework/mesh_projection_layer.cpp b/src/conformance/framework/mesh_projection_layer.cpp index af3990f6..5d572723 100644 --- a/src/conformance/framework/mesh_projection_layer.cpp +++ b/src/conformance/framework/mesh_projection_layer.cpp @@ -65,7 +65,7 @@ namespace Conformance MeshProjectionLayerHelper::MeshProjectionLayerHelper(CompositionHelper& compositionHelper) : m_baseHelper(compositionHelper, XR_REFERENCE_SPACE_TYPE_LOCAL) { - m_bgColors.resize(GetViewCount(), DarkSlateGrey); + m_bgColors.resize(GetViewCount(), Colors::Magenta); // should be overwritten before render } void MeshProjectionLayerHelper::SetMeshes(std::vector&& meshes) diff --git a/src/conformance/framework/xml_test_environment.cpp b/src/conformance/framework/xml_test_environment.cpp index 165f09a8..d7064d98 100644 --- a/src/conformance/framework/xml_test_environment.cpp +++ b/src/conformance/framework/xml_test_environment.cpp @@ -182,7 +182,6 @@ namespace Conformance WriteAvailableInstanceExtensions(xml, globalData.availableInstanceExtensions); if (globalData.IsGraphicsPluginRequired()) { - AutoBasicInstance instance(AutoBasicInstance::createSystemId); auto graphicsPlugin = globalData.GetGraphicsPlugin(); if (graphicsPlugin) { // DescribeGraphics may report only minimal info (name) due to not having a running instance, but this is OK for now. diff --git a/src/conformance/gradle.properties b/src/conformance/gradle.properties new file mode 100644 index 00000000..f4e35f07 --- /dev/null +++ b/src/conformance/gradle.properties @@ -0,0 +1,10 @@ +# Copyright (c) 2020-2023 The Khronos Group Inc. +# +# SPDX-License-Identifier: Apache-2.0 + +org.gradle.jvmargs=-Xmx2048m +android.useAndroidX=true +android.enableJetifier=false +android.defaults.buildfeatures.buildconfig=false +android.nonFinalResIds=true +android.nonTransitiveRClass=true diff --git a/src/conformance/gradle/wrapper/gradle-wrapper.jar b/src/conformance/gradle/wrapper/gradle-wrapper.jar index 249e5832f090a2944b7473328c07c9755baa3196..ccebba7710deaf9f98673a68957ea02138b60d0a 100644 GIT binary patch delta 39304 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 O;LQ#O9OtPYE&l^NyjRr# delta 38549 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 ltSba#rbH)&uY59oK&k!`i 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 diff --git a/src/conformance/gradle/wrapper/gradle-wrapper.properties b/src/conformance/gradle/wrapper/gradle-wrapper.properties index 8049c684..91dc3417 100644 --- a/src/conformance/gradle/wrapper/gradle-wrapper.properties +++ b/src/conformance/gradle/wrapper/gradle-wrapper.properties @@ -1,5 +1,6 @@ distributionBase=GRADLE_USER_HOME distributionPath=wrapper/dists distributionUrl=https\://services.gradle.org/distributions/gradle-7.5-bin.zip +networkTimeout=10000 zipStoreBase=GRADLE_USER_HOME zipStorePath=wrapper/dists diff --git a/src/conformance/gradlew b/src/conformance/gradlew index a69d9cb6..79a61d42 100755 --- a/src/conformance/gradlew +++ b/src/conformance/gradlew @@ -55,7 +55,7 @@ # Darwin, MinGW, and NonStop. # # (3) This script is generated from the Groovy template -# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# https://github.com/gradle/gradle/blob/HEAD/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt # within the Gradle project. # # You can find Gradle at https://github.com/gradle/gradle/. @@ -80,10 +80,10 @@ do esac done -APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit - -APP_NAME="Gradle" +# This is normally unused +# shellcheck disable=SC2034 APP_BASE_NAME=${0##*/} +APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' @@ -143,12 +143,16 @@ fi if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then case $MAX_FD in #( max*) + # In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 MAX_FD=$( ulimit -H -n ) || warn "Could not query maximum file descriptor limit" esac case $MAX_FD in #( '' | soft) :;; #( *) + # In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked. + # shellcheck disable=SC3045 ulimit -n "$MAX_FD" || warn "Could not set maximum file descriptor limit to $MAX_FD" esac diff --git a/src/conformance/gradlew.bat b/src/conformance/gradlew.bat index f127cfd4..93e3f59f 100644 --- a/src/conformance/gradlew.bat +++ b/src/conformance/gradlew.bat @@ -26,6 +26,7 @@ if "%OS%"=="Windows_NT" setlocal set DIRNAME=%~dp0 if "%DIRNAME%"=="" set DIRNAME=. +@rem This is normally unused set APP_BASE_NAME=%~n0 set APP_HOME=%DIRNAME% diff --git a/src/conformance/platform_specific/AndroidManifest.xml b/src/conformance/platform_specific/AndroidManifest.xml index 66737109..94cb24b1 100644 --- a/src/conformance/platform_specific/AndroidManifest.xml +++ b/src/conformance/platform_specific/AndroidManifest.xml @@ -1,6 +1,6 @@ @@ -56,11 +56,14 @@ + android:theme="@android:style/Theme.Black.NoTitleBar.Fullscreen" + tools:ignore="NonResizeableActivity"> ()` consistently between Windows and WSL, instead of `__uuidof()`. -* `/src/dxguids.cpp`: This cpp file can be used as a replacement for linking against `dxguid.lib` on Windows, and as a convenient translation unit to define GUIDs without multiple definitions for WSL. -* `/test`: Simple CMake/Meson projects for validating the headers can be included in a given environment - -## Use on Windows - -Note that these headers may conflict with the headers from the Windows SDK, depending on include ordering. These headers should be added to the include directory list before the SDK, and should be included before other graphics headers (e.g. `d3d11.h`) from the Windows SDK. Otherwise, the corresponding header from the Windows SDK may be included first, and will define the include guards which prevents these headers from being used. - -## Use on WSL - -Note: WSL support is not intended for general purpose application development. At this time, the only recommended usage is for frameworks wishing to provide hardware acceleration for a Linux graphics/compute API in a WSL2 virtualization environment. - -Note: WSL support is only available for 64-bit binaries. - -The headers in the `/include/wsl` directory provide alternative definitions to macros and typedefs normally found in the Windows SDK. For the most part, they should be straightforward, but there are a couple to call attention to: - -|Type|Reason| -|---|---| -|`LONG`/`ULONG`|On 64-bit Windows, a `long` is 4 bytes, but on Linux it is typically 8 bytes. The D3D12 ABI for WSL uses `long` and therefore these should be 8 bytes.| -|`WCHAR`/`WCSTR`|On Windows, a `wchar_t` is 2 bytes, but on Linux it is typically 4 bytes. The D3D12 ABI for WSL uses the native 4-byte `wchar_t`, to enable applications and the runtime to use the system C library to perform string manipulation.| - -Additionally, APIs taking `HANDLE` (`void*`) for Win32 types should instead use `reinterpret_cast(fd)` for an appropriate type of file descriptor. For `ID3D12Fence::SetEventOnCompletion` this should be an `eventfd`, and for shared resources will be an opaque fd. - -## Ways to consume - -There are various ways to consume the headers in this project: - -* Manually: Just copy the headers somewhere and point your project at them. -* CMake subproject: Add this entire project as a subdirectory of your larger project, e.g. as a git submodule, and `add_subdirectory` into it. Use the resulting `DirectX-Headers` and/or `DirectX-Guids` targets as a link dependency -* Installed CMake: After building/installing this project, it can be found through CMake's `find_package` functionality and will expose the same `DirectX-Headers` and `DirectX-Guids` targets. -* FetchContent CMake (3.11+): Fetch this library using Git and easily add it to your project. -* Meson subproject/wrap: Add this entire project as a subproject of your larger project, and use `subproject` or `dependency` to consume it. -* Pkg-config: Use Meson to build this project, and the resulting installed package can be found via pkg-config. -* vcpkg: A vcpkg port has [been added](https://github.com/microsoft/vcpkg/pull/15222). - -Contributions for new mechanisms are welcome. - -## Contributing - -This project welcomes contributions and suggestions. Most contributions require you to agree to a Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us the rights to use your contribution. For details, visit https://cla.opensource.microsoft.com. - -When you submit a pull request, a CLA bot will automatically determine whether you need to provide a CLA and decorate the PR appropriately (e.g., status check, comment). Simply follow the instructions provided by the bot. You will only need to do this once across all repos using our CLA. - -This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments. - -## Trademarks - -This project may contain trademarks or logos for projects, products, or services. Authorized use of Microsoft trademarks or logos is subject to and must follow [Microsoft's Trademark & Brand Guidelines](https://www.microsoft.com/en-us/legal/intellectualproperty/trademarks/usage/general). Use of Microsoft trademarks or logos in modified versions of this project must not cause confusion or imply Microsoft sponsorship. Any use of third-party trademarks or logos are subject to those third-party's policies. \ No newline at end of file diff --git a/src/external/d3dx12/d3dx12.h b/src/external/d3dx12/d3dx12.h index b1355989..562fb53a 100644 --- a/src/external/d3dx12/d3dx12.h +++ b/src/external/d3dx12/d3dx12.h @@ -217,6 +217,8 @@ struct CD3DX12_DEPTH_STENCIL_DESC : public D3D12_DEPTH_STENCIL_DESC }; //------------------------------------------------------------------------------------------------ +// Requires the Windows 10 Creators Update SDK (15063) +#if defined(NTDDI_WIN10_RS2) && (NTDDI_VERSION >= NTDDI_WIN10_RS2) struct CD3DX12_DEPTH_STENCIL_DESC1 : public D3D12_DEPTH_STENCIL_DESC1 { CD3DX12_DEPTH_STENCIL_DESC1() = default; @@ -308,6 +310,7 @@ struct CD3DX12_DEPTH_STENCIL_DESC1 : public D3D12_DEPTH_STENCIL_DESC1 return D; } }; +#endif // NTDDI_WIN10_RS2 //------------------------------------------------------------------------------------------------ struct CD3DX12_BLEND_DESC : public D3D12_BLEND_DESC @@ -572,6 +575,7 @@ struct CD3DX12_RANGE : public D3D12_RANGE }; //------------------------------------------------------------------------------------------------ +#if defined(NTDDI_WIN10_RS2) && (NTDDI_VERSION >= NTDDI_WIN10_RS2) struct CD3DX12_RANGE_UINT64 : public D3D12_RANGE_UINT64 { CD3DX12_RANGE_UINT64() = default; @@ -611,6 +615,7 @@ struct CD3DX12_SUBRESOURCE_RANGE_UINT64 : public D3D12_SUBRESOURCE_RANGE_UINT64 Range.End = end; } }; +#endif // NTDDI_WIN10_RS2 //------------------------------------------------------------------------------------------------ struct CD3DX12_SHADER_BYTECODE : public D3D12_SHADER_BYTECODE @@ -1823,6 +1828,8 @@ inline bool operator!=( const D3D12_RESOURCE_DESC& l, const D3D12_RESOURCE_DESC& { return !( l == r ); } //------------------------------------------------------------------------------------------------ +// Requires the Windows 10 SDK (19041) +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) struct CD3DX12_RESOURCE_DESC1 : public D3D12_RESOURCE_DESC1 { CD3DX12_RESOURCE_DESC1() = default; @@ -1949,8 +1956,11 @@ inline bool operator==( const D3D12_RESOURCE_DESC1& l, const D3D12_RESOURCE_DESC } inline bool operator!=( const D3D12_RESOURCE_DESC1& l, const D3D12_RESOURCE_DESC1& r ) noexcept { return !( l == r ); } +#endif // NTDDI_WIN10_VB //------------------------------------------------------------------------------------------------ +// Requires the Windows 10 Fall Creators Update SDK (16299) +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) struct CD3DX12_VIEW_INSTANCING_DESC : public D3D12_VIEW_INSTANCING_DESC { CD3DX12_VIEW_INSTANCING_DESC() = default; @@ -1973,6 +1983,7 @@ struct CD3DX12_VIEW_INSTANCING_DESC : public D3D12_VIEW_INSTANCING_DESC Flags = InFlags; } }; +#endif // NTDDI_WIN10_RS3 //------------------------------------------------------------------------------------------------ // Row-by-row memcpy @@ -2419,6 +2430,7 @@ inline HRESULT D3DX12SerializeVersionedRootSignature( } //------------------------------------------------------------------------------------------------ +#if defined(NTDDI_WIN10_RS2) && (NTDDI_VERSION >= NTDDI_WIN10_RS2) struct CD3DX12_RT_FORMAT_ARRAY : public D3D12_RT_FORMAT_ARRAY { CD3DX12_RT_FORMAT_ARRAY() = default; @@ -2473,8 +2485,10 @@ typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_STREAM_OUTPUT_DESC, typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_HS> CD3DX12_PIPELINE_STATE_STREAM_HS; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DS> CD3DX12_PIPELINE_STATE_STREAM_DS; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_PS> CD3DX12_PIPELINE_STATE_STREAM_PS; +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_AS> CD3DX12_PIPELINE_STATE_STREAM_AS; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_MS> CD3DX12_PIPELINE_STATE_STREAM_MS; +#endif typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_SHADER_BYTECODE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CS> CD3DX12_PIPELINE_STATE_STREAM_CS; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_BLEND_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_BLEND, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_BLEND_DESC; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_DEPTH_STENCIL_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_DEPTH_STENCIL, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_DEPTH_STENCIL; @@ -2485,7 +2499,9 @@ typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_RT_FORMAT_ARRAY, typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< DXGI_SAMPLE_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_DESC, DefaultSampleDesc> CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_DESC; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< UINT, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_SAMPLE_MASK, DefaultSampleMask> CD3DX12_PIPELINE_STATE_STREAM_SAMPLE_MASK; typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< D3D12_CACHED_PIPELINE_STATE, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_CACHED_PSO> CD3DX12_PIPELINE_STATE_STREAM_CACHED_PSO; +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) typedef CD3DX12_PIPELINE_STATE_STREAM_SUBOBJECT< CD3DX12_VIEW_INSTANCING_DESC, D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VIEW_INSTANCING, CD3DX12_DEFAULT> CD3DX12_PIPELINE_STATE_STREAM_VIEW_INSTANCING; +#endif //------------------------------------------------------------------------------------------------ // Stream Parser Helpers @@ -2506,8 +2522,10 @@ struct ID3DX12PipelineParserCallbacks virtual void DSCb(const D3D12_SHADER_BYTECODE&) {} virtual void PSCb(const D3D12_SHADER_BYTECODE&) {} virtual void CSCb(const D3D12_SHADER_BYTECODE&) {} +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) virtual void ASCb(const D3D12_SHADER_BYTECODE&) {} virtual void MSCb(const D3D12_SHADER_BYTECODE&) {} +#endif virtual void BlendStateCb(const D3D12_BLEND_DESC&) {} virtual void DepthStencilStateCb(const D3D12_DEPTH_STENCIL_DESC&) {} virtual void DepthStencilState1Cb(const D3D12_DEPTH_STENCIL_DESC1&) {} @@ -2516,7 +2534,9 @@ struct ID3DX12PipelineParserCallbacks virtual void RTVFormatsCb(const D3D12_RT_FORMAT_ARRAY&) {} virtual void SampleDescCb(const DXGI_SAMPLE_DESC&) {} virtual void SampleMaskCb(UINT) {} +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) virtual void ViewInstancingCb(const D3D12_VIEW_INSTANCING_DESC&) {} +#endif virtual void CachedPSOCb(const D3D12_CACHED_PIPELINE_STATE&) {} // Error Callbacks @@ -2527,6 +2547,7 @@ struct ID3DX12PipelineParserCallbacks virtual ~ID3DX12PipelineParserCallbacks() = default; }; +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) struct D3DX12_MESH_SHADER_PIPELINE_STATE_DESC { ID3D12RootSignature* pRootSignature; @@ -2665,7 +2686,9 @@ struct CD3DX12_PIPELINE_STATE_STREAM2 return D; } }; +#endif // NTDDI_WIN10_VB +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) // CD3DX12_PIPELINE_STATE_STREAM1 Works on OS Build 16299+ (where there is a new view instancing subobject). // Use CD3DX12_PIPELINE_STATE_STREAM for OS Build 15063+ support. struct CD3DX12_PIPELINE_STATE_STREAM1 @@ -2695,6 +2718,7 @@ struct CD3DX12_PIPELINE_STATE_STREAM1 , CachedPSO(Desc.CachedPSO) , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) {} +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) CD3DX12_PIPELINE_STATE_STREAM1(const D3DX12_MESH_SHADER_PIPELINE_STATE_DESC& Desc) noexcept : Flags(Desc.Flags) , NodeMask(Desc.NodeMask) @@ -2711,6 +2735,7 @@ struct CD3DX12_PIPELINE_STATE_STREAM1 , CachedPSO(Desc.CachedPSO) , ViewInstancingDesc(CD3DX12_VIEW_INSTANCING_DESC(CD3DX12_DEFAULT())) {} +#endif CD3DX12_PIPELINE_STATE_STREAM1(const D3D12_COMPUTE_PIPELINE_STATE_DESC& Desc) noexcept : Flags(Desc.Flags) , NodeMask(Desc.NodeMask) @@ -2779,8 +2804,9 @@ struct CD3DX12_PIPELINE_STATE_STREAM1 return D; } }; +#endif // NTDDI_WIN10_RS3 - +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) struct CD3DX12_PIPELINE_MESH_STATE_STREAM { CD3DX12_PIPELINE_MESH_STATE_STREAM() = default; @@ -2837,6 +2863,7 @@ struct CD3DX12_PIPELINE_MESH_STATE_STREAM return D; } }; +#endif // NTDDI_WIN10_VB // CD3DX12_PIPELINE_STATE_STREAM works on OS Build 15063+ but does not support new subobject(s) added in OS Build 16299+. // See CD3DX12_PIPELINE_STATE_STREAM1 for instance. @@ -2931,6 +2958,7 @@ struct CD3DX12_PIPELINE_STATE_STREAM } }; +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) struct CD3DX12_PIPELINE_STATE_STREAM2_PARSE_HELPER : public ID3DX12PipelineParserCallbacks { CD3DX12_PIPELINE_STATE_STREAM2 PipelineStream; @@ -2990,11 +3018,15 @@ struct CD3DX12_PIPELINE_STATE_STREAM2_PARSE_HELPER : public ID3DX12PipelineParse private: bool SeenDSS; }; - +#endif // NTDDI_WIN10_VB struct CD3DX12_PIPELINE_STATE_STREAM_PARSE_HELPER : public ID3DX12PipelineParserCallbacks { +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) CD3DX12_PIPELINE_STATE_STREAM1 PipelineStream; +#else + CD3DX12_PIPELINE_STATE_STREAM PipelineStream; +#endif CD3DX12_PIPELINE_STATE_STREAM_PARSE_HELPER() noexcept : SeenDSS(false) { @@ -3043,7 +3075,9 @@ struct CD3DX12_PIPELINE_STATE_STREAM_PARSE_HELPER : public ID3DX12PipelineParser void RTVFormatsCb(const D3D12_RT_FORMAT_ARRAY& RTVFormats) override {PipelineStream.RTVFormats = RTVFormats;} void SampleDescCb(const DXGI_SAMPLE_DESC& SampleDesc) override {PipelineStream.SampleDesc = SampleDesc;} void SampleMaskCb(UINT SampleMask) override {PipelineStream.SampleMask = SampleMask;} +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) void ViewInstancingCb(const D3D12_VIEW_INSTANCING_DESC& ViewInstancingDesc) override {PipelineStream.ViewInstancingDesc = CD3DX12_VIEW_INSTANCING_DESC(ViewInstancingDesc);} +#endif void CachedPSOCb(const D3D12_CACHED_PIPELINE_STATE& CachedPSO) override {PipelineStream.CachedPSO = CachedPSO;} private: @@ -3120,6 +3154,7 @@ inline HRESULT D3DX12ParsePipelineStream(const D3D12_PIPELINE_STATE_STREAM_DESC& pCallbacks->CSCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::CS); break; +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_AS: pCallbacks->ASCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM2::AS); @@ -3128,6 +3163,7 @@ inline HRESULT D3DX12ParsePipelineStream(const D3D12_PIPELINE_STATE_STREAM_DESC& pCallbacks->MSCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM2::MS); break; +#endif case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_STREAM_OUTPUT: pCallbacks->StreamOutputCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::StreamOutput); @@ -3188,10 +3224,12 @@ inline HRESULT D3DX12ParsePipelineStream(const D3D12_PIPELINE_STATE_STREAM_DESC& pCallbacks->FlagsCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM::Flags); break; +#if defined(NTDDI_WIN10_RS3) && (NTDDI_VERSION >= NTDDI_WIN10_RS3) case D3D12_PIPELINE_STATE_SUBOBJECT_TYPE_VIEW_INSTANCING: pCallbacks->ViewInstancingCb(*reinterpret_cast(pStream)); SizeOfSubobject = sizeof(CD3DX12_PIPELINE_STATE_STREAM1::ViewInstancingDesc); break; +#endif default: pCallbacks->ErrorUnknownSubobject(SubobjectType); return E_INVALIDARG; @@ -3200,7 +3238,10 @@ inline HRESULT D3DX12ParsePipelineStream(const D3D12_PIPELINE_STATE_STREAM_DESC& return S_OK; } +#endif // NTDDI_WIN10_RS2 +// Requires the Windows 10 October 2018 Update SDK (17763) +#if defined(NTDDI_WIN10_RS5) && (NTDDI_VERSION >= NTDDI_WIN10_RS5) //------------------------------------------------------------------------------------------------ inline bool operator==( const D3D12_CLEAR_VALUE &a, const D3D12_CLEAR_VALUE &b) noexcept { @@ -3282,17 +3323,7 @@ inline bool operator==( const D3D12_RENDER_PASS_DEPTH_STENCIL_DESC &a, const D3D #include #include #include -#ifndef D3DX12_USE_ATL #include -#define D3DX12_COM_PTR Microsoft::WRL::ComPtr -#define D3DX12_COM_PTR_GET(x) x.Get() -#define D3DX12_COM_PTR_ADDRESSOF(x) x.GetAddressOf() -#else -#include -#define D3DX12_COM_PTR ATL::CComPtr -#define D3DX12_COM_PTR_GET(x) x.p -#define D3DX12_COM_PTR_ADDRESSOF(x) &x.p -#endif //------------------------------------------------------------------------------------------------ class CD3DX12_STATE_OBJECT_DESC @@ -3480,7 +3511,9 @@ class CD3DX12_STATE_OBJECT_DESC friend class CD3DX12_HIT_GROUP_SUBOBJECT; friend class CD3DX12_RAYTRACING_SHADER_CONFIG_SUBOBJECT; friend class CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT; +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) friend class CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT; +#endif friend class CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT; friend class CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT; friend class CD3DX12_STATE_OBJECT_CONFIG_SUBOBJECT; @@ -3618,7 +3651,7 @@ class CD3DX12_EXISTING_COLLECTION_SUBOBJECT } void* Data() noexcept override { return &m_Desc; } D3D12_EXISTING_COLLECTION_DESC m_Desc; - D3DX12_COM_PTR m_CollectionRef; + Microsoft::WRL::ComPtr m_CollectionRef; CD3DX12_STATE_OBJECT_DESC::StringContainer m_Strings; std::vector m_Exports; }; @@ -3867,6 +3900,7 @@ class CD3DX12_RAYTRACING_PIPELINE_CONFIG_SUBOBJECT }; //------------------------------------------------------------------------------------------------ +#if defined(NTDDI_WIN10_VB) && (NTDDI_VERSION >= NTDDI_WIN10_VB) class CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT : public CD3DX12_STATE_OBJECT_DESC::SUBOBJECT_HELPER_BASE { @@ -3900,6 +3934,7 @@ class CD3DX12_RAYTRACING_PIPELINE_CONFIG1_SUBOBJECT void* Data() noexcept override { return &m_Desc; } D3D12_RAYTRACING_PIPELINE_CONFIG1 m_Desc; }; +#endif // NTDDI_WIN10_VB //------------------------------------------------------------------------------------------------ class CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT @@ -3924,15 +3959,15 @@ class CD3DX12_GLOBAL_ROOT_SIGNATURE_SUBOBJECT return D3D12_STATE_SUBOBJECT_TYPE_GLOBAL_ROOT_SIGNATURE; } operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } - operator ID3D12RootSignature*() const noexcept { return D3DX12_COM_PTR_GET(m_pRootSig); } + operator ID3D12RootSignature*() const noexcept { return m_pRootSig.Get(); } private: void Init() noexcept { SUBOBJECT_HELPER_BASE::Init(); m_pRootSig = nullptr; } - void* Data() noexcept override { return D3DX12_COM_PTR_ADDRESSOF(m_pRootSig); } - D3DX12_COM_PTR m_pRootSig; + void* Data() noexcept override { return m_pRootSig.GetAddressOf(); } + Microsoft::WRL::ComPtr m_pRootSig; }; //------------------------------------------------------------------------------------------------ @@ -3958,15 +3993,15 @@ class CD3DX12_LOCAL_ROOT_SIGNATURE_SUBOBJECT return D3D12_STATE_SUBOBJECT_TYPE_LOCAL_ROOT_SIGNATURE; } operator const D3D12_STATE_SUBOBJECT&() const noexcept { return *m_pSubobject; } - operator ID3D12RootSignature*() const noexcept { return D3DX12_COM_PTR_GET(m_pRootSig); } + operator ID3D12RootSignature*() const noexcept { return m_pRootSig.Get(); } private: void Init() noexcept { SUBOBJECT_HELPER_BASE::Init(); m_pRootSig = nullptr; } - void* Data() noexcept override { return D3DX12_COM_PTR_ADDRESSOF(m_pRootSig); } - D3DX12_COM_PTR m_pRootSig; + void* Data() noexcept override { return m_pRootSig.GetAddressOf(); } + Microsoft::WRL::ComPtr m_pRootSig; }; //------------------------------------------------------------------------------------------------ @@ -4037,13 +4072,9 @@ class CD3DX12_NODE_MASK_SUBOBJECT D3D12_NODE_MASK m_Desc; }; -#undef D3DX12_COM_PTR -#undef D3DX12_COM_PTR_GET -#undef D3DX12_COM_PTR_ADDRESSOF #endif // #ifndef D3DX12_NO_STATE_OBJECT_HELPERS +#endif // NTDDI_WIN10_RS5 #endif // defined( __cplusplus ) #endif //__D3DX12_H__ - -