openxr_frame_synthesis_extension.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407
  1. /**************************************************************************/
  2. /* openxr_frame_synthesis_extension.cpp */
  3. /**************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /**************************************************************************/
  8. /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
  9. /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /**************************************************************************/
  30. #include "openxr_frame_synthesis_extension.h"
  31. #include "core/config/project_settings.h"
  32. #include "servers/rendering/rendering_server.h"
  33. #include "servers/xr/xr_server.h"
  34. #define GL_RGBA16F 0x881A
  35. #define GL_DEPTH24_STENCIL8 0x88F0
  36. #define VK_FORMAT_R16G16B16A16_SFLOAT 97
  37. #define VK_FORMAT_D24_UNORM_S8_UINT 129
  38. OpenXRFrameSynthesisExtension *OpenXRFrameSynthesisExtension::singleton = nullptr;
  39. OpenXRFrameSynthesisExtension *OpenXRFrameSynthesisExtension::get_singleton() {
  40. return singleton;
  41. }
  42. void OpenXRFrameSynthesisExtension::_bind_methods() {
  43. ClassDB::bind_method(D_METHOD("is_available"), &OpenXRFrameSynthesisExtension::is_available);
  44. ClassDB::bind_method(D_METHOD("is_enabled"), &OpenXRFrameSynthesisExtension::is_enabled);
  45. ClassDB::bind_method(D_METHOD("set_enabled", "enable"), &OpenXRFrameSynthesisExtension::set_enabled);
  46. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "enabled"), "set_enabled", "is_enabled");
  47. ClassDB::bind_method(D_METHOD("get_relax_frame_interval"), &OpenXRFrameSynthesisExtension::get_relax_frame_interval);
  48. ClassDB::bind_method(D_METHOD("set_relax_frame_interval", "relax_frame_interval"), &OpenXRFrameSynthesisExtension::set_relax_frame_interval);
  49. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "relax_frame_interval"), "set_relax_frame_interval", "get_relax_frame_interval");
  50. ClassDB::bind_method(D_METHOD("skip_next_frame"), &OpenXRFrameSynthesisExtension::skip_next_frame);
  51. }
  52. OpenXRFrameSynthesisExtension::OpenXRFrameSynthesisExtension() {
  53. singleton = this;
  54. }
  55. OpenXRFrameSynthesisExtension::~OpenXRFrameSynthesisExtension() {
  56. singleton = nullptr;
  57. }
  58. HashMap<String, bool *> OpenXRFrameSynthesisExtension::get_requested_extensions(XrVersion p_version) {
  59. HashMap<String, bool *> request_extensions;
  60. if (GLOBAL_GET("xr/openxr/extensions/frame_synthesis")) {
  61. request_extensions[XR_EXT_FRAME_SYNTHESIS_EXTENSION_NAME] = &frame_synthesis_ext;
  62. }
  63. return request_extensions;
  64. }
  65. void OpenXRFrameSynthesisExtension::on_instance_created(const XrInstance p_instance) {
  66. // Enable this if our extension was successfully enabled
  67. enabled = frame_synthesis_ext;
  68. render_state.enabled = frame_synthesis_ext;
  69. // Register this as a projection view extension
  70. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  71. ERR_FAIL_NULL(openxr_api);
  72. openxr_api->register_projection_views_extension(this);
  73. }
  74. void OpenXRFrameSynthesisExtension::on_instance_destroyed() {
  75. frame_synthesis_ext = false;
  76. enabled = false;
  77. render_state.enabled = false;
  78. // Unregister this as a projection view extension.
  79. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  80. ERR_FAIL_NULL(openxr_api);
  81. openxr_api->unregister_projection_views_extension(this);
  82. }
  83. void OpenXRFrameSynthesisExtension::prepare_view_configuration(uint32_t p_view_count) {
  84. if (!frame_synthesis_ext) {
  85. return;
  86. }
  87. // Called during initialization, we can safely change this.
  88. render_state.config_views.resize(p_view_count);
  89. for (XrFrameSynthesisConfigViewEXT &config_view : render_state.config_views) {
  90. config_view.type = XR_TYPE_FRAME_SYNTHESIS_CONFIG_VIEW_EXT;
  91. config_view.next = nullptr;
  92. // These will be set by xrEnumerateViewConfigurationViews.
  93. config_view.recommendedMotionVectorImageRectWidth = 0;
  94. config_view.recommendedMotionVectorImageRectHeight = 0;
  95. }
  96. }
  97. void *OpenXRFrameSynthesisExtension::set_view_configuration_and_get_next_pointer(uint32_t p_view, void *p_next_pointer) {
  98. if (!frame_synthesis_ext) {
  99. return nullptr;
  100. }
  101. // Called during initialization, we can safely access this.
  102. ERR_FAIL_UNSIGNED_INDEX_V(p_view, render_state.config_views.size(), nullptr);
  103. XrFrameSynthesisConfigViewEXT &config_view = render_state.config_views[p_view];
  104. config_view.next = p_next_pointer;
  105. return &config_view;
  106. }
  107. void OpenXRFrameSynthesisExtension::print_view_configuration_info(uint32_t p_view) const {
  108. if (!frame_synthesis_ext) {
  109. return;
  110. }
  111. // Called during initialization, we can safely access this.
  112. if (p_view < render_state.config_views.size()) {
  113. const XrFrameSynthesisConfigViewEXT &config_view = render_state.config_views[p_view];
  114. print_line(" - motion vector width: ", itos(config_view.recommendedMotionVectorImageRectWidth));
  115. print_line(" - motion vector height: ", itos(config_view.recommendedMotionVectorImageRectHeight));
  116. }
  117. }
  118. void OpenXRFrameSynthesisExtension::on_session_destroyed() {
  119. if (!frame_synthesis_ext) {
  120. return;
  121. }
  122. // Free our swapchains.
  123. free_swapchains();
  124. }
  125. void OpenXRFrameSynthesisExtension::on_main_swapchains_created() {
  126. if (!frame_synthesis_ext) {
  127. return;
  128. }
  129. // It is possible that our swapchain information gets resized,
  130. // and that our motion vector and depth resolution changes with this.
  131. // So (re)create our swapchains here as well.
  132. // Note that we do this even if motion vectors aren't enabled yet.
  133. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  134. ERR_FAIL_NULL(openxr_api);
  135. RenderingServer *rendering_server = RenderingServer::get_singleton();
  136. ERR_FAIL_NULL(rendering_server);
  137. // Out with the old.
  138. free_swapchains();
  139. // We only support stereo.
  140. size_t view_count = render_state.config_views.size();
  141. ERR_FAIL_COND(view_count != 2);
  142. // Determine specific values for each renderer.
  143. int swapchain_format = 0;
  144. int depth_swapchain_format = 0;
  145. String rendering_driver_name = rendering_server->get_current_rendering_driver_name();
  146. if (rendering_driver_name.contains("opengl")) {
  147. swapchain_format = GL_RGBA16F;
  148. depth_swapchain_format = GL_DEPTH24_STENCIL8;
  149. } else if (rendering_driver_name == "vulkan") {
  150. String rendering_method = rendering_server->get_current_rendering_method();
  151. if (rendering_method == "mobile") {
  152. swapchain_format = VK_FORMAT_R16G16B16A16_SFLOAT;
  153. depth_swapchain_format = VK_FORMAT_D24_UNORM_S8_UINT;
  154. } else {
  155. WARN_PRINT("OpenXR: Frame synthesis not supported for this rendering method!");
  156. frame_synthesis_ext = false;
  157. return;
  158. }
  159. } else {
  160. WARN_PRINT("OpenXR: Frame synthesis not supported for this rendering driver!");
  161. frame_synthesis_ext = false;
  162. return;
  163. }
  164. // We assume the size for each eye is the same, it should be.
  165. uint32_t width = render_state.config_views[0].recommendedMotionVectorImageRectWidth;
  166. uint32_t height = render_state.config_views[0].recommendedMotionVectorImageRectHeight;
  167. // Create swapchains for motion vectors and depth.
  168. render_state.swapchains[SWAPCHAIN_MOTION_VECTOR].create(0, XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_COLOR_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, swapchain_format, width, height, 1, view_count);
  169. render_state.swapchains[SWAPCHAIN_DEPTH].create(0, XR_SWAPCHAIN_USAGE_SAMPLED_BIT | XR_SWAPCHAIN_USAGE_DEPTH_STENCIL_ATTACHMENT_BIT | XR_SWAPCHAIN_USAGE_MUTABLE_FORMAT_BIT, depth_swapchain_format, width, height, 1, view_count);
  170. // Set up our frame synthesis info.
  171. render_state.frame_synthesis_info.resize(view_count);
  172. uint32_t index = 0;
  173. for (XrFrameSynthesisInfoEXT &frame_synthesis_info : render_state.frame_synthesis_info) {
  174. frame_synthesis_info.type = XR_TYPE_FRAME_SYNTHESIS_INFO_EXT;
  175. frame_synthesis_info.next = nullptr;
  176. frame_synthesis_info.layerFlags = 0;
  177. // Set up motion vector.
  178. frame_synthesis_info.motionVectorSubImage.swapchain = render_state.swapchains[SWAPCHAIN_MOTION_VECTOR].get_swapchain();
  179. frame_synthesis_info.motionVectorSubImage.imageArrayIndex = index;
  180. frame_synthesis_info.motionVectorSubImage.imageRect.offset.x = 0;
  181. frame_synthesis_info.motionVectorSubImage.imageRect.offset.y = 0;
  182. frame_synthesis_info.motionVectorSubImage.imageRect.extent.width = width;
  183. frame_synthesis_info.motionVectorSubImage.imageRect.extent.height = height;
  184. // Q: this should be 1.0, -1.0, 1.0. We output OpenGL NDC, frame synthesis expects Vulkan NDC, but might be a problem on runtime I'm testing.
  185. frame_synthesis_info.motionVectorScale = { 1.0, 1.0, 1.0, 0.0 };
  186. frame_synthesis_info.motionVectorOffset = { 0.0, 0.0, 0.0, 0.0 };
  187. frame_synthesis_info.appSpaceDeltaPose = { { 0.0, 0.0, 0.0, 1.0 }, { 0.0, 0.0, 0.0 } };
  188. // Set up depth image.
  189. frame_synthesis_info.depthSubImage.swapchain = render_state.swapchains[SWAPCHAIN_DEPTH].get_swapchain();
  190. frame_synthesis_info.depthSubImage.imageArrayIndex = index;
  191. frame_synthesis_info.depthSubImage.imageRect.offset.x = 0;
  192. frame_synthesis_info.depthSubImage.imageRect.offset.y = 0;
  193. frame_synthesis_info.depthSubImage.imageRect.extent.width = width;
  194. frame_synthesis_info.depthSubImage.imageRect.extent.height = height;
  195. frame_synthesis_info.minDepth = 0.0;
  196. frame_synthesis_info.maxDepth = 1.0;
  197. // Note: reverse-Z, these are just defaults for now.
  198. frame_synthesis_info.nearZ = 100.0;
  199. frame_synthesis_info.farZ = 0.01;
  200. index++;
  201. }
  202. }
  203. void OpenXRFrameSynthesisExtension::on_pre_render() {
  204. if (!frame_synthesis_ext) {
  205. return;
  206. }
  207. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  208. ERR_FAIL_NULL(openxr_api);
  209. size_t view_count = render_state.config_views.size();
  210. if (!enabled || view_count != 2 || render_state.skip_next_frame) {
  211. // Unset these just in case.
  212. openxr_api->set_velocity_texture(RID());
  213. openxr_api->set_velocity_depth_texture(RID());
  214. // Remember our transform just in case we (re)start frame synthesis later on.
  215. render_state.previous_transform = XRServer::get_singleton()->get_world_origin();
  216. return;
  217. }
  218. // Acquire our swapchains.
  219. for (int i = 0; i < SWAPCHAIN_MAX; i++) {
  220. bool should_render = true;
  221. render_state.swapchains[i].acquire(should_render);
  222. }
  223. // Set our images.
  224. openxr_api->set_velocity_texture(render_state.swapchains[SWAPCHAIN_MOTION_VECTOR].get_image());
  225. openxr_api->set_velocity_depth_texture(render_state.swapchains[SWAPCHAIN_DEPTH].get_image());
  226. // Set our size.
  227. uint32_t width = render_state.config_views[0].recommendedMotionVectorImageRectWidth;
  228. uint32_t height = render_state.config_views[0].recommendedMotionVectorImageRectHeight;
  229. openxr_api->set_velocity_target_size(Size2i(width, height));
  230. // Get our head motion
  231. Transform3D world_transform = XRServer::get_singleton()->get_world_origin();
  232. Transform3D delta_transform = render_state.previous_transform.affine_inverse() * world_transform;
  233. Quaternion delta_quat = delta_transform.basis.get_quaternion();
  234. Vector3 delta_origin = delta_transform.origin;
  235. // Z near/far can change per frame, so make sure we update this.
  236. for (XrFrameSynthesisInfoEXT &frame_synthesis_info : render_state.frame_synthesis_info) {
  237. frame_synthesis_info.layerFlags = render_state.relax_frame_interval ? XR_FRAME_SYNTHESIS_INFO_REQUEST_RELAXED_FRAME_INTERVAL_BIT_EXT : 0;
  238. frame_synthesis_info.appSpaceDeltaPose = {
  239. { (float)delta_quat.x, (float)delta_quat.y, (float)delta_quat.z, (float)delta_quat.w },
  240. { (float)delta_origin.x, (float)delta_origin.y, (float)delta_origin.z }
  241. };
  242. // Note: reverse-Z.
  243. frame_synthesis_info.nearZ = openxr_api->get_render_state_z_far();
  244. frame_synthesis_info.farZ = openxr_api->get_render_state_z_near();
  245. }
  246. // Remember our transform.
  247. render_state.previous_transform = world_transform;
  248. }
  249. void OpenXRFrameSynthesisExtension::on_post_draw_viewport(RID p_render_target) {
  250. // Check if our extension is supported and enabled.
  251. if (!frame_synthesis_ext || !enabled || render_state.config_views.size() != 2 || render_state.skip_next_frame) {
  252. return;
  253. }
  254. // Release our swapchains.
  255. for (int i = 0; i < SWAPCHAIN_MAX; i++) {
  256. render_state.swapchains[i].release();
  257. }
  258. }
  259. void *OpenXRFrameSynthesisExtension::set_projection_views_and_get_next_pointer(int p_view_index, void *p_next_pointer) {
  260. // Check if our extension is supported and enabled.
  261. if (!frame_synthesis_ext || !enabled || render_state.config_views.size() != 2) {
  262. return nullptr;
  263. }
  264. // Did we skip this frame?
  265. if (render_state.skip_next_frame) {
  266. // Only unset when we've handled both eyes.
  267. if (p_view_index == 1) {
  268. render_state.skip_next_frame = false;
  269. }
  270. return nullptr;
  271. }
  272. // Check if we can run frame synthesis.
  273. size_t view_count = render_state.config_views.size();
  274. if (enabled && view_count == 2) {
  275. render_state.frame_synthesis_info[p_view_index].next = p_next_pointer;
  276. return &render_state.frame_synthesis_info[p_view_index];
  277. }
  278. return nullptr;
  279. }
  280. bool OpenXRFrameSynthesisExtension::is_available() const {
  281. return frame_synthesis_ext;
  282. }
  283. bool OpenXRFrameSynthesisExtension::is_enabled() const {
  284. return frame_synthesis_ext && enabled;
  285. }
  286. void OpenXRFrameSynthesisExtension::set_enabled(bool p_enabled) {
  287. if (enabled == p_enabled) {
  288. return;
  289. }
  290. ERR_FAIL_COND(!frame_synthesis_ext && p_enabled);
  291. enabled = p_enabled;
  292. RenderingServer *rendering_server = RenderingServer::get_singleton();
  293. ERR_FAIL_NULL(rendering_server);
  294. rendering_server->call_on_render_thread(callable_mp(this, &OpenXRFrameSynthesisExtension::_set_render_state_enabled_rt).bind(enabled));
  295. }
  296. bool OpenXRFrameSynthesisExtension::get_relax_frame_interval() const {
  297. return relax_frame_interval;
  298. }
  299. void OpenXRFrameSynthesisExtension::set_relax_frame_interval(bool p_relax_frame_interval) {
  300. if (relax_frame_interval == p_relax_frame_interval) {
  301. return;
  302. }
  303. relax_frame_interval = p_relax_frame_interval;
  304. RenderingServer *rendering_server = RenderingServer::get_singleton();
  305. ERR_FAIL_NULL(rendering_server);
  306. rendering_server->call_on_render_thread(callable_mp(this, &OpenXRFrameSynthesisExtension::_set_relax_frame_interval_rt).bind(relax_frame_interval));
  307. }
  308. void OpenXRFrameSynthesisExtension::_set_render_state_enabled_rt(bool p_enabled) {
  309. render_state.enabled = p_enabled;
  310. }
  311. void OpenXRFrameSynthesisExtension::_set_relax_frame_interval_rt(bool p_relax_frame_interval) {
  312. render_state.relax_frame_interval = p_relax_frame_interval;
  313. }
  314. void OpenXRFrameSynthesisExtension::free_swapchains() {
  315. for (int i = 0; i < SWAPCHAIN_MAX; i++) {
  316. render_state.swapchains[i].queue_free();
  317. }
  318. }
  319. void OpenXRFrameSynthesisExtension::skip_next_frame() {
  320. RenderingServer *rendering_server = RenderingServer::get_singleton();
  321. ERR_FAIL_NULL(rendering_server);
  322. rendering_server->call_on_render_thread(callable_mp(this, &OpenXRFrameSynthesisExtension::_set_skip_next_frame_rt));
  323. }
  324. void OpenXRFrameSynthesisExtension::_set_skip_next_frame_rt() {
  325. render_state.skip_next_frame = true;
  326. }