// Copyright 2020, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief Implementation of native code for Android custom surface. * @author Ryan Pavlik * @author Lubosz Sarnecki * @ingroup aux_android */ #include "android_custom_surface.h" #include "android_load_class.hpp" #include "xrt/xrt_config_android.h" #include "util/u_logging.h" #include "wrap/android.app.h" #include "wrap/android.view.h" #include "org.freedesktop.monado.auxiliary.hpp" #include using wrap::android::app::Activity; using wrap::android::view::SurfaceHolder; using wrap::org::freedesktop::monado::auxiliary::MonadoView; struct android_custom_surface { explicit android_custom_surface(jobject act); ~android_custom_surface(); Activity activity{}; MonadoView monadoView{}; jni::Class monadoViewClass{}; }; android_custom_surface::android_custom_surface(jobject act) : activity(act) {} android_custom_surface::~android_custom_surface() { // Tell Java that native code is done with this. try { if (!monadoView.isNull()) { monadoView.markAsDiscardedByNative(); } } catch (std::exception const &e) { // Must catch and ignore any exceptions in the destructor! U_LOG_E("Failure while marking MonadoView as discarded: %s", e.what()); } } constexpr auto FULLY_QUALIFIED_CLASSNAME = "org.freedesktop.monado.auxiliary.MonadoView"; struct android_custom_surface * android_custom_surface_async_start(struct _JavaVM *vm, void *activity) { jni::init(vm); try { auto info = getAppInfo(XRT_ANDROID_PACKAGE, (jobject)activity); if (info.isNull()) { U_LOG_E( "Could not get application info for package '%s'", "org.freedesktop.monado.openxr_runtime"); return nullptr; } auto clazz = loadClassFromPackage(info, (jobject)activity, FULLY_QUALIFIED_CLASSNAME); if (clazz.isNull()) { U_LOG_E("Could not load class '%s' from package '%s'", FULLY_QUALIFIED_CLASSNAME, XRT_ANDROID_PACKAGE); return nullptr; } // Teach the wrapper our class before we start to use it. MonadoView::staticInitClass((jclass)clazz.object().getHandle()); std::unique_ptr ret = std::make_unique((jobject)activity); // the 0 is to avoid this being considered "temporary" and to // create a global ref. ret->monadoViewClass = jni::Class((jclass)clazz.object().getHandle(), 0); if (ret->monadoViewClass.isNull()) { U_LOG_E("monadoViewClass was null"); return nullptr; } std::string clazz_name = ret->monadoViewClass.getName(); if (clazz_name != FULLY_QUALIFIED_CLASSNAME) { U_LOG_E("Unexpected class name: %s", clazz_name.c_str()); return nullptr; } ret->monadoView = MonadoView::attachToActivity(ret->activity); return ret.release(); } catch (std::exception const &e) { U_LOG_E( "Could not start attaching our custom surface to activity: " "%s", e.what()); return nullptr; } } void android_custom_surface_destroy( struct android_custom_surface **ptr_custom_surface) { if (ptr_custom_surface == NULL) { return; } struct android_custom_surface *custom_surface = *ptr_custom_surface; if (custom_surface == NULL) { return; } delete custom_surface; *ptr_custom_surface = NULL; } ANativeWindow * android_custom_surface_wait_get_surface( struct android_custom_surface *custom_surface, uint64_t timeout_ms) { SurfaceHolder surfaceHolder{}; try { surfaceHolder = custom_surface->monadoView.waitGetSurfaceHolder(timeout_ms); } catch (std::exception const &e) { // do nothing right now. U_LOG_E( "Could not wait for our custom surface: " "%s", e.what()); return nullptr; } if (surfaceHolder.isNull()) { return nullptr; } auto surf = surfaceHolder.getSurface(); if (surf.isNull()) { return nullptr; } return ANativeWindow_fromSurface(jni::env(), surf.object().makeLocalReference()); }