axmol/3rdparty/jni.hpp/include/jni/advanced_ownership.hpp

113 lines
3.8 KiB
C++

#pragma once
#include <jni/functions.hpp>
namespace jni
{
// A deleter that gets the JNIEnv via GetEnv, rather than storing the value passed to the constructor.
// The deleting thread must have a JVM attachment.
//
// Useful when deletion will happen on an auxiliary thread, particularly the finalizer thread. In such
// cases, you may use one of the following:
//
// low-level: UniqueGlobalRef<jobject, EnvGettingDeleter> and NewGlobalRef<EnvGettingDeleter>
// high-level: Global<Object<Tag>, EnvGettingDeleter> and obj.NewGlobalRef<EnvGettingDeleter>
//
template < RefDeletionMethod DeleteRef >
class EnvGettingDeleter
{
private:
JavaVM* vm = nullptr;
public:
EnvGettingDeleter() = default;
EnvGettingDeleter(JNIEnv& e) : vm(&GetJavaVM(e)) {}
void operator()(jobject* p) const
{
if (p)
{
assert(vm);
(GetEnv(*vm).*DeleteRef)(Unwrap(p));
}
}
};
// A deleter that first tries GetEnv, but falls back to AttachCurrentThread if a JVM is not already attached.
// In the latter case, it detaches after deleting the reference.
//
// Useful when deletion will happen on an auxiliary thread which may or may not have a JVM attachment. In such
// cases, you may use one of the following:
//
// low-level: UniqueGlobalRef<jobject, EnvAttachingDeleter> and NewGlobalRef<EnvAttachingDeleter>
// high-level: Global<Object<Tag>, EnvAttachingDeleter> and obj.NewGlobalRef<EnvAttachingDeleter>
//
template < RefDeletionMethod DeleteRef >
class EnvAttachingDeleter
{
private:
JavaVM* vm = nullptr;
public:
EnvAttachingDeleter() = default;
EnvAttachingDeleter(JNIEnv& e) : vm(&GetJavaVM(e)) {}
void operator()(jobject* p) const
{
if (p)
{
assert(vm);
JNIEnv* env = nullptr;
jint err = vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_1);
if (err == JNI_OK)
{
(env->*DeleteRef)(Unwrap(p));
}
else if (err == JNI_EDETACHED)
{
((*AttachCurrentThread(*vm)).*DeleteRef)(Unwrap(p));
}
else
{
CheckErrorCode(err);
}
}
}
};
// A deleter that tries to get the JNIEnv via GetEnv, and does nothing if that fails.
//
// This is used to ignore GlobalRef deletions that happen after a thread has been detached,
// for instance during process shutdown, when there's no need to release the reference anyway.
// Specifically, it's what Class<T>::Singleton uses.
//
template < RefDeletionMethod DeleteRef >
class EnvIgnoringDeleter
{
private:
JavaVM* vm = nullptr;
public:
EnvIgnoringDeleter() = default;
EnvIgnoringDeleter(JNIEnv& e) : vm(&GetJavaVM(e)) {}
void operator()(jobject* p) const
{
if (p)
{
assert(vm);
JNIEnv* env = nullptr;
jint err = vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_1);
if (err == JNI_OK)
{
(env->*DeleteRef)(Unwrap(p));
}
else if (err != JNI_EDETACHED)
{
CheckErrorCode(err);
}
}
}
};
}