1*795d594fSAndroid Build Coastguard Worker /*
2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2022 The Android Open Source Project
3*795d594fSAndroid Build Coastguard Worker *
4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*795d594fSAndroid Build Coastguard Worker *
8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*795d594fSAndroid Build Coastguard Worker *
10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*795d594fSAndroid Build Coastguard Worker * limitations under the License.
15*795d594fSAndroid Build Coastguard Worker */
16*795d594fSAndroid Build Coastguard Worker
17*795d594fSAndroid Build Coastguard Worker #include "local_reference_table-inl.h"
18*795d594fSAndroid Build Coastguard Worker
19*795d594fSAndroid Build Coastguard Worker #include "android-base/stringprintf.h"
20*795d594fSAndroid Build Coastguard Worker
21*795d594fSAndroid Build Coastguard Worker #include "class_root-inl.h"
22*795d594fSAndroid Build Coastguard Worker #include "common_runtime_test.h"
23*795d594fSAndroid Build Coastguard Worker #include "mirror/class-alloc-inl.h"
24*795d594fSAndroid Build Coastguard Worker #include "mirror/object-inl.h"
25*795d594fSAndroid Build Coastguard Worker #include "scoped_thread_state_change-inl.h"
26*795d594fSAndroid Build Coastguard Worker
27*795d594fSAndroid Build Coastguard Worker namespace art HIDDEN {
28*795d594fSAndroid Build Coastguard Worker namespace jni {
29*795d594fSAndroid Build Coastguard Worker
30*795d594fSAndroid Build Coastguard Worker using android::base::StringPrintf;
31*795d594fSAndroid Build Coastguard Worker
32*795d594fSAndroid Build Coastguard Worker class LocalReferenceTableTest : public CommonRuntimeTest {
33*795d594fSAndroid Build Coastguard Worker protected:
LocalReferenceTableTest()34*795d594fSAndroid Build Coastguard Worker LocalReferenceTableTest() {
35*795d594fSAndroid Build Coastguard Worker use_boot_image_ = true; // Make the Runtime creation cheaper.
36*795d594fSAndroid Build Coastguard Worker }
37*795d594fSAndroid Build Coastguard Worker
38*795d594fSAndroid Build Coastguard Worker static void CheckDump(LocalReferenceTable* lrt, size_t num_objects, size_t num_unique)
39*795d594fSAndroid Build Coastguard Worker REQUIRES_SHARED(Locks::mutator_lock_);
40*795d594fSAndroid Build Coastguard Worker
41*795d594fSAndroid Build Coastguard Worker void BasicTest(bool check_jni, size_t max_count);
42*795d594fSAndroid Build Coastguard Worker void BasicHolesTest(bool check_jni, size_t max_count);
43*795d594fSAndroid Build Coastguard Worker void BasicResizeTest(bool check_jni, size_t max_count);
44*795d594fSAndroid Build Coastguard Worker void TestAddRemove(bool check_jni, size_t max_count, size_t fill_count = 0u);
45*795d594fSAndroid Build Coastguard Worker void TestAddRemoveMixed(bool start_check_jni);
46*795d594fSAndroid Build Coastguard Worker };
47*795d594fSAndroid Build Coastguard Worker
CheckDump(LocalReferenceTable * lrt,size_t num_objects,size_t num_unique)48*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::CheckDump(
49*795d594fSAndroid Build Coastguard Worker LocalReferenceTable* lrt, size_t num_objects, size_t num_unique) {
50*795d594fSAndroid Build Coastguard Worker std::ostringstream oss;
51*795d594fSAndroid Build Coastguard Worker lrt->Dump(oss);
52*795d594fSAndroid Build Coastguard Worker if (num_objects == 0) {
53*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(oss.str().find("java.lang.Object"), std::string::npos) << oss.str();
54*795d594fSAndroid Build Coastguard Worker } else if (num_objects == 1) {
55*795d594fSAndroid Build Coastguard Worker EXPECT_NE(oss.str().find("1 of java.lang.Object"), std::string::npos) << oss.str();
56*795d594fSAndroid Build Coastguard Worker } else {
57*795d594fSAndroid Build Coastguard Worker EXPECT_NE(oss.str().find(StringPrintf("%zd of java.lang.Object (%zd unique instances)",
58*795d594fSAndroid Build Coastguard Worker num_objects, num_unique)),
59*795d594fSAndroid Build Coastguard Worker std::string::npos)
60*795d594fSAndroid Build Coastguard Worker << "\n Expected number of objects: " << num_objects
61*795d594fSAndroid Build Coastguard Worker << "\n Expected unique objects: " << num_unique << "\n"
62*795d594fSAndroid Build Coastguard Worker << oss.str();
63*795d594fSAndroid Build Coastguard Worker }
64*795d594fSAndroid Build Coastguard Worker }
65*795d594fSAndroid Build Coastguard Worker
BasicTest(bool check_jni,size_t max_count)66*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::BasicTest(bool check_jni, size_t max_count) {
67*795d594fSAndroid Build Coastguard Worker // This will lead to error messages in the log.
68*795d594fSAndroid Build Coastguard Worker ScopedLogSeverity sls(LogSeverity::FATAL);
69*795d594fSAndroid Build Coastguard Worker
70*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
71*795d594fSAndroid Build Coastguard Worker StackHandleScope<5> hs(soa.Self());
72*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> c = hs.NewHandle(GetClassRoot<mirror::Object>());
73*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(c != nullptr);
74*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj0 = hs.NewHandle(c->AllocObject(soa.Self()));
75*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj0 != nullptr);
76*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj1 = hs.NewHandle(c->AllocObject(soa.Self()));
77*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj1 != nullptr);
78*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj2 = hs.NewHandle(c->AllocObject(soa.Self()));
79*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj2 != nullptr);
80*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj3 = hs.NewHandle(c->AllocObject(soa.Self()));
81*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj3 != nullptr);
82*795d594fSAndroid Build Coastguard Worker
83*795d594fSAndroid Build Coastguard Worker std::string error_msg;
84*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
85*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
86*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
87*795d594fSAndroid Build Coastguard Worker
88*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
89*795d594fSAndroid Build Coastguard Worker
90*795d594fSAndroid Build Coastguard Worker if (check_jni) {
91*795d594fSAndroid Build Coastguard Worker IndirectRef bad_iref = (IndirectRef) 0x11110;
92*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.Remove(bad_iref)) << "unexpectedly successful removal";
93*795d594fSAndroid Build Coastguard Worker }
94*795d594fSAndroid Build Coastguard Worker
95*795d594fSAndroid Build Coastguard Worker // Add three, check, remove in the order in which they were added.
96*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
97*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
98*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
99*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
100*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
101*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
102*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj2.Get(), &error_msg);
103*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref2 != nullptr);
104*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
105*795d594fSAndroid Build Coastguard Worker
106*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ(obj0.Get(), lrt.Get(iref0));
107*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ(obj1.Get(), lrt.Get(iref1));
108*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ(obj2.Get(), lrt.Get(iref2));
109*795d594fSAndroid Build Coastguard Worker
110*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref0));
111*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
112*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref1));
113*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
114*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref2));
115*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
116*795d594fSAndroid Build Coastguard Worker
117*795d594fSAndroid Build Coastguard Worker // Table should be empty now.
118*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(0U, lrt.Capacity());
119*795d594fSAndroid Build Coastguard Worker
120*795d594fSAndroid Build Coastguard Worker // Check that the entry off the end of the list is not valid.
121*795d594fSAndroid Build Coastguard Worker // (CheckJNI shall abort for such entries.)
122*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref0, &error_msg));
123*795d594fSAndroid Build Coastguard Worker
124*795d594fSAndroid Build Coastguard Worker // Add three, remove in the opposite order.
125*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
126*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
127*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj1.Get(), &error_msg);
128*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
129*795d594fSAndroid Build Coastguard Worker iref2 = lrt.Add(obj2.Get(), &error_msg);
130*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref2 != nullptr);
131*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
132*795d594fSAndroid Build Coastguard Worker
133*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref2));
134*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
135*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1));
136*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
137*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
138*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
139*795d594fSAndroid Build Coastguard Worker
140*795d594fSAndroid Build Coastguard Worker // Table should be empty now.
141*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity());
142*795d594fSAndroid Build Coastguard Worker
143*795d594fSAndroid Build Coastguard Worker // Add three, remove middle / middle / bottom / top. (Second attempt
144*795d594fSAndroid Build Coastguard Worker // to remove middle should fail.)
145*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
146*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
147*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj1.Get(), &error_msg);
148*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
149*795d594fSAndroid Build Coastguard Worker iref2 = lrt.Add(obj2.Get(), &error_msg);
150*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref2 != nullptr);
151*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
152*795d594fSAndroid Build Coastguard Worker
153*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(3U, lrt.Capacity());
154*795d594fSAndroid Build Coastguard Worker
155*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1));
156*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
157*795d594fSAndroid Build Coastguard Worker if (check_jni) {
158*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref1));
159*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
160*795d594fSAndroid Build Coastguard Worker }
161*795d594fSAndroid Build Coastguard Worker
162*795d594fSAndroid Build Coastguard Worker // Check that the reference to the hole is not valid.
163*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref1, &error_msg));
164*795d594fSAndroid Build Coastguard Worker
165*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref2));
166*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
167*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
168*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
169*795d594fSAndroid Build Coastguard Worker
170*795d594fSAndroid Build Coastguard Worker // Table should be empty now.
171*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity());
172*795d594fSAndroid Build Coastguard Worker
173*795d594fSAndroid Build Coastguard Worker // Add four entries. Remove #1, add new entry, verify that table size
174*795d594fSAndroid Build Coastguard Worker // is still 4 (i.e. holes are getting filled). Remove #1 and #3, verify
175*795d594fSAndroid Build Coastguard Worker // that we delete one and don't hole-compact the other.
176*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
177*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
178*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj1.Get(), &error_msg);
179*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
180*795d594fSAndroid Build Coastguard Worker iref2 = lrt.Add(obj2.Get(), &error_msg);
181*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref2 != nullptr);
182*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj3.Get(), &error_msg);
183*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref3 != nullptr);
184*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 4, 4);
185*795d594fSAndroid Build Coastguard Worker
186*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1));
187*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
188*795d594fSAndroid Build Coastguard Worker
189*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj1.Get(), &error_msg);
190*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
191*795d594fSAndroid Build Coastguard Worker
192*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(4U, lrt.Capacity()) << "hole not filled";
193*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 4, 4);
194*795d594fSAndroid Build Coastguard Worker
195*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1));
196*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
197*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref3));
198*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
199*795d594fSAndroid Build Coastguard Worker
200*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(3U, lrt.Capacity()) << "should be 3 after two deletions";
201*795d594fSAndroid Build Coastguard Worker
202*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref2));
203*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
204*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
205*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
206*795d594fSAndroid Build Coastguard Worker
207*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity()) << "not empty after split remove";
208*795d594fSAndroid Build Coastguard Worker
209*795d594fSAndroid Build Coastguard Worker // Add an entry, remove it, add a new entry, and try to use the original
210*795d594fSAndroid Build Coastguard Worker // iref. They have the same slot number but are for different objects.
211*795d594fSAndroid Build Coastguard Worker // With the extended checks in place, this should fail.
212*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
213*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
214*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
215*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
216*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
217*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj1.Get(), &error_msg);
218*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
219*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
220*795d594fSAndroid Build Coastguard Worker if (check_jni) {
221*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref0)) << "mismatched del succeeded";
222*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
223*795d594fSAndroid Build Coastguard Worker }
224*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1)) << "switched del failed";
225*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity()) << "switching del not empty";
226*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
227*795d594fSAndroid Build Coastguard Worker
228*795d594fSAndroid Build Coastguard Worker // Same as above, but with the same object. A more rigorous checker
229*795d594fSAndroid Build Coastguard Worker // (e.g. with slot serialization) will catch this.
230*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
231*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
232*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
233*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
234*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
235*795d594fSAndroid Build Coastguard Worker iref1 = lrt.Add(obj0.Get(), &error_msg);
236*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref1 != nullptr);
237*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
238*795d594fSAndroid Build Coastguard Worker if (iref0 != iref1) {
239*795d594fSAndroid Build Coastguard Worker // Try 0, should not work.
240*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref0)) << "temporal del succeeded";
241*795d594fSAndroid Build Coastguard Worker }
242*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref1)) << "temporal cleanup failed";
243*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity()) << "temporal del not empty";
244*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
245*795d594fSAndroid Build Coastguard Worker
246*795d594fSAndroid Build Coastguard Worker // Stale reference is not valid.
247*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
248*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(iref0 != nullptr);
249*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
250*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0));
251*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref0, &error_msg)) << "stale lookup succeeded";
252*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
253*795d594fSAndroid Build Coastguard Worker
254*795d594fSAndroid Build Coastguard Worker // Test table resizing.
255*795d594fSAndroid Build Coastguard Worker // These ones fit...
256*795d594fSAndroid Build Coastguard Worker static const size_t kTableInitial = max_count / 2;
257*795d594fSAndroid Build Coastguard Worker IndirectRef manyRefs[kTableInitial];
258*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i < kTableInitial; i++) {
259*795d594fSAndroid Build Coastguard Worker manyRefs[i] = lrt.Add(obj0.Get(), &error_msg);
260*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(manyRefs[i] != nullptr) << "Failed adding " << i;
261*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, i + 1, 1);
262*795d594fSAndroid Build Coastguard Worker }
263*795d594fSAndroid Build Coastguard Worker // ...this one causes overflow.
264*795d594fSAndroid Build Coastguard Worker iref0 = lrt.Add(obj0.Get(), &error_msg);
265*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(iref0 != nullptr);
266*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(kTableInitial + 1, lrt.Capacity());
267*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, kTableInitial + 1, 1);
268*795d594fSAndroid Build Coastguard Worker
269*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i < kTableInitial; i++) {
270*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(manyRefs[i])) << "failed removing " << i;
271*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, kTableInitial - i, 1);
272*795d594fSAndroid Build Coastguard Worker }
273*795d594fSAndroid Build Coastguard Worker // Because of removal order, should have 11 entries, 10 of them holes.
274*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(kTableInitial + 1, lrt.Capacity());
275*795d594fSAndroid Build Coastguard Worker
276*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref0)) << "multi-remove final failed";
277*795d594fSAndroid Build Coastguard Worker
278*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(0U, lrt.Capacity()) << "multi-del not empty";
279*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
280*795d594fSAndroid Build Coastguard Worker }
281*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicTest)282*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicTest) {
283*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ false, /*max_count=*/ 20u);
284*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ false, /*max_count=*/ kSmallLrtEntries);
285*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ false, /*max_count=*/ 2u * kSmallLrtEntries);
286*795d594fSAndroid Build Coastguard Worker }
287*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicTestCheckJNI)288*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicTestCheckJNI) {
289*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ true, /*max_count=*/ 20u);
290*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ true, /*max_count=*/ kSmallLrtEntries);
291*795d594fSAndroid Build Coastguard Worker BasicTest(/*check_jni=*/ true, /*max_count=*/ 2u * kSmallLrtEntries);
292*795d594fSAndroid Build Coastguard Worker }
293*795d594fSAndroid Build Coastguard Worker
BasicHolesTest(bool check_jni,size_t max_count)294*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::BasicHolesTest(bool check_jni, size_t max_count) {
295*795d594fSAndroid Build Coastguard Worker // Test the explicitly named cases from the LRT implementation:
296*795d594fSAndroid Build Coastguard Worker //
297*795d594fSAndroid Build Coastguard Worker // 1) Segment with holes (current_num_holes_ > 0), push new segment, add/remove reference
298*795d594fSAndroid Build Coastguard Worker // 2) Segment with holes (current_num_holes_ > 0), pop segment, add/remove reference
299*795d594fSAndroid Build Coastguard Worker // 3) Segment with holes (current_num_holes_ > 0), push new segment, pop segment, add/remove
300*795d594fSAndroid Build Coastguard Worker // reference
301*795d594fSAndroid Build Coastguard Worker // 4) Empty segment, push new segment, create a hole, pop a segment, add/remove a reference
302*795d594fSAndroid Build Coastguard Worker // 5) Base segment, push new segment, create a hole, pop a segment, push new segment, add/remove
303*795d594fSAndroid Build Coastguard Worker // reference
304*795d594fSAndroid Build Coastguard Worker
305*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
306*795d594fSAndroid Build Coastguard Worker StackHandleScope<6> hs(soa.Self());
307*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> c = hs.NewHandle(GetClassRoot<mirror::Object>());
308*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(c != nullptr);
309*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj0 = hs.NewHandle(c->AllocObject(soa.Self()));
310*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj0 != nullptr);
311*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj1 = hs.NewHandle(c->AllocObject(soa.Self()));
312*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj1 != nullptr);
313*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj2 = hs.NewHandle(c->AllocObject(soa.Self()));
314*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj2 != nullptr);
315*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj3 = hs.NewHandle(c->AllocObject(soa.Self()));
316*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj3 != nullptr);
317*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj4 = hs.NewHandle(c->AllocObject(soa.Self()));
318*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj4 != nullptr);
319*795d594fSAndroid Build Coastguard Worker
320*795d594fSAndroid Build Coastguard Worker std::string error_msg;
321*795d594fSAndroid Build Coastguard Worker
322*795d594fSAndroid Build Coastguard Worker // 1) Segment with holes (current_num_holes_ > 0), push new segment, add/remove reference.
323*795d594fSAndroid Build Coastguard Worker {
324*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
325*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
326*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
327*795d594fSAndroid Build Coastguard Worker
328*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
329*795d594fSAndroid Build Coastguard Worker
330*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
331*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
332*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj2.Get(), &error_msg);
333*795d594fSAndroid Build Coastguard Worker
334*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref1));
335*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 3u);
336*795d594fSAndroid Build Coastguard Worker
337*795d594fSAndroid Build Coastguard Worker // New segment.
338*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie = lrt.PushFrame();
339*795d594fSAndroid Build Coastguard Worker
340*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj3.Get(), &error_msg);
341*795d594fSAndroid Build Coastguard Worker
342*795d594fSAndroid Build Coastguard Worker // Must not have filled the previous hole.
343*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 4u);
344*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref1, &error_msg));
345*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
346*795d594fSAndroid Build Coastguard Worker
347*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie);
348*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 3u);
349*795d594fSAndroid Build Coastguard Worker
350*795d594fSAndroid Build Coastguard Worker UNUSED(iref0, iref1, iref2, iref3);
351*795d594fSAndroid Build Coastguard Worker }
352*795d594fSAndroid Build Coastguard Worker
353*795d594fSAndroid Build Coastguard Worker // 2) Segment with holes (current_num_holes_ > 0), pop segment, add/remove reference
354*795d594fSAndroid Build Coastguard Worker {
355*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
356*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
357*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
358*795d594fSAndroid Build Coastguard Worker
359*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
360*795d594fSAndroid Build Coastguard Worker
361*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
362*795d594fSAndroid Build Coastguard Worker
363*795d594fSAndroid Build Coastguard Worker // New segment.
364*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie = lrt.PushFrame();
365*795d594fSAndroid Build Coastguard Worker
366*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
367*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj2.Get(), &error_msg);
368*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj3.Get(), &error_msg);
369*795d594fSAndroid Build Coastguard Worker
370*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref2));
371*795d594fSAndroid Build Coastguard Worker
372*795d594fSAndroid Build Coastguard Worker // Pop segment.
373*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie);
374*795d594fSAndroid Build Coastguard Worker
375*795d594fSAndroid Build Coastguard Worker IndirectRef iref4 = lrt.Add(obj4.Get(), &error_msg);
376*795d594fSAndroid Build Coastguard Worker
377*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 2u);
378*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref2, &error_msg));
379*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
380*795d594fSAndroid Build Coastguard Worker
381*795d594fSAndroid Build Coastguard Worker UNUSED(iref0, iref1, iref2, iref3, iref4);
382*795d594fSAndroid Build Coastguard Worker }
383*795d594fSAndroid Build Coastguard Worker
384*795d594fSAndroid Build Coastguard Worker // 3) Segment with holes (current_num_holes_ > 0), push new segment, pop segment, add/remove
385*795d594fSAndroid Build Coastguard Worker // reference.
386*795d594fSAndroid Build Coastguard Worker {
387*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
388*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
389*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
390*795d594fSAndroid Build Coastguard Worker
391*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
392*795d594fSAndroid Build Coastguard Worker
393*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
394*795d594fSAndroid Build Coastguard Worker
395*795d594fSAndroid Build Coastguard Worker // New segment.
396*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0 = lrt.PushFrame();
397*795d594fSAndroid Build Coastguard Worker
398*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
399*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj2.Get(), &error_msg);
400*795d594fSAndroid Build Coastguard Worker
401*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref1));
402*795d594fSAndroid Build Coastguard Worker
403*795d594fSAndroid Build Coastguard Worker // New segment.
404*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie1 = lrt.PushFrame();
405*795d594fSAndroid Build Coastguard Worker
406*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj3.Get(), &error_msg);
407*795d594fSAndroid Build Coastguard Worker
408*795d594fSAndroid Build Coastguard Worker // Pop segment.
409*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
410*795d594fSAndroid Build Coastguard Worker
411*795d594fSAndroid Build Coastguard Worker IndirectRef iref4 = lrt.Add(obj4.Get(), &error_msg);
412*795d594fSAndroid Build Coastguard Worker
413*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 3u);
414*795d594fSAndroid Build Coastguard Worker if (check_jni) {
415*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref1, &error_msg));
416*795d594fSAndroid Build Coastguard Worker }
417*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 3, 3);
418*795d594fSAndroid Build Coastguard Worker
419*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
420*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
421*795d594fSAndroid Build Coastguard Worker
422*795d594fSAndroid Build Coastguard Worker UNUSED(iref0, iref1, iref2, iref3, iref4);
423*795d594fSAndroid Build Coastguard Worker }
424*795d594fSAndroid Build Coastguard Worker
425*795d594fSAndroid Build Coastguard Worker // 4) Empty segment, push new segment, create a hole, pop a segment, add/remove a reference.
426*795d594fSAndroid Build Coastguard Worker {
427*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
428*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
429*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
430*795d594fSAndroid Build Coastguard Worker
431*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
432*795d594fSAndroid Build Coastguard Worker
433*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
434*795d594fSAndroid Build Coastguard Worker
435*795d594fSAndroid Build Coastguard Worker // New segment.
436*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0 = lrt.PushFrame();
437*795d594fSAndroid Build Coastguard Worker
438*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
439*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref1));
440*795d594fSAndroid Build Coastguard Worker
441*795d594fSAndroid Build Coastguard Worker // Emptied segment, push new one.
442*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie1 = lrt.PushFrame();
443*795d594fSAndroid Build Coastguard Worker
444*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj1.Get(), &error_msg);
445*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj2.Get(), &error_msg);
446*795d594fSAndroid Build Coastguard Worker IndirectRef iref4 = lrt.Add(obj3.Get(), &error_msg);
447*795d594fSAndroid Build Coastguard Worker
448*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref3));
449*795d594fSAndroid Build Coastguard Worker
450*795d594fSAndroid Build Coastguard Worker // Pop segment.
451*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
452*795d594fSAndroid Build Coastguard Worker
453*795d594fSAndroid Build Coastguard Worker IndirectRef iref5 = lrt.Add(obj4.Get(), &error_msg);
454*795d594fSAndroid Build Coastguard Worker
455*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 2u);
456*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref3, &error_msg));
457*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
458*795d594fSAndroid Build Coastguard Worker
459*795d594fSAndroid Build Coastguard Worker // Pop segment.
460*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
461*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 1, 1);
462*795d594fSAndroid Build Coastguard Worker
463*795d594fSAndroid Build Coastguard Worker UNUSED(iref0, iref1, iref2, iref3, iref4, iref5);
464*795d594fSAndroid Build Coastguard Worker }
465*795d594fSAndroid Build Coastguard Worker
466*795d594fSAndroid Build Coastguard Worker // 5) Base segment, push new segment, create a hole, pop a segment, push new segment, add/remove
467*795d594fSAndroid Build Coastguard Worker // reference
468*795d594fSAndroid Build Coastguard Worker {
469*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
470*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
471*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
472*795d594fSAndroid Build Coastguard Worker
473*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
474*795d594fSAndroid Build Coastguard Worker
475*795d594fSAndroid Build Coastguard Worker IndirectRef iref0 = lrt.Add(obj0.Get(), &error_msg);
476*795d594fSAndroid Build Coastguard Worker
477*795d594fSAndroid Build Coastguard Worker // New segment.
478*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0 = lrt.PushFrame();
479*795d594fSAndroid Build Coastguard Worker
480*795d594fSAndroid Build Coastguard Worker IndirectRef iref1 = lrt.Add(obj1.Get(), &error_msg);
481*795d594fSAndroid Build Coastguard Worker IndirectRef iref2 = lrt.Add(obj1.Get(), &error_msg);
482*795d594fSAndroid Build Coastguard Worker IndirectRef iref3 = lrt.Add(obj2.Get(), &error_msg);
483*795d594fSAndroid Build Coastguard Worker
484*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(iref2));
485*795d594fSAndroid Build Coastguard Worker
486*795d594fSAndroid Build Coastguard Worker // Pop segment.
487*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
488*795d594fSAndroid Build Coastguard Worker
489*795d594fSAndroid Build Coastguard Worker // Push segment.
490*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0_second = lrt.PushFrame();
491*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(cookie0.top_index, cookie0_second.top_index);
492*795d594fSAndroid Build Coastguard Worker
493*795d594fSAndroid Build Coastguard Worker IndirectRef iref4 = lrt.Add(obj3.Get(), &error_msg);
494*795d594fSAndroid Build Coastguard Worker
495*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), 2u);
496*795d594fSAndroid Build Coastguard Worker EXPECT_FALSE(lrt.IsValidReference(iref3, &error_msg));
497*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 2, 2);
498*795d594fSAndroid Build Coastguard Worker
499*795d594fSAndroid Build Coastguard Worker UNUSED(iref0, iref1, iref2, iref3, iref4);
500*795d594fSAndroid Build Coastguard Worker }
501*795d594fSAndroid Build Coastguard Worker }
502*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicHolesTest)503*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicHolesTest) {
504*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ false, 20u);
505*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ false, /*max_count=*/ kSmallLrtEntries);
506*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ false, /*max_count=*/ 2u * kSmallLrtEntries);
507*795d594fSAndroid Build Coastguard Worker }
508*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicHolesTestCheckJNI)509*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicHolesTestCheckJNI) {
510*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ true, 20u);
511*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ true, /*max_count=*/ kSmallLrtEntries);
512*795d594fSAndroid Build Coastguard Worker BasicHolesTest(/*check_jni=*/ true, /*max_count=*/ 2u * kSmallLrtEntries);
513*795d594fSAndroid Build Coastguard Worker }
514*795d594fSAndroid Build Coastguard Worker
BasicResizeTest(bool check_jni,size_t max_count)515*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::BasicResizeTest(bool check_jni, size_t max_count) {
516*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
517*795d594fSAndroid Build Coastguard Worker StackHandleScope<2> hs(soa.Self());
518*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> c = hs.NewHandle(GetClassRoot<mirror::Object>());
519*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(c != nullptr);
520*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj0 = hs.NewHandle(c->AllocObject(soa.Self()));
521*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj0 != nullptr);
522*795d594fSAndroid Build Coastguard Worker
523*795d594fSAndroid Build Coastguard Worker std::string error_msg;
524*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
525*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
526*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
527*795d594fSAndroid Build Coastguard Worker
528*795d594fSAndroid Build Coastguard Worker CheckDump(&lrt, 0, 0);
529*795d594fSAndroid Build Coastguard Worker
530*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != max_count + 1; ++i) {
531*795d594fSAndroid Build Coastguard Worker lrt.Add(obj0.Get(), &error_msg);
532*795d594fSAndroid Build Coastguard Worker }
533*795d594fSAndroid Build Coastguard Worker
534*795d594fSAndroid Build Coastguard Worker EXPECT_EQ(lrt.Capacity(), max_count + 1);
535*795d594fSAndroid Build Coastguard Worker }
536*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicResizeTest)537*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicResizeTest) {
538*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ false, 20u);
539*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ false, /*max_count=*/ kSmallLrtEntries);
540*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ false, /*max_count=*/ 2u * kSmallLrtEntries);
541*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ false, /*max_count=*/ gPageSize / sizeof(LrtEntry));
542*795d594fSAndroid Build Coastguard Worker }
543*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,BasicResizeTestCheckJNI)544*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, BasicResizeTestCheckJNI) {
545*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ true, 20u);
546*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ true, /*max_count=*/ kSmallLrtEntries);
547*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ true, /*max_count=*/ 2u * kSmallLrtEntries);
548*795d594fSAndroid Build Coastguard Worker BasicResizeTest(/*check_jni=*/ true, /*max_count=*/ gPageSize / sizeof(LrtEntry));
549*795d594fSAndroid Build Coastguard Worker }
550*795d594fSAndroid Build Coastguard Worker
TestAddRemove(bool check_jni,size_t max_count,size_t fill_count)551*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::TestAddRemove(bool check_jni, size_t max_count, size_t fill_count) {
552*795d594fSAndroid Build Coastguard Worker // This will lead to error messages in the log.
553*795d594fSAndroid Build Coastguard Worker ScopedLogSeverity sls(LogSeverity::FATAL);
554*795d594fSAndroid Build Coastguard Worker
555*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
556*795d594fSAndroid Build Coastguard Worker StackHandleScope<9> hs(soa.Self());
557*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> c = hs.NewHandle(GetClassRoot<mirror::Object>());
558*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(c != nullptr);
559*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj0 = hs.NewHandle(c->AllocObject(soa.Self()));
560*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj0 != nullptr);
561*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj0x = hs.NewHandle(c->AllocObject(soa.Self()));
562*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj0x != nullptr);
563*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj1 = hs.NewHandle(c->AllocObject(soa.Self()));
564*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj1 != nullptr);
565*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj1x = hs.NewHandle(c->AllocObject(soa.Self()));
566*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj1x != nullptr);
567*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj2 = hs.NewHandle(c->AllocObject(soa.Self()));
568*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj2 != nullptr);
569*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj2x = hs.NewHandle(c->AllocObject(soa.Self()));
570*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj2x != nullptr);
571*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj3 = hs.NewHandle(c->AllocObject(soa.Self()));
572*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj3 != nullptr);
573*795d594fSAndroid Build Coastguard Worker Handle<mirror::Object> obj3x = hs.NewHandle(c->AllocObject(soa.Self()));
574*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(obj3x != nullptr);
575*795d594fSAndroid Build Coastguard Worker
576*795d594fSAndroid Build Coastguard Worker std::string error_msg;
577*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(check_jni);
578*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(max_count, &error_msg);
579*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
580*795d594fSAndroid Build Coastguard Worker
581*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != fill_count; ++i) {
582*795d594fSAndroid Build Coastguard Worker IndirectRef iref = lrt.Add(c.Get(), &error_msg);
583*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(iref != nullptr) << error_msg;
584*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(i + 1u, lrt.Capacity());
585*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ(c.Get(), lrt.Get(iref));
586*795d594fSAndroid Build Coastguard Worker }
587*795d594fSAndroid Build Coastguard Worker
588*795d594fSAndroid Build Coastguard Worker IndirectRef iref0, iref1, iref2, iref3;
589*795d594fSAndroid Build Coastguard Worker
590*795d594fSAndroid Build Coastguard Worker #define ADD_REF(iref, obj, expected_capacity) \
591*795d594fSAndroid Build Coastguard Worker do { \
592*795d594fSAndroid Build Coastguard Worker (iref) = lrt.Add((obj).Get(), &error_msg); \
593*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE((iref) != nullptr) << error_msg; \
594*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(fill_count + (expected_capacity), lrt.Capacity()); \
595*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ((obj).Get(), lrt.Get(iref)); \
596*795d594fSAndroid Build Coastguard Worker } while (false)
597*795d594fSAndroid Build Coastguard Worker #define REMOVE_REF(iref, expected_capacity) \
598*795d594fSAndroid Build Coastguard Worker do { \
599*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(iref)); \
600*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(fill_count + (expected_capacity), lrt.Capacity()); \
601*795d594fSAndroid Build Coastguard Worker } while (false)
602*795d594fSAndroid Build Coastguard Worker #define POP_SEGMENT(cookie, expected_capacity) \
603*795d594fSAndroid Build Coastguard Worker do { \
604*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie); \
605*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(fill_count + (expected_capacity), lrt.Capacity()); \
606*795d594fSAndroid Build Coastguard Worker } while (false)
607*795d594fSAndroid Build Coastguard Worker
608*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0 = lrt.PushFrame();
609*795d594fSAndroid Build Coastguard Worker ADD_REF(iref0, obj0, 1u);
610*795d594fSAndroid Build Coastguard Worker ADD_REF(iref1, obj1, 2u);
611*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref1, 1u); // Remove top entry.
612*795d594fSAndroid Build Coastguard Worker if (check_jni) {
613*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref1));
614*795d594fSAndroid Build Coastguard Worker }
615*795d594fSAndroid Build Coastguard Worker ADD_REF(iref1, obj1x, 2u);
616*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref0, 2u); // Create hole.
617*795d594fSAndroid Build Coastguard Worker IndirectRef obsolete_iref0 = iref0;
618*795d594fSAndroid Build Coastguard Worker if (check_jni) {
619*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref0));
620*795d594fSAndroid Build Coastguard Worker }
621*795d594fSAndroid Build Coastguard Worker ADD_REF(iref0, obj0x, 2u); // Reuse hole
622*795d594fSAndroid Build Coastguard Worker if (check_jni) {
623*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(obsolete_iref0));
624*795d594fSAndroid Build Coastguard Worker }
625*795d594fSAndroid Build Coastguard Worker
626*795d594fSAndroid Build Coastguard Worker // Test addition to the second segment without a hole in the first segment.
627*795d594fSAndroid Build Coastguard Worker // Also test removal from the wrong segment here.
628*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie1 = lrt.PushFrame(); // Create second segment.
629*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref0)); // Cannot remove from inactive segment.
630*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2, 3u);
631*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie1, 2u); // Pop the second segment.
632*795d594fSAndroid Build Coastguard Worker if (check_jni) {
633*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(iref2)); // Cannot remove from popped segment.
634*795d594fSAndroid Build Coastguard Worker }
635*795d594fSAndroid Build Coastguard Worker
636*795d594fSAndroid Build Coastguard Worker // Test addition to the second segment with a hole in the first.
637*795d594fSAndroid Build Coastguard Worker // Use one more reference in the first segment to allow hitting the small table
638*795d594fSAndroid Build Coastguard Worker // overflow path either above or here, based on the provided `fill_count`.
639*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2x, 3u);
640*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref1, 3u); // Create hole.
641*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Create second segment.
642*795d594fSAndroid Build Coastguard Worker ADD_REF(iref3, obj3, 4u);
643*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie1, 3u); // Pop the second segment.
644*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref2, 1u); // Remove top entry, prune previous entry.
645*795d594fSAndroid Build Coastguard Worker ADD_REF(iref1, obj1, 2u);
646*795d594fSAndroid Build Coastguard Worker
647*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Create second segment.
648*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2, 3u);
649*795d594fSAndroid Build Coastguard Worker ADD_REF(iref3, obj3, 4u);
650*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref2, 4u); // Create hole in second segment.
651*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie1, 2u); // Pop the second segment with hole.
652*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2x, 3u); // Prune free list, use new entry.
653*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref2, 2u);
654*795d594fSAndroid Build Coastguard Worker
655*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref0, 2u); // Create hole.
656*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Create second segment.
657*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2, 3u);
658*795d594fSAndroid Build Coastguard Worker ADD_REF(iref3, obj3x, 4u);
659*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref2, 4u); // Create hole in second segment.
660*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie1, 2u); // Pop the second segment with hole.
661*795d594fSAndroid Build Coastguard Worker ADD_REF(iref0, obj0, 2u); // Prune free list, use remaining entry from free list.
662*795d594fSAndroid Build Coastguard Worker
663*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref0, 2u); // Create hole.
664*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Create second segment.
665*795d594fSAndroid Build Coastguard Worker ADD_REF(iref2, obj2x, 3u);
666*795d594fSAndroid Build Coastguard Worker ADD_REF(iref3, obj3, 4u);
667*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref2, 4u); // Create hole in second segment.
668*795d594fSAndroid Build Coastguard Worker REMOVE_REF(iref3, 2u); // Remove top entry, prune previous entry, keep hole above.
669*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie1, 2u); // Pop the empty second segment.
670*795d594fSAndroid Build Coastguard Worker ADD_REF(iref0, obj0x, 2u); // Reuse hole.
671*795d594fSAndroid Build Coastguard Worker
672*795d594fSAndroid Build Coastguard Worker POP_SEGMENT(cookie0, 0u); // Pop the first segment.
673*795d594fSAndroid Build Coastguard Worker
674*795d594fSAndroid Build Coastguard Worker #undef REMOVE_REF
675*795d594fSAndroid Build Coastguard Worker #undef ADD_REF
676*795d594fSAndroid Build Coastguard Worker }
677*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,TestAddRemove)678*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, TestAddRemove) {
679*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ false, /*max_count=*/ 20u);
680*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ false, /*max_count=*/ kSmallLrtEntries);
681*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ false, /*max_count=*/ 2u * kSmallLrtEntries);
682*795d594fSAndroid Build Coastguard Worker static_assert(kSmallLrtEntries >= 4u);
683*795d594fSAndroid Build Coastguard Worker for (size_t fill_count = kSmallLrtEntries - 4u; fill_count != kSmallLrtEntries; ++fill_count) {
684*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ false, /*max_count=*/ kSmallLrtEntries, fill_count);
685*795d594fSAndroid Build Coastguard Worker }
686*795d594fSAndroid Build Coastguard Worker }
687*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,TestAddRemoveCheckJNI)688*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, TestAddRemoveCheckJNI) {
689*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ true, /*max_count=*/ 20u);
690*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ true, /*max_count=*/ kSmallLrtEntries);
691*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ true, /*max_count=*/ 2u * kSmallLrtEntries);
692*795d594fSAndroid Build Coastguard Worker static_assert(kSmallLrtEntries >= 4u);
693*795d594fSAndroid Build Coastguard Worker for (size_t fill_count = kSmallLrtEntries - 4u; fill_count != kSmallLrtEntries; ++fill_count) {
694*795d594fSAndroid Build Coastguard Worker TestAddRemove(/*check_jni=*/ true, /*max_count=*/ kSmallLrtEntries, fill_count);
695*795d594fSAndroid Build Coastguard Worker }
696*795d594fSAndroid Build Coastguard Worker }
697*795d594fSAndroid Build Coastguard Worker
TestAddRemoveMixed(bool start_check_jni)698*795d594fSAndroid Build Coastguard Worker void LocalReferenceTableTest::TestAddRemoveMixed(bool start_check_jni) {
699*795d594fSAndroid Build Coastguard Worker // This will lead to error messages in the log.
700*795d594fSAndroid Build Coastguard Worker ScopedLogSeverity sls(LogSeverity::FATAL);
701*795d594fSAndroid Build Coastguard Worker
702*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
703*795d594fSAndroid Build Coastguard Worker static constexpr size_t kMaxUniqueRefs = 16;
704*795d594fSAndroid Build Coastguard Worker StackHandleScope<kMaxUniqueRefs + 1u> hs(soa.Self());
705*795d594fSAndroid Build Coastguard Worker Handle<mirror::Class> c = hs.NewHandle(GetClassRoot<mirror::Object>());
706*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(c != nullptr);
707*795d594fSAndroid Build Coastguard Worker std::array<Handle<mirror::Object>, kMaxUniqueRefs> objs;
708*795d594fSAndroid Build Coastguard Worker for (size_t i = 0u; i != kMaxUniqueRefs; ++i) {
709*795d594fSAndroid Build Coastguard Worker objs[i] = hs.NewHandle(c->AllocObject(soa.Self()));
710*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(objs[i] != nullptr);
711*795d594fSAndroid Build Coastguard Worker }
712*795d594fSAndroid Build Coastguard Worker
713*795d594fSAndroid Build Coastguard Worker std::string error_msg;
714*795d594fSAndroid Build Coastguard Worker std::array<IndirectRef, kMaxUniqueRefs> irefs;
715*795d594fSAndroid Build Coastguard Worker
716*795d594fSAndroid Build Coastguard Worker #define ADD_REF(iref, obj) \
717*795d594fSAndroid Build Coastguard Worker do { \
718*795d594fSAndroid Build Coastguard Worker (iref) = lrt.Add((obj).Get(), &error_msg); \
719*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE((iref) != nullptr) << error_msg; \
720*795d594fSAndroid Build Coastguard Worker EXPECT_OBJ_PTR_EQ((obj).Get(), lrt.Get(iref)); \
721*795d594fSAndroid Build Coastguard Worker } while (false)
722*795d594fSAndroid Build Coastguard Worker
723*795d594fSAndroid Build Coastguard Worker for (size_t split = 1u; split < kMaxUniqueRefs - 1u; ++split) {
724*795d594fSAndroid Build Coastguard Worker for (size_t total = split + 1u; total < kMaxUniqueRefs; ++total) {
725*795d594fSAndroid Build Coastguard Worker for (size_t deleted_at_start = 0u; deleted_at_start + 1u < split; ++deleted_at_start) {
726*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(/*check_jni=*/ start_check_jni);
727*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(kSmallLrtEntries, &error_msg);
728*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
729*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != split; ++i) {
730*795d594fSAndroid Build Coastguard Worker ADD_REF(irefs[i], objs[i]);
731*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(i + 1u, lrt.Capacity());
732*795d594fSAndroid Build Coastguard Worker }
733*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != deleted_at_start; ++i) {
734*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(irefs[i]));
735*795d594fSAndroid Build Coastguard Worker if (lrt.IsCheckJniEnabled()) {
736*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(irefs[i]));
737*795d594fSAndroid Build Coastguard Worker }
738*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(split, lrt.Capacity());
739*795d594fSAndroid Build Coastguard Worker }
740*795d594fSAndroid Build Coastguard Worker lrt.SetCheckJniEnabled(!start_check_jni);
741*795d594fSAndroid Build Coastguard Worker // Check top index instead of `Capacity()` after changing the CheckJNI setting.
742*795d594fSAndroid Build Coastguard Worker auto get_segment_state = [&lrt]() {
743*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie0 = lrt.PushFrame();
744*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie1 = lrt.PushFrame();
745*795d594fSAndroid Build Coastguard Worker uint32_t result = cookie1.top_index;
746*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
747*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
748*795d594fSAndroid Build Coastguard Worker return result;
749*795d594fSAndroid Build Coastguard Worker };
750*795d594fSAndroid Build Coastguard Worker uint32_t split_top_index = get_segment_state();
751*795d594fSAndroid Build Coastguard Worker uint32_t last_top_index = split_top_index;
752*795d594fSAndroid Build Coastguard Worker for (size_t i = split; i != total; ++i) {
753*795d594fSAndroid Build Coastguard Worker ADD_REF(irefs[i], objs[i]);
754*795d594fSAndroid Build Coastguard Worker ASSERT_LT(last_top_index, get_segment_state());
755*795d594fSAndroid Build Coastguard Worker last_top_index = get_segment_state();
756*795d594fSAndroid Build Coastguard Worker }
757*795d594fSAndroid Build Coastguard Worker for (size_t i = split; i != total; ++i) {
758*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(irefs[i]));
759*795d594fSAndroid Build Coastguard Worker if (lrt.IsCheckJniEnabled()) {
760*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(lrt.Remove(irefs[i]));
761*795d594fSAndroid Build Coastguard Worker }
762*795d594fSAndroid Build Coastguard Worker if (i + 1u != total) {
763*795d594fSAndroid Build Coastguard Worker ASSERT_LE(last_top_index, get_segment_state());
764*795d594fSAndroid Build Coastguard Worker } else {
765*795d594fSAndroid Build Coastguard Worker ASSERT_GT(last_top_index, get_segment_state());
766*795d594fSAndroid Build Coastguard Worker ASSERT_LE(split_top_index, get_segment_state());
767*795d594fSAndroid Build Coastguard Worker }
768*795d594fSAndroid Build Coastguard Worker }
769*795d594fSAndroid Build Coastguard Worker }
770*795d594fSAndroid Build Coastguard Worker }
771*795d594fSAndroid Build Coastguard Worker }
772*795d594fSAndroid Build Coastguard Worker
773*795d594fSAndroid Build Coastguard Worker #undef ADD_REF
774*795d594fSAndroid Build Coastguard Worker }
775*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,TestAddRemoveMixed)776*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, TestAddRemoveMixed) {
777*795d594fSAndroid Build Coastguard Worker TestAddRemoveMixed(/*start_check_jni=*/ false);
778*795d594fSAndroid Build Coastguard Worker TestAddRemoveMixed(/*start_check_jni=*/ true);
779*795d594fSAndroid Build Coastguard Worker }
780*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,RegressionTestB276210372)781*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, RegressionTestB276210372) {
782*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(/*check_jni=*/ false);
783*795d594fSAndroid Build Coastguard Worker std::string error_msg;
784*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(kSmallLrtEntries, &error_msg);
785*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
786*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
787*795d594fSAndroid Build Coastguard Worker ObjPtr<mirror::Class> c = GetClassRoot<mirror::Object>();
788*795d594fSAndroid Build Coastguard Worker
789*795d594fSAndroid Build Coastguard Worker auto get_previous_state = [&lrt]() {
790*795d594fSAndroid Build Coastguard Worker LRTSegmentState previous_state = lrt.PushFrame();
791*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(previous_state);
792*795d594fSAndroid Build Coastguard Worker return previous_state;
793*795d594fSAndroid Build Coastguard Worker };
794*795d594fSAndroid Build Coastguard Worker
795*795d594fSAndroid Build Coastguard Worker // Create the first segment with two references.
796*795d594fSAndroid Build Coastguard Worker IndirectRef ref0 = lrt.Add(c, &error_msg);
797*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(ref0 != nullptr);
798*795d594fSAndroid Build Coastguard Worker IndirectRef ref1 = lrt.Add(c, &error_msg);
799*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(ref1 != nullptr);
800*795d594fSAndroid Build Coastguard Worker
801*795d594fSAndroid Build Coastguard Worker // Create a second segment with a hole, then pop it.
802*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0A = lrt.PushFrame();
803*795d594fSAndroid Build Coastguard Worker const LRTSegmentState previous_state_A = get_previous_state();
804*795d594fSAndroid Build Coastguard Worker IndirectRef ref2a = lrt.Add(c, &error_msg);
805*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(ref2a != nullptr);
806*795d594fSAndroid Build Coastguard Worker IndirectRef ref3a = lrt.Add(c, &error_msg);
807*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(ref3a != nullptr);
808*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(ref2a));
809*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0A);
810*795d594fSAndroid Build Coastguard Worker
811*795d594fSAndroid Build Coastguard Worker // Create a hole in the first segment.
812*795d594fSAndroid Build Coastguard Worker // There was previously a bug that `Remove()` would not prune the popped free entries,
813*795d594fSAndroid Build Coastguard Worker // so the new free entry would point to the hole in the popped segment. The code below
814*795d594fSAndroid Build Coastguard Worker // would then overwrite that hole with a new segment, pop that segment, reuse the good
815*795d594fSAndroid Build Coastguard Worker // free entry and then crash trying to prune the overwritten hole. b/276210372
816*795d594fSAndroid Build Coastguard Worker EXPECT_TRUE(lrt.Remove(ref0));
817*795d594fSAndroid Build Coastguard Worker
818*795d594fSAndroid Build Coastguard Worker // Create a second segment again and overwite the old hole, then pop the segment.
819*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0B = lrt.PushFrame();
820*795d594fSAndroid Build Coastguard Worker const LRTSegmentState previous_state_B = get_previous_state();
821*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(cookie0B.top_index, cookie0A.top_index);
822*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(previous_state_B.top_index, previous_state_A.top_index);
823*795d594fSAndroid Build Coastguard Worker IndirectRef ref2b = lrt.Add(c, &error_msg);
824*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(ref2b != nullptr);
825*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0B);
826*795d594fSAndroid Build Coastguard Worker
827*795d594fSAndroid Build Coastguard Worker // Reuse the hole in first segment.
828*795d594fSAndroid Build Coastguard Worker IndirectRef reused0 = lrt.Add(c, &error_msg);
829*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(reused0 != nullptr);
830*795d594fSAndroid Build Coastguard Worker
831*795d594fSAndroid Build Coastguard Worker // Add a new reference.
832*795d594fSAndroid Build Coastguard Worker IndirectRef new_ref = lrt.Add(c, &error_msg);
833*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(new_ref != nullptr);
834*795d594fSAndroid Build Coastguard Worker }
835*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,RegressionTestB276864369)836*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, RegressionTestB276864369) {
837*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(/*check_jni=*/ false);
838*795d594fSAndroid Build Coastguard Worker std::string error_msg;
839*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(kSmallLrtEntries, &error_msg);
840*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
841*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
842*795d594fSAndroid Build Coastguard Worker ObjPtr<mirror::Class> c = GetClassRoot<mirror::Object>();
843*795d594fSAndroid Build Coastguard Worker
844*795d594fSAndroid Build Coastguard Worker // Add refs to fill all small tables and one bigger table.
845*795d594fSAndroid Build Coastguard Worker const size_t refs_per_page = gPageSize / sizeof(LrtEntry);
846*795d594fSAndroid Build Coastguard Worker std::vector<IndirectRef> refs;
847*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != 2 * refs_per_page; ++i) {
848*795d594fSAndroid Build Coastguard Worker refs.push_back(lrt.Add(c, &error_msg));
849*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs.back() != nullptr);
850*795d594fSAndroid Build Coastguard Worker }
851*795d594fSAndroid Build Coastguard Worker
852*795d594fSAndroid Build Coastguard Worker // We had a bug in `Trim()` where we would try to skip one more table than available
853*795d594fSAndroid Build Coastguard Worker // if the capacity was exactly at the end of table. If the next table was not allocated,
854*795d594fSAndroid Build Coastguard Worker // we would hit a `DCHECK()` in `dchecked_vector<>` in debug mode but in release
855*795d594fSAndroid Build Coastguard Worker // mode we would proceed to use memory outside the allocated chunk. b/276864369
856*795d594fSAndroid Build Coastguard Worker lrt.Trim();
857*795d594fSAndroid Build Coastguard Worker }
858*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,Trim)859*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, Trim) {
860*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(/*check_jni=*/ false);
861*795d594fSAndroid Build Coastguard Worker std::string error_msg;
862*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(kSmallLrtEntries, &error_msg);
863*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
864*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
865*795d594fSAndroid Build Coastguard Worker ObjPtr<mirror::Class> c = GetClassRoot<mirror::Object>();
866*795d594fSAndroid Build Coastguard Worker
867*795d594fSAndroid Build Coastguard Worker // Add refs to fill all small tables.
868*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie0 = lrt.PushFrame();
869*795d594fSAndroid Build Coastguard Worker const size_t refs_per_page = gPageSize / sizeof(LrtEntry);
870*795d594fSAndroid Build Coastguard Worker std::vector<IndirectRef> refs0;
871*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != refs_per_page; ++i) {
872*795d594fSAndroid Build Coastguard Worker refs0.push_back(lrt.Add(c, &error_msg));
873*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs0.back() != nullptr);
874*795d594fSAndroid Build Coastguard Worker }
875*795d594fSAndroid Build Coastguard Worker
876*795d594fSAndroid Build Coastguard Worker // Nothing to trim.
877*795d594fSAndroid Build Coastguard Worker lrt.Trim();
878*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs0.back())->IsNull());
879*795d594fSAndroid Build Coastguard Worker
880*795d594fSAndroid Build Coastguard Worker // Add refs to fill the next, page-sized table.
881*795d594fSAndroid Build Coastguard Worker std::vector<IndirectRef> refs1;
882*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie1 = lrt.PushFrame();
883*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != refs_per_page; ++i) {
884*795d594fSAndroid Build Coastguard Worker refs1.push_back(lrt.Add(c, &error_msg));
885*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs1.back() != nullptr);
886*795d594fSAndroid Build Coastguard Worker }
887*795d594fSAndroid Build Coastguard Worker
888*795d594fSAndroid Build Coastguard Worker // Nothing to trim.
889*795d594fSAndroid Build Coastguard Worker lrt.Trim();
890*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs1.back())->IsNull());
891*795d594fSAndroid Build Coastguard Worker
892*795d594fSAndroid Build Coastguard Worker // Pop one reference and try to trim, there is no page to trim.
893*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(refs1.back()));
894*795d594fSAndroid Build Coastguard Worker lrt.Trim();
895*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(
896*795d594fSAndroid Build Coastguard Worker IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs1[refs1.size() - 2u])->IsNull());
897*795d594fSAndroid Build Coastguard Worker
898*795d594fSAndroid Build Coastguard Worker // Pop the entire segment with the page-sized table and trim, clearing the page.
899*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
900*795d594fSAndroid Build Coastguard Worker lrt.Trim();
901*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs1) {
902*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
903*795d594fSAndroid Build Coastguard Worker }
904*795d594fSAndroid Build Coastguard Worker refs1.clear();
905*795d594fSAndroid Build Coastguard Worker
906*795d594fSAndroid Build Coastguard Worker // Add refs to fill the page-sized table and half of the next one.
907*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Push a new segment.
908*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != 2 * refs_per_page; ++i) {
909*795d594fSAndroid Build Coastguard Worker refs1.push_back(lrt.Add(c, &error_msg));
910*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs1.back() != nullptr);
911*795d594fSAndroid Build Coastguard Worker }
912*795d594fSAndroid Build Coastguard Worker
913*795d594fSAndroid Build Coastguard Worker // Add refs to fill the other half of the table with two pages.
914*795d594fSAndroid Build Coastguard Worker std::vector<IndirectRef> refs2;
915*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie2 = lrt.PushFrame();
916*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != refs_per_page; ++i) {
917*795d594fSAndroid Build Coastguard Worker refs2.push_back(lrt.Add(c, &error_msg));
918*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs2.back() != nullptr);
919*795d594fSAndroid Build Coastguard Worker }
920*795d594fSAndroid Build Coastguard Worker
921*795d594fSAndroid Build Coastguard Worker // Nothing to trim.
922*795d594fSAndroid Build Coastguard Worker lrt.Trim();
923*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs1.back())->IsNull());
924*795d594fSAndroid Build Coastguard Worker
925*795d594fSAndroid Build Coastguard Worker // Pop the last segment with one page worth of references and trim that page.
926*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie2);
927*795d594fSAndroid Build Coastguard Worker lrt.Trim();
928*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs2) {
929*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
930*795d594fSAndroid Build Coastguard Worker }
931*795d594fSAndroid Build Coastguard Worker refs2.clear();
932*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs1) {
933*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
934*795d594fSAndroid Build Coastguard Worker }
935*795d594fSAndroid Build Coastguard Worker
936*795d594fSAndroid Build Coastguard Worker // Pop the middle segment with two pages worth of references, and trim those pages.
937*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
938*795d594fSAndroid Build Coastguard Worker lrt.Trim();
939*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs1) {
940*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
941*795d594fSAndroid Build Coastguard Worker }
942*795d594fSAndroid Build Coastguard Worker refs1.clear();
943*795d594fSAndroid Build Coastguard Worker
944*795d594fSAndroid Build Coastguard Worker // Pop the first segment with small tables and try to trim. Small tables are never trimmed.
945*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
946*795d594fSAndroid Build Coastguard Worker lrt.Trim();
947*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs0) {
948*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
949*795d594fSAndroid Build Coastguard Worker }
950*795d594fSAndroid Build Coastguard Worker refs0.clear();
951*795d594fSAndroid Build Coastguard Worker
952*795d594fSAndroid Build Coastguard Worker // Fill small tables and one more reference, then another segment up to 4 pages.
953*795d594fSAndroid Build Coastguard Worker LRTSegmentState cookie0_second = lrt.PushFrame();
954*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(cookie0.top_index, cookie0_second.top_index);
955*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != refs_per_page + 1u; ++i) {
956*795d594fSAndroid Build Coastguard Worker refs0.push_back(lrt.Add(c, &error_msg));
957*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs0.back() != nullptr);
958*795d594fSAndroid Build Coastguard Worker }
959*795d594fSAndroid Build Coastguard Worker cookie1 = lrt.PushFrame(); // Push a new segment.
960*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != 3u * refs_per_page - 1u; ++i) {
961*795d594fSAndroid Build Coastguard Worker refs1.push_back(lrt.Add(c, &error_msg));
962*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs1.back() != nullptr);
963*795d594fSAndroid Build Coastguard Worker }
964*795d594fSAndroid Build Coastguard Worker
965*795d594fSAndroid Build Coastguard Worker // Nothing to trim.
966*795d594fSAndroid Build Coastguard Worker lrt.Trim();
967*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs1.back())->IsNull());
968*795d594fSAndroid Build Coastguard Worker
969*795d594fSAndroid Build Coastguard Worker // Pop the middle segment, trim two pages.
970*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie1);
971*795d594fSAndroid Build Coastguard Worker lrt.Trim();
972*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : refs0) {
973*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
974*795d594fSAndroid Build Coastguard Worker }
975*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(refs0.size(), lrt.Capacity());
976*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : ArrayRef<IndirectRef>(refs1).SubArray(0u, refs_per_page - 1u)) {
977*795d594fSAndroid Build Coastguard Worker // Popped but not trimmed as these are at the same page as the last entry in `refs0`.
978*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
979*795d594fSAndroid Build Coastguard Worker }
980*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : ArrayRef<IndirectRef>(refs1).SubArray(refs_per_page - 1u)) {
981*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
982*795d594fSAndroid Build Coastguard Worker }
983*795d594fSAndroid Build Coastguard Worker }
984*795d594fSAndroid Build Coastguard Worker
TEST_F(LocalReferenceTableTest,PruneBeforeTrim)985*795d594fSAndroid Build Coastguard Worker TEST_F(LocalReferenceTableTest, PruneBeforeTrim) {
986*795d594fSAndroid Build Coastguard Worker LocalReferenceTable lrt(/*check_jni=*/ false);
987*795d594fSAndroid Build Coastguard Worker std::string error_msg;
988*795d594fSAndroid Build Coastguard Worker bool success = lrt.Initialize(kSmallLrtEntries, &error_msg);
989*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(success) << error_msg;
990*795d594fSAndroid Build Coastguard Worker ScopedObjectAccess soa(Thread::Current());
991*795d594fSAndroid Build Coastguard Worker ObjPtr<mirror::Class> c = GetClassRoot<mirror::Object>();
992*795d594fSAndroid Build Coastguard Worker
993*795d594fSAndroid Build Coastguard Worker // Add refs to fill all small tables and one bigger table.
994*795d594fSAndroid Build Coastguard Worker const LRTSegmentState cookie0 = lrt.PushFrame();
995*795d594fSAndroid Build Coastguard Worker const size_t refs_per_page = gPageSize / sizeof(LrtEntry);
996*795d594fSAndroid Build Coastguard Worker std::vector<IndirectRef> refs;
997*795d594fSAndroid Build Coastguard Worker for (size_t i = 0; i != 2 * refs_per_page; ++i) {
998*795d594fSAndroid Build Coastguard Worker refs.push_back(lrt.Add(c, &error_msg));
999*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(refs.back() != nullptr);
1000*795d594fSAndroid Build Coastguard Worker }
1001*795d594fSAndroid Build Coastguard Worker
1002*795d594fSAndroid Build Coastguard Worker // Nothing to trim.
1003*795d594fSAndroid Build Coastguard Worker lrt.Trim();
1004*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(refs.back())->IsNull());
1005*795d594fSAndroid Build Coastguard Worker
1006*795d594fSAndroid Build Coastguard Worker // Create a hole in the last page.
1007*795d594fSAndroid Build Coastguard Worker IndirectRef removed = refs[refs.size() - 2u];
1008*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(lrt.Remove(removed));
1009*795d594fSAndroid Build Coastguard Worker
1010*795d594fSAndroid Build Coastguard Worker // Pop the entire segment and trim. Small tables are not pruned.
1011*795d594fSAndroid Build Coastguard Worker lrt.PopFrame(cookie0);
1012*795d594fSAndroid Build Coastguard Worker lrt.Trim();
1013*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : ArrayRef<IndirectRef>(refs).SubArray(0u, refs_per_page)) {
1014*795d594fSAndroid Build Coastguard Worker ASSERT_FALSE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
1015*795d594fSAndroid Build Coastguard Worker }
1016*795d594fSAndroid Build Coastguard Worker for (IndirectRef ref : ArrayRef<IndirectRef>(refs).SubArray(refs_per_page)) {
1017*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(IndirectReferenceTable::ClearIndirectRefKind<LrtEntry*>(ref)->IsNull());
1018*795d594fSAndroid Build Coastguard Worker }
1019*795d594fSAndroid Build Coastguard Worker
1020*795d594fSAndroid Build Coastguard Worker // Add a new reference and check that it reused the first slot rather than the old hole.
1021*795d594fSAndroid Build Coastguard Worker IndirectRef new_ref = lrt.Add(c, &error_msg);
1022*795d594fSAndroid Build Coastguard Worker ASSERT_TRUE(new_ref != nullptr);
1023*795d594fSAndroid Build Coastguard Worker ASSERT_NE(new_ref, removed);
1024*795d594fSAndroid Build Coastguard Worker ASSERT_EQ(new_ref, refs[0]);
1025*795d594fSAndroid Build Coastguard Worker }
1026*795d594fSAndroid Build Coastguard Worker
1027*795d594fSAndroid Build Coastguard Worker } // namespace jni
1028*795d594fSAndroid Build Coastguard Worker } // namespace art
1029