Add Scoped Local and Global Ref
Summary: Add ScopedLocalRef, ScopedGlobalRef and some common methods to be used later. Reviewed By: amir-shalem Differential Revision: D17711284 fbshipit-source-id: be43d5e246bc2406765057783be11854877c41f1
This commit is contained in:
committed by
Facebook Github Bot
parent
7c2683fe52
commit
b9b0217a07
140
java/jni/ScopedLocalRef.h
Normal file
140
java/jni/ScopedLocalRef.h
Normal file
@@ -0,0 +1,140 @@
|
||||
/*
|
||||
* Copyright (c) Facebook, Inc. and its affiliates.
|
||||
*
|
||||
* This source code is licensed under the MIT license found in the LICENSE
|
||||
* file in the root directory of this source tree.
|
||||
*/
|
||||
/**
|
||||
* This is a modified version of Android's ScopedLocalRef class that can be
|
||||
* found in the Android's JNI code.
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#include <jni.h>
|
||||
#include <cstddef>
|
||||
#include <type_traits>
|
||||
|
||||
namespace facebook {
|
||||
namespace yoga {
|
||||
namespace vanillajni {
|
||||
|
||||
/**
|
||||
* ScopedLocalRef is a sort of smart reference that allows us to control the
|
||||
* lifespan of a JNI local reference.
|
||||
*
|
||||
* This class is designed so that when a ScopedLocalRef goes out of scope, its
|
||||
* destructor will delete -JNIEnv->DeleteLocalRef()- the underlying JNI
|
||||
* reference.
|
||||
*
|
||||
* This class should be used to wrap all the local references that JNI
|
||||
* gives us other than those that are passed to native methods at
|
||||
* invocation time. The idea behind this is that in JNI we should be very
|
||||
* explicit about the lifespan of local references. Local references can quickly
|
||||
* get out of control, and the developer should always be very aware of the
|
||||
* lifespan of each local reference that is created in JNI so that leaks are
|
||||
* prevented.
|
||||
*
|
||||
* This class is very explicit in its behavior, and it does not allow to perform
|
||||
* unexpected conversions or unexpected ownership transfer. In practice, this
|
||||
* class acts as a unique pointer where the underying JNI reference can have one
|
||||
* and just one owner. Transfering ownership is allowed but it is an explicit
|
||||
* operation (implemneted via move semantics and also via explicity API calls).
|
||||
*
|
||||
* As with standard JNI local references it is not a valid operation to keep a
|
||||
* reference around between different native method calls.
|
||||
*/
|
||||
template <typename T>
|
||||
class ScopedLocalRef {
|
||||
static_assert(
|
||||
std::is_same<T, jclass>() || std::is_same<T, jobject>() ||
|
||||
std::is_same<T, jstring>() || std::is_same<T, jthrowable>() ||
|
||||
std::is_same<T, jbyteArray>() || std::is_same<T, jintArray>() ||
|
||||
std::is_same<T, jshortArray>() || std::is_same<T, jcharArray>() ||
|
||||
std::is_same<T, jlongArray>() || std::is_same<T, jfloatArray>() ||
|
||||
std::is_same<T, jdoubleArray>() || std::is_same<T, jobjectArray>() ||
|
||||
std::is_same<T, jbooleanArray>(),
|
||||
"ScopedLocalRef instantiated for invalid type");
|
||||
|
||||
public:
|
||||
/**
|
||||
* Constructs a ScopedLocalRef with a JNI local reference.
|
||||
*
|
||||
* @param localRef the local reference to wrap. Can be NULL.
|
||||
*/
|
||||
ScopedLocalRef(JNIEnv* env, T localRef) : mEnv(env), mLocalRef(localRef) {}
|
||||
|
||||
/**
|
||||
* Equivalent to ScopedLocalRef(env, NULL)
|
||||
*/
|
||||
explicit ScopedLocalRef(JNIEnv* env) : mEnv(env), mLocalRef(NULL) {}
|
||||
|
||||
/**
|
||||
* Move construction is allowed.
|
||||
*/
|
||||
ScopedLocalRef(ScopedLocalRef&& s) : mEnv(s.mEnv), mLocalRef(s.release()) {}
|
||||
|
||||
/**
|
||||
* Move assignment is allowed.
|
||||
*/
|
||||
ScopedLocalRef& operator=(ScopedLocalRef&& s) {
|
||||
reset(s.release());
|
||||
mEnv = s.mEnv;
|
||||
return *this;
|
||||
}
|
||||
|
||||
~ScopedLocalRef() {
|
||||
reset();
|
||||
}
|
||||
|
||||
/**
|
||||
* Deletes the currently held reference and reassigns a new one to the
|
||||
* ScopedLocalRef.
|
||||
*/
|
||||
void reset(T ptr = NULL) {
|
||||
if (ptr != mLocalRef) {
|
||||
if (mLocalRef != NULL) {
|
||||
mEnv->DeleteLocalRef(mLocalRef);
|
||||
}
|
||||
mLocalRef = ptr;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Makes this ScopedLocalRef not own the underlying JNI local reference. After
|
||||
* calling this method, the ScopedLocalRef will not delete the JNI local
|
||||
* reference when the ScopedLocalRef goes out of scope.
|
||||
*/
|
||||
T release() {
|
||||
T localRef = mLocalRef;
|
||||
mLocalRef = NULL;
|
||||
return localRef;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the underlying JNI local reference.
|
||||
*/
|
||||
T get() const { return mLocalRef; }
|
||||
|
||||
/**
|
||||
* Returns true if the underlying JNI reference is not NULL.
|
||||
*/
|
||||
operator bool() const {
|
||||
return mLocalRef != NULL;
|
||||
}
|
||||
|
||||
ScopedLocalRef(const ScopedLocalRef& ref) = delete;
|
||||
ScopedLocalRef& operator=(const ScopedLocalRef& other) = delete;
|
||||
|
||||
private:
|
||||
JNIEnv* mEnv;
|
||||
T mLocalRef;
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
ScopedLocalRef<T> make_local_ref(JNIEnv* env, T localRef) {
|
||||
return ScopedLocalRef<T>(env, localRef);
|
||||
}
|
||||
|
||||
} // namespace vanillajni
|
||||
} // namespace yoga
|
||||
} // namespace facebook
|
Reference in New Issue
Block a user