openxr_render_model_extension.cpp 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795
  1. /**************************************************************************/
  2. /* openxr_render_model_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_render_model_extension.h"
  31. #include "../openxr_api.h"
  32. #include "../openxr_interface.h"
  33. #include "core/config/project_settings.h"
  34. #include "core/string/print_string.h"
  35. #include "servers/xr_server.h"
  36. OpenXRRenderModelExtension *OpenXRRenderModelExtension::singleton = nullptr;
  37. OpenXRRenderModelExtension *OpenXRRenderModelExtension::get_singleton() {
  38. return singleton;
  39. }
  40. void OpenXRRenderModelExtension::_bind_methods() {
  41. ClassDB::bind_method(D_METHOD("is_active"), &OpenXRRenderModelExtension::is_active);
  42. ClassDB::bind_method(D_METHOD("render_model_create", "render_model_id"), &OpenXRRenderModelExtension::render_model_create);
  43. ClassDB::bind_method(D_METHOD("render_model_destroy", "render_model"), &OpenXRRenderModelExtension::render_model_destroy);
  44. ClassDB::bind_method(D_METHOD("render_model_get_all"), &OpenXRRenderModelExtension::render_model_get_all);
  45. ClassDB::bind_method(D_METHOD("render_model_new_scene_instance", "render_model"), &OpenXRRenderModelExtension::render_model_new_scene_instance);
  46. ClassDB::bind_method(D_METHOD("render_model_get_subaction_paths", "render_model"), &OpenXRRenderModelExtension::render_model_get_subaction_paths);
  47. ClassDB::bind_method(D_METHOD("render_model_get_top_level_path", "render_model"), &OpenXRRenderModelExtension::render_model_get_top_level_path_as_string);
  48. ClassDB::bind_method(D_METHOD("render_model_get_confidence", "render_model"), &OpenXRRenderModelExtension::render_model_get_confidence);
  49. ClassDB::bind_method(D_METHOD("render_model_get_root_transform", "render_model"), &OpenXRRenderModelExtension::render_model_get_root_transform);
  50. ClassDB::bind_method(D_METHOD("render_model_get_animatable_node_count", "render_model"), &OpenXRRenderModelExtension::render_model_get_animatable_node_count);
  51. ClassDB::bind_method(D_METHOD("render_model_get_animatable_node_name", "render_model", "index"), &OpenXRRenderModelExtension::render_model_get_animatable_node_name);
  52. ClassDB::bind_method(D_METHOD("render_model_is_animatable_node_visible", "render_model", "index"), &OpenXRRenderModelExtension::render_model_is_animatable_node_visible);
  53. ClassDB::bind_method(D_METHOD("render_model_get_animatable_node_transform", "render_model", "index"), &OpenXRRenderModelExtension::render_model_get_animatable_node_transform);
  54. ADD_SIGNAL(MethodInfo("render_model_added", PropertyInfo(Variant::RID, "render_model")));
  55. ADD_SIGNAL(MethodInfo("render_model_removed", PropertyInfo(Variant::RID, "render_model")));
  56. ADD_SIGNAL(MethodInfo("render_model_top_level_path_changed", PropertyInfo(Variant::RID, "render_model")));
  57. }
  58. OpenXRRenderModelExtension::OpenXRRenderModelExtension() {
  59. singleton = this;
  60. }
  61. OpenXRRenderModelExtension::~OpenXRRenderModelExtension() {
  62. singleton = nullptr;
  63. }
  64. HashMap<String, bool *> OpenXRRenderModelExtension::get_requested_extensions() {
  65. HashMap<String, bool *> request_extensions;
  66. if (GLOBAL_GET("xr/openxr/extensions/render_model")) {
  67. request_extensions[XR_EXT_UUID_EXTENSION_NAME] = &uuid_ext;
  68. request_extensions[XR_EXT_RENDER_MODEL_EXTENSION_NAME] = &render_model_ext;
  69. request_extensions[XR_EXT_INTERACTION_RENDER_MODEL_EXTENSION_NAME] = &interaction_render_model_ext;
  70. }
  71. return request_extensions;
  72. }
  73. void OpenXRRenderModelExtension::on_instance_created(const XrInstance p_instance) {
  74. // Standard entry points we use.
  75. EXT_INIT_XR_FUNC(xrLocateSpace);
  76. EXT_INIT_XR_FUNC(xrDestroySpace);
  77. EXT_INIT_XR_FUNC(xrPathToString);
  78. if (render_model_ext) {
  79. EXT_INIT_XR_FUNC(xrCreateRenderModelEXT);
  80. EXT_INIT_XR_FUNC(xrDestroyRenderModelEXT);
  81. EXT_INIT_XR_FUNC(xrGetRenderModelPropertiesEXT);
  82. EXT_INIT_XR_FUNC(xrCreateRenderModelSpaceEXT);
  83. EXT_INIT_XR_FUNC(xrCreateRenderModelAssetEXT);
  84. EXT_INIT_XR_FUNC(xrDestroyRenderModelAssetEXT);
  85. EXT_INIT_XR_FUNC(xrGetRenderModelAssetDataEXT);
  86. EXT_INIT_XR_FUNC(xrGetRenderModelAssetPropertiesEXT);
  87. EXT_INIT_XR_FUNC(xrGetRenderModelStateEXT);
  88. }
  89. if (interaction_render_model_ext) {
  90. EXT_INIT_XR_FUNC(xrEnumerateInteractionRenderModelIdsEXT);
  91. EXT_INIT_XR_FUNC(xrEnumerateRenderModelSubactionPathsEXT);
  92. EXT_INIT_XR_FUNC(xrGetRenderModelPoseTopLevelUserPathEXT);
  93. }
  94. }
  95. void OpenXRRenderModelExtension::on_session_created(const XrSession p_session) {
  96. _interaction_data_dirty = true;
  97. }
  98. void OpenXRRenderModelExtension::on_instance_destroyed() {
  99. xrCreateRenderModelEXT_ptr = nullptr;
  100. xrDestroyRenderModelEXT_ptr = nullptr;
  101. xrGetRenderModelPropertiesEXT_ptr = nullptr;
  102. xrCreateRenderModelSpaceEXT_ptr = nullptr;
  103. xrCreateRenderModelAssetEXT_ptr = nullptr;
  104. xrDestroyRenderModelAssetEXT_ptr = nullptr;
  105. xrGetRenderModelAssetDataEXT_ptr = nullptr;
  106. xrGetRenderModelAssetPropertiesEXT_ptr = nullptr;
  107. xrGetRenderModelStateEXT_ptr = nullptr;
  108. xrEnumerateInteractionRenderModelIdsEXT_ptr = nullptr;
  109. xrEnumerateRenderModelSubactionPathsEXT_ptr = nullptr;
  110. xrGetRenderModelPoseTopLevelUserPathEXT_ptr = nullptr;
  111. uuid_ext = false;
  112. render_model_ext = false;
  113. interaction_render_model_ext = false;
  114. }
  115. void OpenXRRenderModelExtension::on_session_destroyed() {
  116. _clear_interaction_data();
  117. _clear_render_model_data();
  118. // We no longer have valid sync data.
  119. xr_sync_has_run = false;
  120. }
  121. bool OpenXRRenderModelExtension::on_event_polled(const XrEventDataBuffer &event) {
  122. if (event.type == XR_TYPE_EVENT_DATA_INTERACTION_RENDER_MODELS_CHANGED_EXT) {
  123. // Mark interaction data as dirty so that we update it on sync.
  124. _interaction_data_dirty = true;
  125. return true;
  126. } else if (event.type == XR_TYPE_EVENT_DATA_INTERACTION_PROFILE_CHANGED) {
  127. // If our controller bindings changed, its likely our render models change too.
  128. // We should be getting a XR_TYPE_EVENT_DATA_INTERACTION_RENDER_MODELS_CHANGED_EXT
  129. // but checking for this scenario just in case.
  130. _interaction_data_dirty = true;
  131. // Do not consider this handled, we simply do additional logic.
  132. return false;
  133. }
  134. return false;
  135. }
  136. void OpenXRRenderModelExtension::on_sync_actions() {
  137. if (!is_active()) {
  138. return;
  139. }
  140. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  141. ERR_FAIL_NULL(openxr_api);
  142. // Mark sync as run
  143. xr_sync_has_run = true;
  144. // Update our interaction data if needed
  145. if (_interaction_data_dirty) {
  146. _update_interaction_data();
  147. }
  148. // Loop through all of our render models to update our space and state info
  149. LocalVector<RID> owned = render_model_owner.get_owned_list();
  150. for (const RID &rid : owned) {
  151. RenderModel *render_model = render_model_owner.get_or_null(rid);
  152. if (render_model && render_model->xr_space != XR_NULL_HANDLE) {
  153. XrSpaceLocation render_model_location = {
  154. XR_TYPE_SPACE_LOCATION, // type
  155. nullptr, // next
  156. 0, // locationFlags
  157. { { 0.0, 0.0, 0.0, 1.0 }, { 0.0, 0.0, 0.0 } }, // pose
  158. };
  159. XrResult result = xrLocateSpace(render_model->xr_space, openxr_api->get_play_space(), openxr_api->get_predicted_display_time(), &render_model_location);
  160. ERR_CONTINUE_MSG(XR_FAILED(result), "OpenXR: Failed to locate render model space [" + openxr_api->get_error_string(result) + "]");
  161. render_model->confidence = openxr_api->transform_from_location(render_model_location, render_model->root_transform);
  162. if (!render_model->node_states.is_empty()) {
  163. // Get node states.
  164. XrRenderModelStateGetInfoEXT get_state_info = {
  165. XR_TYPE_RENDER_MODEL_STATE_GET_INFO_EXT, // type
  166. nullptr, // next
  167. openxr_api->get_predicted_display_time() // displayTime
  168. };
  169. XrRenderModelStateEXT state = {
  170. XR_TYPE_RENDER_MODEL_STATE_EXT, // type
  171. nullptr, // next
  172. render_model->animatable_node_count, // nodeStateCount
  173. render_model->node_states.ptr(), // nodeStates
  174. };
  175. result = xrGetRenderModelStateEXT(render_model->xr_render_model, &get_state_info, &state);
  176. if (XR_FAILED(result)) {
  177. ERR_PRINT("OpenXR: Failed to update node states [" + openxr_api->get_error_string(result) + "]");
  178. }
  179. }
  180. XrPath new_path = XR_NULL_PATH;
  181. if (toplevel_paths.is_empty()) {
  182. // Set this up just once with paths we support here.
  183. toplevel_paths.push_back(openxr_api->get_xr_path("/user/hand/left"));
  184. toplevel_paths.push_back(openxr_api->get_xr_path("/user/hand/right"));
  185. }
  186. XrInteractionRenderModelTopLevelUserPathGetInfoEXT info = {
  187. XR_TYPE_INTERACTION_RENDER_MODEL_TOP_LEVEL_USER_PATH_GET_INFO_EXT, // type
  188. nullptr, // next
  189. (uint32_t)toplevel_paths.size(), // topLevelUserPathCount
  190. toplevel_paths.ptr() // topLevelUserPaths
  191. };
  192. result = xrGetRenderModelPoseTopLevelUserPathEXT(render_model->xr_render_model, &info, &new_path);
  193. if (XR_FAILED(result)) {
  194. ERR_PRINT("OpenXR: Failed to update the top level path for render models [" + openxr_api->get_error_string(result) + "]");
  195. } else if (new_path != render_model->top_level_path) {
  196. print_verbose("OpenXR: Render model top level path changed to " + openxr_api->get_xr_path_name(new_path));
  197. // Set the new path
  198. render_model->top_level_path = new_path;
  199. // And broadcast it
  200. // Note, converting an XrPath to a String has overhead, so we won't do this automatically.
  201. emit_signal(SNAME("render_model_top_level_path_changed"), rid);
  202. }
  203. }
  204. }
  205. }
  206. bool OpenXRRenderModelExtension::is_active() const {
  207. return render_model_ext && interaction_render_model_ext;
  208. }
  209. void OpenXRRenderModelExtension::_clear_interaction_data() {
  210. for (const KeyValue<XrRenderModelIdEXT, RID> &e : interaction_render_models) {
  211. render_model_destroy(e.value);
  212. }
  213. interaction_render_models.clear();
  214. }
  215. bool OpenXRRenderModelExtension::_update_interaction_data() {
  216. ERR_FAIL_COND_V_MSG(!interaction_render_model_ext, false, "Interaction render model extension hasn't been enabled.");
  217. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  218. ERR_FAIL_NULL_V(openxr_api, false);
  219. XrSession session = openxr_api->get_session();
  220. ERR_FAIL_COND_V(session == XR_NULL_HANDLE, false);
  221. // Check if syncActions has been run at least once or there is no point in getting data.
  222. if (!xr_sync_has_run) {
  223. // Do not treat this as an error.
  224. return true;
  225. }
  226. // If we get this far, no longer mark as dirty.
  227. // Else we just repeat the same error over and over again.
  228. _interaction_data_dirty = false;
  229. // Obtain interaction info.
  230. XrInteractionRenderModelIdsEnumerateInfoEXT interaction_info = {
  231. XR_TYPE_INTERACTION_RENDER_MODEL_IDS_ENUMERATE_INFO_EXT, // type
  232. nullptr, // next
  233. };
  234. // Obtain count.
  235. uint32_t interaction_count = 0;
  236. XrResult result = xrEnumerateInteractionRenderModelIdsEXT(session, &interaction_info, 0, &interaction_count, nullptr);
  237. if (XR_FAILED(result)) {
  238. // not successful? then we do nothing.
  239. ERR_FAIL_V_MSG(false, "OpenXR: Failed to obtain render model interaction id count [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  240. }
  241. // Create some storage
  242. LocalVector<XrRenderModelIdEXT> render_model_interaction_ids;
  243. render_model_interaction_ids.resize(interaction_count);
  244. // Only need to fetch data if there is something to fetch (/we've got storage).
  245. if (!render_model_interaction_ids.is_empty()) {
  246. // Obtain interaction ids
  247. result = xrEnumerateInteractionRenderModelIdsEXT(session, &interaction_info, render_model_interaction_ids.size(), &interaction_count, render_model_interaction_ids.ptr());
  248. if (XR_FAILED(result)) {
  249. ERR_FAIL_V_MSG(false, "OpenXR: Failed to obtain render model interaction ids [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  250. }
  251. }
  252. // Remove render models that are no longer tracked
  253. LocalVector<XrRenderModelIdEXT> erase_ids;
  254. for (const KeyValue<XrRenderModelIdEXT, RID> &e : interaction_render_models) {
  255. if (!render_model_interaction_ids.has(e.key)) {
  256. if (e.value.is_valid()) {
  257. render_model_destroy(e.value);
  258. }
  259. erase_ids.push_back(e.key);
  260. }
  261. }
  262. // Remove these from our hashmap
  263. for (const XrRenderModelIdEXT &id : erase_ids) {
  264. interaction_render_models.erase(id);
  265. }
  266. // Now update our models
  267. for (const XrRenderModelIdEXT &id : render_model_interaction_ids) {
  268. if (!interaction_render_models.has(id)) {
  269. // Even if this fails we add it so we don't repeat trying to create it
  270. interaction_render_models[id] = render_model_create(id);
  271. }
  272. }
  273. return true;
  274. }
  275. bool OpenXRRenderModelExtension::has_render_model(RID p_render_model) const {
  276. return render_model_owner.owns(p_render_model);
  277. }
  278. RID OpenXRRenderModelExtension::render_model_create(XrRenderModelIdEXT p_render_model_id) {
  279. ERR_FAIL_COND_V_MSG(!render_model_ext, RID(), "Render model extension hasn't been enabled.");
  280. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  281. ERR_FAIL_NULL_V(openxr_api, RID());
  282. XrSession session = openxr_api->get_session();
  283. ERR_FAIL_COND_V(session == XR_NULL_HANDLE, RID());
  284. RenderModel render_model;
  285. render_model.xr_render_model_id = p_render_model_id;
  286. // Start with the extensions that are supported in our base (see GLTFDocument::_parse_gltf_extensions).
  287. Vector<const char *> supported_gltf_extensions = {
  288. "KHR_lights_punctual",
  289. "KHR_materials_pbrSpecularGlossiness",
  290. "KHR_texture_transform",
  291. "KHR_materials_unlit",
  292. "KHR_materials_emissive_strength",
  293. };
  294. // Now find anything we support through plugins, which is a bit of a pain as they are converted to Strings
  295. // and we need to convert them back.
  296. Vector<CharString> char_extensions; // Just for temp storage of our c-strings.
  297. Vector<Ref<GLTFDocumentExtension>> gltf_document_extensions = GLTFDocument::get_all_gltf_document_extensions();
  298. for (Ref<GLTFDocumentExtension> &gltf_document_extension : gltf_document_extensions) {
  299. Vector<String> supported_extensions = gltf_document_extension->get_supported_extensions();
  300. for (const String &extension : supported_extensions) {
  301. char_extensions.push_back(extension.utf8());
  302. }
  303. }
  304. // Now we can add them to our supported extensions list.
  305. for (const CharString &cs : char_extensions) {
  306. supported_gltf_extensions.push_back(cs.get_data());
  307. }
  308. XrRenderModelCreateInfoEXT create_info = {
  309. XR_TYPE_RENDER_MODEL_CREATE_INFO_EXT, // type
  310. nullptr, // next
  311. p_render_model_id, // renderModelId
  312. uint32_t(supported_gltf_extensions.size()), // gltfExtensionCount
  313. supported_gltf_extensions.ptr(), // gltfExtensions
  314. };
  315. XrResult result = xrCreateRenderModelEXT(session, &create_info, &render_model.xr_render_model);
  316. if (XR_FAILED(result)) {
  317. ERR_FAIL_V_MSG(RID(), "OpenXR: Failed to create render model [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  318. }
  319. XrRenderModelPropertiesGetInfoEXT properties_info = {
  320. XR_TYPE_RENDER_MODEL_PROPERTIES_GET_INFO_EXT, // type
  321. nullptr, // next
  322. };
  323. XrRenderModelPropertiesEXT properties;
  324. result = xrGetRenderModelPropertiesEXT(render_model.xr_render_model, &properties_info, &properties);
  325. if (XR_FAILED(result)) {
  326. ERR_PRINT("OpenXR: Failed to get render model properties [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  327. } else {
  328. render_model.animatable_node_count = properties.animatableNodeCount;
  329. render_model.render_model_data = _get_render_model_data(properties.cacheId, properties.animatableNodeCount);
  330. }
  331. // Create space for positioning our asset.
  332. XrRenderModelSpaceCreateInfoEXT space_create_info = {
  333. XR_TYPE_RENDER_MODEL_SPACE_CREATE_INFO_EXT, // type
  334. nullptr, // next
  335. render_model.xr_render_model // renderModel
  336. };
  337. result = xrCreateRenderModelSpaceEXT(session, &space_create_info, &render_model.xr_space);
  338. if (XR_FAILED(result)) {
  339. ERR_PRINT("OpenXR: Failed to create render model space [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  340. }
  341. if (render_model.animatable_node_count > 0) {
  342. render_model.node_states.resize(render_model.animatable_node_count);
  343. }
  344. RID new_rid = render_model_owner.make_rid(render_model);
  345. emit_signal(SNAME("render_model_added"), new_rid);
  346. return new_rid;
  347. }
  348. RID OpenXRRenderModelExtension::_render_model_create(uint64_t p_render_model_id) {
  349. RID ret;
  350. ERR_FAIL_COND_V(p_render_model_id == XR_NULL_RENDER_MODEL_ID_EXT, ret);
  351. if (is_active()) {
  352. ret = render_model_create(XrRenderModelIdEXT(p_render_model_id));
  353. }
  354. return ret;
  355. }
  356. void OpenXRRenderModelExtension::render_model_destroy(RID p_render_model) {
  357. ERR_FAIL_COND_MSG(!render_model_ext, "Render model extension hasn't been enabled.");
  358. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  359. ERR_FAIL_NULL(render_model);
  360. emit_signal(SNAME("render_model_removed"), p_render_model);
  361. // Clean up.
  362. if (render_model->xr_space != XR_NULL_HANDLE) {
  363. xrDestroySpace(render_model->xr_space);
  364. }
  365. render_model->node_states.clear();
  366. // And destroy our model.
  367. XrResult result = xrDestroyRenderModelEXT(render_model->xr_render_model);
  368. if (XR_FAILED(result)) {
  369. ERR_PRINT("OpenXR: Failed to destroy render model [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  370. }
  371. render_model_owner.free(p_render_model);
  372. }
  373. TypedArray<RID> OpenXRRenderModelExtension::render_model_get_all() {
  374. TypedArray<RID> ret;
  375. LocalVector<RID> rids = render_model_owner.get_owned_list();
  376. for (const RID &rid : rids) {
  377. ret.push_back(rid);
  378. }
  379. return ret;
  380. }
  381. Node3D *OpenXRRenderModelExtension::render_model_new_scene_instance(RID p_render_model) const {
  382. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  383. ERR_FAIL_NULL_V(render_model, nullptr);
  384. if (render_model->render_model_data.is_null()) {
  385. // We never loaded it (don't spam errors here).
  386. return nullptr;
  387. }
  388. return render_model->render_model_data->new_scene_instance();
  389. }
  390. PackedStringArray OpenXRRenderModelExtension::render_model_get_subaction_paths(RID p_render_model) {
  391. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  392. ERR_FAIL_NULL_V(openxr_api, PackedStringArray());
  393. XrInstance instance = openxr_api->get_instance();
  394. ERR_FAIL_COND_V(instance == XR_NULL_HANDLE, PackedStringArray());
  395. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  396. ERR_FAIL_NULL_V(render_model, PackedStringArray());
  397. PackedStringArray subaction_paths;
  398. XrInteractionRenderModelSubactionPathInfoEXT subaction_info = {
  399. XR_TYPE_INTERACTION_RENDER_MODEL_SUBACTION_PATH_INFO_EXT, // type
  400. nullptr, // next
  401. };
  402. uint32_t capacity;
  403. XrResult result = xrEnumerateRenderModelSubactionPathsEXT(render_model->xr_render_model, &subaction_info, 0, &capacity, nullptr);
  404. if (XR_FAILED(result)) {
  405. ERR_FAIL_V_MSG(PackedStringArray(), "OpenXR: Failed to obtain render model subaction path count [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  406. }
  407. if (capacity > 0) {
  408. LocalVector<XrPath> paths;
  409. paths.resize(capacity);
  410. result = xrEnumerateRenderModelSubactionPathsEXT(render_model->xr_render_model, &subaction_info, capacity, &capacity, paths.ptr());
  411. if (XR_FAILED(result)) {
  412. ERR_FAIL_V_MSG(PackedStringArray(), "OpenXR: Failed to obtain render model subaction paths [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  413. }
  414. for (uint32_t i = 0; i < capacity; i++) {
  415. char buffer[1024];
  416. uint32_t size = 0;
  417. xrPathToString(instance, paths[i], 1024, &size, buffer);
  418. if (size > 0) {
  419. subaction_paths.push_back(String(buffer));
  420. }
  421. }
  422. }
  423. return subaction_paths;
  424. }
  425. XrPath OpenXRRenderModelExtension::render_model_get_top_level_path(RID p_render_model) const {
  426. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  427. ERR_FAIL_NULL_V(render_model, XRPose::TrackingConfidence::XR_TRACKING_CONFIDENCE_NONE);
  428. return render_model->top_level_path;
  429. }
  430. String OpenXRRenderModelExtension::render_model_get_top_level_path_as_string(RID p_render_model) const {
  431. String ret;
  432. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  433. ERR_FAIL_NULL_V(openxr_api, ret);
  434. if (is_active() && has_render_model(p_render_model)) {
  435. XrPath path = render_model_get_top_level_path(p_render_model);
  436. if (path == XR_NULL_PATH) {
  437. return "None";
  438. } else {
  439. return openxr_api->get_xr_path_name(path);
  440. }
  441. }
  442. return ret;
  443. }
  444. XRPose::TrackingConfidence OpenXRRenderModelExtension::render_model_get_confidence(RID p_render_model) const {
  445. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  446. ERR_FAIL_NULL_V(render_model, XRPose::TrackingConfidence::XR_TRACKING_CONFIDENCE_NONE);
  447. return render_model->confidence;
  448. }
  449. Transform3D OpenXRRenderModelExtension::render_model_get_root_transform(RID p_render_model) const {
  450. XRServer *xr_server = XRServer::get_singleton();
  451. ERR_FAIL_NULL_V(xr_server, Transform3D());
  452. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  453. ERR_FAIL_NULL_V(render_model, Transform3D());
  454. // Scale our root transform
  455. real_t world_scale = xr_server->get_world_scale();
  456. Transform3D root_transform = render_model->root_transform.scaled(Vector3(world_scale, world_scale, world_scale));
  457. return xr_server->get_reference_frame() * root_transform;
  458. }
  459. uint32_t OpenXRRenderModelExtension::render_model_get_animatable_node_count(RID p_render_model) const {
  460. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  461. ERR_FAIL_NULL_V(render_model, 0);
  462. return render_model->animatable_node_count;
  463. }
  464. String OpenXRRenderModelExtension::render_model_get_animatable_node_name(RID p_render_model, uint32_t p_index) const {
  465. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  466. ERR_FAIL_NULL_V(render_model, String());
  467. if (render_model->render_model_data.is_null()) {
  468. // We never loaded it (don't spam errors here).
  469. return String();
  470. }
  471. return render_model->render_model_data->get_node_name(p_index);
  472. }
  473. bool OpenXRRenderModelExtension::render_model_is_animatable_node_visible(RID p_render_model, uint32_t p_index) const {
  474. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  475. ERR_FAIL_NULL_V(render_model, false);
  476. ERR_FAIL_UNSIGNED_INDEX_V(p_index, render_model->animatable_node_count, false);
  477. if (render_model->node_states.is_empty()) {
  478. // Never allocated (don't spam errors here).
  479. return false;
  480. }
  481. return render_model->node_states[p_index].isVisible;
  482. }
  483. Transform3D OpenXRRenderModelExtension::render_model_get_animatable_node_transform(RID p_render_model, uint32_t p_index) const {
  484. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  485. ERR_FAIL_NULL_V(openxr_api, Transform3D());
  486. RenderModel *render_model = render_model_owner.get_or_null(p_render_model);
  487. ERR_FAIL_NULL_V(render_model, Transform3D());
  488. ERR_FAIL_UNSIGNED_INDEX_V(p_index, render_model->animatable_node_count, Transform3D());
  489. if (render_model->node_states.is_empty()) {
  490. // Never allocated (don't spam errors here).
  491. return Transform3D();
  492. }
  493. return openxr_api->transform_from_pose(render_model->node_states[p_index].nodePose);
  494. }
  495. Ref<OpenXRRenderModelData> OpenXRRenderModelExtension::_get_render_model_data(XrUuidEXT p_cache_id, uint32_t p_animatable_node_count) {
  496. if (render_model_data_cache.has(p_cache_id)) {
  497. return render_model_data_cache[p_cache_id];
  498. }
  499. // We don't have this cached, lets load it up
  500. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  501. ERR_FAIL_NULL_V(openxr_api, nullptr);
  502. XrSession session = openxr_api->get_session();
  503. ERR_FAIL_COND_V(session == XR_NULL_HANDLE, nullptr);
  504. XrRenderModelAssetEXT asset;
  505. XrRenderModelAssetCreateInfoEXT create_info = {
  506. XR_TYPE_RENDER_MODEL_ASSET_CREATE_INFO_EXT, // type
  507. nullptr, // next
  508. p_cache_id // cacheId
  509. };
  510. XrResult result = xrCreateRenderModelAssetEXT(session, &create_info, &asset);
  511. if (XR_FAILED(result)) {
  512. ERR_FAIL_V_MSG(nullptr, "OpenXR: Failed to create render model asset [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  513. }
  514. Ref<OpenXRRenderModelData> render_model_data = _load_asset(asset, p_animatable_node_count);
  515. // We're done with this :)
  516. result = xrDestroyRenderModelAssetEXT(asset);
  517. if (XR_FAILED(result)) {
  518. ERR_PRINT("OpenXR: Failed to destroy render model asset [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  519. }
  520. // And cache it
  521. render_model_data_cache[p_cache_id] = render_model_data;
  522. return render_model_data;
  523. }
  524. Ref<OpenXRRenderModelData> OpenXRRenderModelExtension::_load_asset(XrRenderModelAssetEXT p_asset, uint32_t p_animatable_node_count) {
  525. XrRenderModelAssetDataGetInfoEXT get_info = {
  526. XR_TYPE_RENDER_MODEL_ASSET_DATA_GET_INFO_EXT, // type
  527. nullptr, // next
  528. };
  529. XrRenderModelAssetDataEXT asset_data = {
  530. XR_TYPE_RENDER_MODEL_ASSET_DATA_EXT, // type
  531. nullptr, // next
  532. 0, // bufferCapacityInput;
  533. 0, // bufferCountOutput;
  534. nullptr // buffer;
  535. };
  536. // Obtain required size for the buffer.
  537. XrResult result = xrGetRenderModelAssetDataEXT(p_asset, &get_info, &asset_data);
  538. if (XR_FAILED(result)) {
  539. ERR_FAIL_V_MSG(nullptr, "OpenXR: Failed to get render model buffer size [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  540. }
  541. ERR_FAIL_COND_V(asset_data.bufferCountOutput == 0, nullptr);
  542. // Allocate data
  543. PackedByteArray buffer;
  544. buffer.resize(asset_data.bufferCountOutput);
  545. asset_data.buffer = buffer.ptrw();
  546. asset_data.bufferCapacityInput = asset_data.bufferCountOutput;
  547. // Now get our actual data.
  548. result = xrGetRenderModelAssetDataEXT(p_asset, &get_info, &asset_data);
  549. if (XR_FAILED(result)) {
  550. ERR_FAIL_V_MSG(nullptr, "OpenXR: Failed to get render model buffer [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  551. }
  552. // Get the names of any animatable nodes
  553. PackedStringArray node_names;
  554. if (p_animatable_node_count > 0) {
  555. Vector<XrRenderModelAssetNodePropertiesEXT> node_properties;
  556. node_properties.resize(p_animatable_node_count);
  557. XrRenderModelAssetPropertiesGetInfoEXT properties_info = {
  558. XR_TYPE_RENDER_MODEL_ASSET_PROPERTIES_GET_INFO_EXT, // type
  559. nullptr, // next
  560. };
  561. XrRenderModelAssetPropertiesEXT asset_properties = {
  562. XR_TYPE_RENDER_MODEL_ASSET_PROPERTIES_EXT, // type
  563. nullptr, // next
  564. uint32_t(node_properties.size()), // nodePropertyCount
  565. node_properties.ptrw(), // nodeProperties
  566. };
  567. result = xrGetRenderModelAssetPropertiesEXT(p_asset, &properties_info, &asset_properties);
  568. if (XR_FAILED(result)) {
  569. ERR_FAIL_V_MSG(nullptr, "OpenXR: Failed to get render model property info [" + OpenXRAPI::get_singleton()->get_error_string(result) + "]");
  570. }
  571. node_names.resize(p_animatable_node_count);
  572. String *node_names_ptrw = node_names.ptrw();
  573. for (uint32_t i = 0; i < p_animatable_node_count; i++) {
  574. node_names_ptrw[i] = String(node_properties[i].uniqueName);
  575. }
  576. }
  577. Ref<OpenXRRenderModelData> render_model_data;
  578. render_model_data.instantiate();
  579. render_model_data->parse_gltf_document(buffer);
  580. render_model_data->set_node_names(node_names);
  581. return render_model_data;
  582. }
  583. void OpenXRRenderModelExtension::_clear_render_model_data() {
  584. // Clear our toplevel paths filter.
  585. toplevel_paths.clear();
  586. // Clear our render model cache.
  587. render_model_data_cache.clear();
  588. // Loop through all of our render models and destroy them.
  589. LocalVector<RID> owned = render_model_owner.get_owned_list();
  590. for (const RID &rid : owned) {
  591. render_model_destroy(rid);
  592. }
  593. }
  594. bool OpenXRRenderModelData::parse_gltf_document(const PackedByteArray &p_bytes) {
  595. // State holds our data, document parses GLTF
  596. Ref<GLTFState> new_state;
  597. new_state.instantiate();
  598. Ref<GLTFDocument> new_gltf_document;
  599. new_gltf_document.instantiate();
  600. Error err = new_gltf_document->append_from_buffer(p_bytes, "", new_state);
  601. if (err != OK) {
  602. ERR_FAIL_V_MSG(false, "OpenXR: Failed to parse GLTF data.");
  603. }
  604. gltf_document = new_gltf_document;
  605. gltf_state = new_state;
  606. return true;
  607. }
  608. Node3D *OpenXRRenderModelData::new_scene_instance() {
  609. ERR_FAIL_COND_V(gltf_document.is_null(), nullptr);
  610. ERR_FAIL_COND_V(gltf_state.is_null(), nullptr);
  611. return Object::cast_to<Node3D>(gltf_document->generate_scene(gltf_state));
  612. }
  613. void OpenXRRenderModelData::set_node_names(const PackedStringArray &p_node_names) {
  614. node_names = p_node_names;
  615. }
  616. PackedStringArray OpenXRRenderModelData::get_node_names() const {
  617. return node_names;
  618. }
  619. const String OpenXRRenderModelData::get_node_name(uint32_t p_node_index) const {
  620. ERR_FAIL_UNSIGNED_INDEX_V(p_node_index, node_names.size(), String());
  621. return node_names[p_node_index];
  622. }
  623. OpenXRRenderModelData::OpenXRRenderModelData() {
  624. }
  625. OpenXRRenderModelData::~OpenXRRenderModelData() {
  626. }