// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef BASE_ANDROID_SCOPED_JAVA_REF_H_ #define BASE_ANDROID_SCOPED_JAVA_REF_H_ #include #include #include "base/base_export.h" #include "base/logging.h" #include "base/macros.h" #include "base/template_util.h" namespace base { namespace android { // Creates a new local reference frame, in which at least a given number of // local references can be created. Note that local references already created // in previous local frames are still valid in the current local frame. class BASE_EXPORT ScopedJavaLocalFrame { public: explicit ScopedJavaLocalFrame(JNIEnv* env); ScopedJavaLocalFrame(JNIEnv* env, int capacity); ~ScopedJavaLocalFrame(); private: // This class is only good for use on the thread it was created on so // it's safe to cache the non-threadsafe JNIEnv* inside this object. JNIEnv* env_; DISALLOW_COPY_AND_ASSIGN(ScopedJavaLocalFrame); }; // Forward declare the generic java reference template class. template class JavaRef; // Template specialization of JavaRef, which acts as the base class for all // other JavaRef<> template types. This allows you to e.g. pass // ScopedJavaLocalRef into a function taking const JavaRef& template<> class BASE_EXPORT JavaRef { public: // Allow nullptr to be converted to JavaRef. This avoids having to declare an // empty ScopedJavaLocalRef just to pass null to a function with a JavaRef // parameter, and makes C++ "nullptr" and Java "null" equivalent. JavaRef(std::nullptr_t) : JavaRef() {} // Public to allow destruction of temporary JavaRef objects created by the // nullptr conversion. Don't add anything else here; it's inlined. ~JavaRef() {} jobject obj() const { return obj_; } bool is_null() const { return obj_ == NULL; } protected: // Initializes a NULL reference. Don't add anything else here; it's inlined. JavaRef() : obj_(NULL) {} // Takes ownership of the |obj| reference passed; requires it to be a local // reference type. #if DCHECK_IS_ON() // Implementation contains a DCHECK; implement out-of-line when DCHECK_IS_ON. JavaRef(JNIEnv* env, jobject obj); #else // Don't add anything else here; it's inlined. JavaRef(JNIEnv* env, jobject obj) : obj_(obj) {} #endif // The following are implementation detail convenience methods, for // use by the sub-classes. JNIEnv* SetNewLocalRef(JNIEnv* env, jobject obj); void SetNewGlobalRef(JNIEnv* env, jobject obj); void ResetLocalRef(JNIEnv* env); void ResetGlobalRef(); jobject ReleaseInternal(); private: jobject obj_; DISALLOW_COPY_AND_ASSIGN(JavaRef); }; // Generic base class for ScopedJavaLocalRef and ScopedJavaGlobalRef. Useful // for allowing functions to accept a reference without having to mandate // whether it is a local or global type. template class JavaRef : public JavaRef { public: JavaRef(std::nullptr_t) : JavaRef(nullptr) {} ~JavaRef() {} T obj() const { return static_cast(JavaRef::obj()); } protected: JavaRef() {} JavaRef(JNIEnv* env, T obj) : JavaRef(env, obj) {} private: DISALLOW_COPY_AND_ASSIGN(JavaRef); }; // Holds a local reference to a JNI method parameter. // Method parameters should not be deleted, and so this class exists purely to // wrap them as a JavaRef in the JNI binding generator. Do not create // instances manually. template class JavaParamRef : public JavaRef { public: // Assumes that |obj| is a parameter passed to a JNI method from Java. // Does not assume ownership as parameters should not be deleted. JavaParamRef(JNIEnv* env, T obj) : JavaRef(env, obj) {} // Allow nullptr to be converted to JavaParamRef. Some unit tests call JNI // methods directly from C++ and pass null for objects which are not actually // used by the implementation (e.g. the caller object); allow this to keep // working. JavaParamRef(std::nullptr_t) : JavaRef(nullptr) {} ~JavaParamRef() {} // TODO(torne): remove this cast once we're using JavaRef consistently. // http://crbug.com/506850 operator T() const { return JavaRef::obj(); } private: DISALLOW_COPY_AND_ASSIGN(JavaParamRef); }; // Holds a local reference to a Java object. The local reference is scoped // to the lifetime of this object. // Instances of this class may hold onto any JNIEnv passed into it until // destroyed. Therefore, since a JNIEnv is only suitable for use on a single // thread, objects of this class must be created, used, and destroyed, on a // single thread. // Therefore, this class should only be used as a stack-based object and from a // single thread. If you wish to have the reference outlive the current // callstack (e.g. as a class member) or you wish to pass it across threads, // use a ScopedJavaGlobalRef instead. template class ScopedJavaLocalRef : public JavaRef { public: ScopedJavaLocalRef() : env_(NULL) {} // Non-explicit copy constructor, to allow ScopedJavaLocalRef to be returned // by value as this is the normal usage pattern. ScopedJavaLocalRef(const ScopedJavaLocalRef& other) : env_(other.env_) { this->SetNewLocalRef(env_, other.obj()); } template explicit ScopedJavaLocalRef(const U& other) : env_(NULL) { this->Reset(other); } // Assumes that |obj| is a local reference to a Java object and takes // ownership of this local reference. ScopedJavaLocalRef(JNIEnv* env, T obj) : JavaRef(env, obj), env_(env) {} ~ScopedJavaLocalRef() { this->Reset(); } // Overloaded assignment operator defined for consistency with the implicit // copy constructor. void operator=(const ScopedJavaLocalRef& other) { this->Reset(other); } void Reset() { this->ResetLocalRef(env_); } template void Reset(const ScopedJavaLocalRef& other) { // We can copy over env_ here as |other| instance must be from the same // thread as |this| local ref. (See class comment for multi-threading // limitations, and alternatives). this->Reset(other.env_, other.obj()); } template void Reset(const U& other) { // If |env_| was not yet set (is still NULL) it will be attached to the // current thread in SetNewLocalRef(). this->Reset(env_, other.obj()); } template void Reset(JNIEnv* env, U obj) { static_assert(base::is_convertible::value, "U must be convertible to T"); env_ = this->SetNewLocalRef(env, obj); } // Releases the local reference to the caller. The caller *must* delete the // local reference when it is done with it. Note that calling a Java method // is *not* a transfer of ownership and Release() should not be used. T Release() { return static_cast(this->ReleaseInternal()); } private: // This class is only good for use on the thread it was created on so // it's safe to cache the non-threadsafe JNIEnv* inside this object. JNIEnv* env_; // Prevent ScopedJavaLocalRef(JNIEnv*, T obj) from being used to take // ownership of a JavaParamRef's underlying object - parameters are not // allowed to be deleted and so should not be owned by ScopedJavaLocalRef. // TODO(torne): this can be removed once JavaParamRef no longer has an // implicit conversion back to T. ScopedJavaLocalRef(JNIEnv* env, const JavaParamRef& other); }; // Holds a global reference to a Java object. The global reference is scoped // to the lifetime of this object. This class does not hold onto any JNIEnv* // passed to it, hence it is safe to use across threads (within the constraints // imposed by the underlying Java object that it references). template class ScopedJavaGlobalRef : public JavaRef { public: ScopedJavaGlobalRef() {} ScopedJavaGlobalRef(const ScopedJavaGlobalRef& other) { this->Reset(other); } ScopedJavaGlobalRef(JNIEnv* env, T obj) { this->Reset(env, obj); } template explicit ScopedJavaGlobalRef(const U& other) { this->Reset(other); } ~ScopedJavaGlobalRef() { this->Reset(); } // Overloaded assignment operator defined for consistency with the implicit // copy constructor. void operator=(const ScopedJavaGlobalRef& other) { this->Reset(other); } void Reset() { this->ResetGlobalRef(); } template void Reset(const U& other) { this->Reset(NULL, other.obj()); } template void Reset(JNIEnv* env, const JavaParamRef& other) { this->Reset(env, other.obj()); } template void Reset(JNIEnv* env, U obj) { static_assert(base::is_convertible::value, "U must be convertible to T"); this->SetNewGlobalRef(env, obj); } // Releases the global reference to the caller. The caller *must* delete the // global reference when it is done with it. Note that calling a Java method // is *not* a transfer of ownership and Release() should not be used. T Release() { return static_cast(this->ReleaseInternal()); } }; } // namespace android } // namespace base #endif // BASE_ANDROID_SCOPED_JAVA_REF_H_