Skip to content

Commit

Permalink
Add documentation for NewRef.
Browse files Browse the repository at this point in the history
See #207.

Also, fix minor typos to be able to finally cut a new release.

PiperOrigin-RevId: 581788597
  • Loading branch information
jwhpryor authored and copybara-github committed Nov 13, 2023
1 parent 58e8b53 commit 12db7ae
Show file tree
Hide file tree
Showing 33 changed files with 343 additions and 157 deletions.
1 change: 1 addition & 0 deletions BUILD
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@ cc_library(
"//implementation:no_idx",
"//implementation:params",
"//implementation:promotion_mechanics",
"//implementation:promotion_mechanics_tags",
"//implementation:return",
"//implementation:selector_static_info",
"//implementation:self",
Expand Down
10 changes: 7 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -166,13 +166,13 @@ jni::Class definitions are static (the class names of any Java object is known i

Local and global objects manage lifetimes of underlying `jobjects` using the normal RAII mechanism of C++. **`jni::LocalObject` always fall off scope at the conclusion of the surrounding JNI call and are valid only to a single thread, however `jni::GlobalObject` may be held indefinitely and are thread safe**.

`jni::LocalObject` is built by either wrapping constructing it from a `jobject` passed to native JNI from Java, or constructing a new object from native (see [Constructors](constructors)).
`jni::LocalObject` is built by either wrapping a `jobject` passed to native JNI from Java, or constructing a new object from native (see [Constructors](constructors)).

When `jni::LocalObject` or `jni::GlobalObject` falls off scope, it will unpin the underlying `jobject`, making it available for garbage collection by the JVM. If you want to to prevent this call `Release()`. This is useful to return a `jobject` back from native, or to simply pass to another native component that isn't JNI Bind aware. Calling methods for a released object is undefined.
When `jni::LocalObject` or `jni::GlobalObject` falls off scope, it will unpin the underlying `jobject`, making it available for garbage collection by the JVM. If you want to to prevent this, call `Release()`. This is useful to return a `jobject` back from native, or to simply pass to another native component that isn't JNI Bind aware. Calling methods for a released object is undefined.

When possible try to avoid using raw `jobject`. Managing lifetimes with regular JNI is difficult, e.g. `jobject` can mean either local or global object (the former will be automatically unpinned at the end of the JNI call, but the latter won't and must be deleted _exactly_ once).

Because jobject does not uniquely identify its underlying storage, it is presumed to always be local. If you want to build a global, you must use either `jni::PromoteToGlobal` or `jni::AdoptGlobal`. e.g.
Because `jobject` does not uniquely identify its underlying storage, it is presumed to always be local. If you want to build a global, you must use either `jni::PromoteToGlobal` or `jni::AdoptGlobal`. e.g.

```cpp
jobject obj1, obj2, obj3;
Expand All @@ -181,6 +181,10 @@ jni::LocalObject local_obj {obj1}; // Fine, given local semantics.
jni::GlobalObject global_obj_1 {PromoteToGlobal{}, obj2}; // obj2 will be promoted.
jni::GlobalObject global_obj_2 {AdoptGlobal{}, obj3}; // obj3 will *not* be promoted.
```
When using a jobject you may add `NewRef{}` which creates a new local reference, or `AdoptLocal{}` which takes full ownership.
*Because JNI objects passed to native should never be deleted, `NewRef` is used by default (so that `LocalObject` may always call delete). A non-deleting `FastLocal` that does not delete may be added in the future. In general you shouldn't need to worry about this.*
[Sample C++](javatests/com/jnibind/test/context_test_jni.cc), [Sample Java](javatests/com/jnibind/test/ContextTest.java)
Expand Down
20 changes: 20 additions & 0 deletions implementation/BUILD
Original file line number Diff line number Diff line change
Expand Up @@ -29,10 +29,13 @@ cc_library(
":array",
":array_view",
":class",
":class_ref",
":default_class_loader",
":forward_declarations",
":jni_type",
":local_object",
":object_ref",
":promotion_mechanics_tags",
":ref_base",
"//:jni_dep",
"//implementation/jni_helper:jni_array_helper",
Expand Down Expand Up @@ -292,7 +295,10 @@ cc_library(
":field_selection",
":id",
":id_type",
":promotion_mechanics_tags",
":proxy",
":proxy_convenience_aliases",
":ref_base",
"//:jni_dep",
"//implementation/jni_helper",
"//implementation/jni_helper:field_value_getter",
Expand Down Expand Up @@ -526,6 +532,7 @@ cc_library(
":jvm",
":no_idx",
":object",
":promotion_mechanics_tags",
"//:jni_dep",
"//implementation/jni_helper:jni_array_helper",
"//implementation/jni_helper:lifecycle_object",
Expand Down Expand Up @@ -594,10 +601,12 @@ cc_library(
":class",
":class_loader",
":class_loader_ref",
":forward_declarations",
":jvm",
":jvm_ref",
":local_object",
":promotion_mechanics",
":promotion_mechanics_tags",
"//:jni_dep",
"//class_defs:java_lang_classes",
"//implementation/jni_helper:lifecycle_object",
Expand All @@ -610,11 +619,13 @@ cc_library(
deps = [
":class",
":class_loader",
":forward_declarations",
":jni_type",
":jvm",
":jvm_ref",
":object_ref",
":promotion_mechanics",
":promotion_mechanics_tags",
"//:jni_dep",
"//implementation/jni_helper:lifecycle_object",
],
Expand All @@ -635,8 +646,10 @@ cc_library(
name = "local_string",
hdrs = ["local_string.h"],
deps = [
":forward_declarations",
":local_object",
":promotion_mechanics",
":promotion_mechanics_tags",
":ref_base",
":string_ref",
"//:jni_dep",
Expand All @@ -663,6 +676,7 @@ cc_library(
":jni_type",
":method",
":params",
":promotion_mechanics_tags",
":proxy",
":proxy_convenience_aliases",
":proxy_definitions",
Expand Down Expand Up @@ -819,6 +833,7 @@ cc_library(
":forward_declarations",
":jni_type",
":object_ref",
":promotion_mechanics_tags",
":ref_base",
"//:jni_dep",
"//implementation/jni_helper",
Expand All @@ -829,6 +844,11 @@ cc_library(
],
)

cc_library(
name = "promotion_mechanics_tags",
hdrs = ["promotion_mechanics_tags.h"],
)

cc_library(
name = "proxy",
hdrs = ["proxy.h"],
Expand Down
39 changes: 25 additions & 14 deletions implementation/array_ref.h
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,16 @@
#include "implementation/array.h"
#include "implementation/array_view.h"
#include "implementation/class.h"
#include "implementation/class_ref.h"
#include "implementation/default_class_loader.h"
#include "implementation/forward_declarations.h"
#include "implementation/jni_helper/jni_array_helper.h"
#include "implementation/jni_helper/lifecycle.h"
#include "implementation/jni_helper/lifecycle_object.h"
#include "implementation/jni_type.h"
#include "implementation/local_object.h"
#include "implementation/object_ref.h"
#include "implementation/promotion_mechanics_tags.h"
#include "implementation/ref_base.h"
#include "jni_dep.h"

Expand All @@ -48,7 +51,12 @@ class ArrayRef : public ScopedArrayImpl<JniT> {
using SpanType = typename JniT::SpanType;

ArrayRef(std::size_t size)
: Base(JniArrayHelper<SpanType, JniT::kRank>::NewArray(size)) {}
: Base(AdoptLocal{},
JniArrayHelper<SpanType, JniT::kRank>::NewArray(size)) {}

template <typename T>
ArrayRef(const ArrayViewHelper<T>& array_view_helper)
: Base(AdoptLocal{}, array_view_helper.val) {}

explicit ArrayRef(int size) : ArrayRef(static_cast<std::size_t>(size)) {}

Expand Down Expand Up @@ -79,17 +87,18 @@ class ArrayRefBase : public ScopedArrayImpl<JniT> {

// Construct array with given size and null values.
explicit ArrayRefBase(std::size_t size)
: Base(JniArrayHelper<jobject, JniT::kRank>::NewArray(
size, ClassRef_t<JniT>::GetAndMaybeLoadClassRef(nullptr),
static_cast<jobject>(nullptr))) {}
: Base(AdoptLocal{},
JniArrayHelper<jobject, JniT::kRank>::NewArray(
size, ClassRef_t<JniT>::GetAndMaybeLoadClassRef(nullptr),
static_cast<jobject>(nullptr))) {}

// Construct from jobject lvalue (object is used as template).
explicit ArrayRefBase(std::size_t size, jobject obj)
: Base(JniArrayHelper<jobject, JniT::kRank>::NewArray(
size,
ClassRef_t<JniT>::GetAndMaybeLoadClassRef(
static_cast<jobject>(obj)),
static_cast<jobject>(obj))) {}
: Base(AdoptLocal{}, JniArrayHelper<jobject, JniT::kRank>::NewArray(
size,
ClassRef_t<JniT>::GetAndMaybeLoadClassRef(
static_cast<jobject>(obj)),
static_cast<jobject>(obj))) {}

// Object arrays cannot be efficiently pinned like primitive types can.
ArrayView<SpanType, JniT::kRank> Pin() {
Expand All @@ -108,8 +117,8 @@ class ArrayRefBase : public ScopedArrayImpl<JniT> {
void Set(
std::size_t idx,
LocalObject<JniT::class_v, JniT::class_loader_v, JniT::jvm_v>&& val) {
JniArrayHelper<jobject, JniT::kRank>::SetArrayElement(Base::object_ref_,
idx, val.Release());
AdoptLocal{}, JniArrayHelper<jobject, JniT::kRank>::SetArrayElement(
Base::object_ref_, idx, val.Release());
}
};

Expand Down Expand Up @@ -158,9 +167,11 @@ class ArrayRef<JniT, std::enable_if_t<(JniT::kRank > 1)>>

LocalArray<typename JniT::SpanType, JniT::kRank - 1, clazz, class_loader, jvm>
Get(std::size_t idx) {
return {static_cast<jarray>(
JniArrayHelper<typename JniT::SpanType, JniT::kRank>::GetArrayElement(
Base::object_ref_, idx))};
return {AdoptLocal{},
static_cast<jarray>(
JniArrayHelper<typename JniT::SpanType,
JniT::kRank>::GetArrayElement(Base::object_ref_,
idx))};
}

template <typename SpanType, std::size_t kRank_, const auto& class_v_,
Expand Down
16 changes: 12 additions & 4 deletions implementation/array_view.h
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,14 @@

namespace jni {

template <typename T>
struct ArrayViewHelper {
const T& val_;
operator T() const { return val_; }

ArrayViewHelper(const T& val) : val_(val) {}
};

// Primitive Rank 1 Arrays.
template <typename SpanType, std::size_t kRank = 1, typename Enable = void>
class ArrayView {
Expand Down Expand Up @@ -145,12 +153,12 @@ class ArrayView<
return tmp;
}

PinHelper_t operator*() const {
ArrayViewHelper<PinHelper_t> operator*() const {
if constexpr (kRank >= 2) {
return static_cast<PinHelper_t>(
JniArrayHelper<jobject, kRank>::GetArrayElement(arr_, idx_));
return {static_cast<PinHelper_t>(
JniArrayHelper<jobject, kRank>::GetArrayElement(arr_, idx_))};
} else {
return JniArrayHelper<SpanType, kRank>::GetArrayElement(arr_, idx_);
return {JniArrayHelper<SpanType, kRank>::GetArrayElement(arr_, idx_)};
}
}

Expand Down
31 changes: 16 additions & 15 deletions implementation/array_view_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@

namespace {

using ::jni::AdoptLocal;
using ::jni::ArrayView;
using ::jni::CDecl_t;
using ::jni::Class;
Expand Down Expand Up @@ -95,14 +96,14 @@ TEST_F(JniTest, ArrayView_GetsAndReleaseArrayBuffer) {
Eq(Fake<jdoubleArray>()),
Eq(Fake<jdouble*>()), 0));

LocalArray<jboolean> boolean_array{Fake<jbooleanArray>()};
LocalArray<jbyte> byte_array{Fake<jbyteArray>()};
LocalArray<jchar> char_array{Fake<jcharArray>()};
LocalArray<jshort> short_array{Fake<jshortArray>()};
LocalArray<jint> int_array{Fake<jintArray>()};
LocalArray<jlong> long_array{Fake<jlongArray>()};
LocalArray<jfloat> float_array{Fake<jfloatArray>()};
LocalArray<jdouble> double_array{Fake<jdoubleArray>()};
LocalArray<jboolean> boolean_array{AdoptLocal{}, Fake<jbooleanArray>()};
LocalArray<jbyte> byte_array{AdoptLocal{}, Fake<jbyteArray>()};
LocalArray<jchar> char_array{AdoptLocal{}, Fake<jcharArray>()};
LocalArray<jshort> short_array{AdoptLocal{}, Fake<jshortArray>()};
LocalArray<jint> int_array{AdoptLocal{}, Fake<jintArray>()};
LocalArray<jlong> long_array{AdoptLocal{}, Fake<jlongArray>()};
LocalArray<jfloat> float_array{AdoptLocal{}, Fake<jfloatArray>()};
LocalArray<jdouble> double_array{AdoptLocal{}, Fake<jdoubleArray>()};

ArrayView<jboolean, 1> boolean_array_pin = {boolean_array.Pin()};
ArrayView<jbyte, 1> byte_array_pin = {byte_array.Pin()};
Expand All @@ -121,7 +122,7 @@ TEST_F(JniTest, LocalArrayView_AllowsCTAD) {
Eq(Fake<jbooleanArray>()),
Eq(Fake<jboolean*>()), 0));

LocalArray<jboolean> boolean_array{Fake<jbooleanArray>()};
LocalArray<jboolean> boolean_array{AdoptLocal{}, Fake<jbooleanArray>()};
ArrayView ctad_array_view{boolean_array.Pin()};

// Despite supporting construction from xvalue, move ctor is deleted (good).
Expand Down Expand Up @@ -300,7 +301,7 @@ TEST_F(JniTest, ArrayView_ShallowObjectsAreIterable) {
.WillOnce(::testing::Return(Fake<jobject>(2)))
.WillOnce(::testing::Return(Fake<jobject>(3)));

LocalArray<jobject> obj_arr{Fake<jobjectArray>()};
LocalArray<jobject> obj_arr{AdoptLocal{}, Fake<jobjectArray>()};
ArrayView<jobject, 1> obj_view = obj_arr.Pin();

EXPECT_TRUE(std::equal(obj_view.begin(), obj_view.end(), fake_vals.begin(),
Expand All @@ -318,7 +319,7 @@ TEST_F(JniTest, ArrayView_RichObjectsAreIterable) {
.WillOnce(::testing::Return(Fake<jobject>(2)))
.WillOnce(::testing::Return(Fake<jobject>(3)));

LocalArray<jobject, 1, kClass> obj_arr{Fake<jobjectArray>()};
LocalArray<jobject, 1, kClass> obj_arr{AdoptLocal{}, Fake<jobjectArray>()};
auto obj_view = obj_arr.Pin();

// Note: GlobalObject will fail to compile here. This is good, the user
Expand Down Expand Up @@ -350,7 +351,7 @@ TEST_F(JniTest, ArrayView_Rank2IntArraysAreIterable) {
EXPECT_CALL(*env_, GetObjectArrayElement(Fake<jobjectArray>(), 2))
.WillOnce(::testing::Return(Fake<jintArray>(3)));

LocalArray<jint, 2> int_arr_rank_2{Fake<jobjectArray>()};
LocalArray<jint, 2> int_arr_rank_2{AdoptLocal{}, Fake<jobjectArray>()};
ArrayView<jint, 2> int_rank2_view = int_arr_rank_2.Pin();

EXPECT_TRUE(std::equal(int_rank2_view.begin(), int_rank2_view.end(),
Expand All @@ -375,7 +376,7 @@ TEST_F(JniTest, ArrayView_Rank2ObjectkArraysAreIterable) {
EXPECT_CALL(*env_, GetObjectArrayElement(Fake<jobjectArray>(), 2))
.WillOnce(::testing::Return(Fake<jobjectArray>(3)));

LocalArray<jobject, 2> int_arr_rank_2{Fake<jobjectArray>()};
LocalArray<jobject, 2> int_arr_rank_2{AdoptLocal{}, Fake<jobjectArray>()};
ArrayView<jobject, 2> int_rank2_view = int_arr_rank_2.Pin();

EXPECT_TRUE(std::equal(int_rank2_view.begin(), int_rank2_view.end(),
Expand Down Expand Up @@ -403,7 +404,7 @@ TEST_F(JniTest, ArrayView_Rank3IntArraysAreIterable) {
EXPECT_CALL(*env_, GetObjectArrayElement(Fake<jobjectArray>(), 2))
.WillOnce(::testing::Return(Fake<jobjectArray>()));

LocalArray<jint, 3> int_arr_rank_3{Fake<jobjectArray>()};
LocalArray<jint, 3> int_arr_rank_3{AdoptLocal{}, Fake<jobjectArray>()};
ArrayView<jint, 3> int_rank_3_view = int_arr_rank_3.Pin();

EXPECT_TRUE(std::equal(int_rank_3_view.begin(), int_rank_3_view.end(),
Expand All @@ -426,7 +427,7 @@ TEST_F(JniTest, ArrayView_Rank3ObjectkArraysAreIterable) {
EXPECT_CALL(*env_, GetObjectArrayElement(Fake<jobjectArray>(0), 2))
.WillOnce(::testing::Return(Fake<jobjectArray>(3)));

LocalArray<jobject, 3> object_arr_rank_3{Fake<jobjectArray>(0)};
LocalArray<jobject, 3> object_arr_rank_3{AdoptLocal{}, Fake<jobjectArray>(0)};
ArrayView<jobject, 3> object_rank_3_view = object_arr_rank_3.Pin();

EXPECT_TRUE(std::equal(object_rank_3_view.begin(), object_rank_3_view.end(),
Expand Down
3 changes: 2 additions & 1 deletion implementation/class_loader_ref_second_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@

namespace {

using ::jni::AdoptLocal;
using ::jni::Class;
using ::jni::ClassLoader;
using ::jni::Constructor;
Expand Down Expand Up @@ -83,7 +84,7 @@ TEST_F(JniTestWithNoDefaultJvmRef,
// Code under test.
JvmRef<kClassLoaderJvm> jvm_ref{jvm_.get()};
LocalClassLoader<kClassLoader, kClassLoaderJvm> class_loader{
Fake<jobject>(1)};
AdoptLocal{}, Fake<jobject>(1)};

auto custom_loader_object = class_loader.BuildLocalObject<kClass>(jint{1});

Expand Down
6 changes: 4 additions & 2 deletions implementation/class_loader_ref_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
namespace {

using ::jni::AdoptGlobal;
using ::jni::AdoptLocal;
using ::jni::Class;
using ::jni::ClassLoader;
using ::jni::Constructor;
Expand Down Expand Up @@ -157,7 +158,8 @@ TEST_F(JniTestForClassLoaders,
ClassLoaderRefTest_DefaultLoadedObjectBuildsWithClassLoadedObject) {
JvmRef<kJvm> jvm_ref{jvm_.get()};

LocalClassLoader<kClassLoader, kJvm> local_class_loader{Fake<jobject>()};
LocalClassLoader<kClassLoader, kJvm> local_class_loader{AdoptLocal{},
Fake<jobject>()};
LocalObject<kClass1, kClassLoader> a =
local_class_loader.BuildLocalObject<kClass1>();
LocalObject<kClass2> b{a};
Expand Down Expand Up @@ -308,7 +310,7 @@ TEST_F(JniTestWithNoDefaultJvmRef,
// Code under test.
jni::JvmRef<atypical_jvm_definition> jvm_ref{jvm_.get()};
jni::LocalObject<class_under_test, class_loader, atypical_jvm_definition>
obj1{Fake<jobject>(1)};
obj1{AdoptLocal{}, Fake<jobject>(1)};
obj1("Foo");

this->TearDown();
Expand Down
Loading

0 comments on commit 12db7ae

Please sign in to comment.