• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "system_weak.h"
18 
19 #include <stdint.h>
20 #include <stdio.h>
21 #include <memory>
22 
23 #include "base/mutex.h"
24 #include "collector_type.h"
25 #include "common_runtime_test.h"
26 #include "gc_root-inl.h"
27 #include "handle_scope-inl.h"
28 #include "heap.h"
29 #include "mirror/string.h"
30 #include "scoped_thread_state_change-inl.h"
31 #include "thread_list.h"
32 
33 namespace art {
34 namespace gc {
35 
36 class SystemWeakTest : public CommonRuntimeTest {
37 };
38 
39 struct CountingSystemWeakHolder : public SystemWeakHolder {
CountingSystemWeakHolderart::gc::CountingSystemWeakHolder40   CountingSystemWeakHolder()
41       : SystemWeakHolder(kAllocTrackerLock),
42         allow_count_(0),
43         disallow_count_(0),
44         sweep_count_(0) {}
45 
Allowart::gc::CountingSystemWeakHolder46   void Allow() OVERRIDE
47       REQUIRES_SHARED(Locks::mutator_lock_)
48       REQUIRES(!allow_disallow_lock_) {
49     SystemWeakHolder::Allow();
50 
51     allow_count_++;
52   }
53 
Disallowart::gc::CountingSystemWeakHolder54   void Disallow() OVERRIDE
55       REQUIRES_SHARED(Locks::mutator_lock_)
56       REQUIRES(!allow_disallow_lock_) {
57     SystemWeakHolder::Disallow();
58 
59     disallow_count_++;
60   }
61 
Broadcastart::gc::CountingSystemWeakHolder62   void Broadcast(bool broadcast_for_checkpoint) OVERRIDE
63       REQUIRES(!allow_disallow_lock_) {
64     SystemWeakHolder::Broadcast(broadcast_for_checkpoint);
65 
66     if (!broadcast_for_checkpoint) {
67       // Don't count the broadcasts for running checkpoints.
68       allow_count_++;
69     }
70   }
71 
Sweepart::gc::CountingSystemWeakHolder72   void Sweep(IsMarkedVisitor* visitor) OVERRIDE
73       REQUIRES_SHARED(Locks::mutator_lock_)
74       REQUIRES(!allow_disallow_lock_) {
75     MutexLock mu(Thread::Current(), allow_disallow_lock_);
76     mirror::Object* old_object = weak_.Read<kWithoutReadBarrier>();
77     mirror::Object* new_object = old_object == nullptr ? nullptr : visitor->IsMarked(old_object);
78     weak_ = GcRoot<mirror::Object>(new_object);
79 
80     sweep_count_++;
81   }
82 
Getart::gc::CountingSystemWeakHolder83   GcRoot<mirror::Object> Get()
84       REQUIRES_SHARED(Locks::mutator_lock_)
85       REQUIRES(!allow_disallow_lock_) {
86     Thread* self = Thread::Current();
87     MutexLock mu(self, allow_disallow_lock_);
88     Wait(self);
89 
90     return weak_;
91   }
92 
Setart::gc::CountingSystemWeakHolder93   void Set(GcRoot<mirror::Object> obj)
94       REQUIRES_SHARED(Locks::mutator_lock_)
95       REQUIRES(!allow_disallow_lock_) {
96     Thread* self = Thread::Current();
97     MutexLock mu(self, allow_disallow_lock_);
98     Wait(self);
99 
100     weak_ = obj;
101   }
102 
103   size_t allow_count_;
104   size_t disallow_count_;
105   size_t sweep_count_;
106   GcRoot<mirror::Object> weak_ GUARDED_BY(allow_disallow_lock_);
107 };
108 
CollectorDoesAllowOrBroadcast()109 static bool CollectorDoesAllowOrBroadcast() {
110   CollectorType type = Runtime::Current()->GetHeap()->CurrentCollectorType();
111   switch (type) {
112     case CollectorType::kCollectorTypeCMS:
113     case CollectorType::kCollectorTypeCC:
114       return true;
115 
116     default:
117       return false;
118   }
119 }
120 
CollectorDoesDisallow()121 static bool CollectorDoesDisallow() {
122   CollectorType type = Runtime::Current()->GetHeap()->CurrentCollectorType();
123   switch (type) {
124     case CollectorType::kCollectorTypeCMS:
125       return true;
126 
127     default:
128       return false;
129   }
130 }
131 
TEST_F(SystemWeakTest,Keep)132 TEST_F(SystemWeakTest, Keep) {
133   CountingSystemWeakHolder cswh;
134   Runtime::Current()->AddSystemWeakHolder(&cswh);
135 
136   ScopedObjectAccess soa(Thread::Current());
137 
138   StackHandleScope<1> hs(soa.Self());
139 
140   // We use Strings because they are very easy to allocate.
141   Handle<mirror::String> s(hs.NewHandle(mirror::String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
142   cswh.Set(GcRoot<mirror::Object>(s.Get()));
143 
144   // Trigger a GC.
145   Runtime::Current()->GetHeap()->CollectGarbage(/* clear_soft_references */ false);
146 
147   // Expect the holder to have been called.
148   EXPECT_EQ(CollectorDoesAllowOrBroadcast() ? 1U : 0U, cswh.allow_count_);
149   EXPECT_EQ(CollectorDoesDisallow() ? 1U : 0U, cswh.disallow_count_);
150   EXPECT_EQ(1U, cswh.sweep_count_);
151 
152   // Expect the weak to not be cleared.
153   EXPECT_FALSE(cswh.Get().IsNull());
154   EXPECT_EQ(cswh.Get().Read(), s.Get());
155 }
156 
TEST_F(SystemWeakTest,Discard)157 TEST_F(SystemWeakTest, Discard) {
158   CountingSystemWeakHolder cswh;
159   Runtime::Current()->AddSystemWeakHolder(&cswh);
160 
161   ScopedObjectAccess soa(Thread::Current());
162 
163   cswh.Set(GcRoot<mirror::Object>(mirror::String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
164 
165   // Trigger a GC.
166   Runtime::Current()->GetHeap()->CollectGarbage(/* clear_soft_references */ false);
167 
168   // Expect the holder to have been called.
169   EXPECT_EQ(CollectorDoesAllowOrBroadcast() ? 1U : 0U, cswh.allow_count_);
170   EXPECT_EQ(CollectorDoesDisallow() ? 1U : 0U, cswh.disallow_count_);
171   EXPECT_EQ(1U, cswh.sweep_count_);
172 
173   // Expect the weak to be cleared.
174   EXPECT_TRUE(cswh.Get().IsNull());
175 }
176 
TEST_F(SystemWeakTest,Remove)177 TEST_F(SystemWeakTest, Remove) {
178   CountingSystemWeakHolder cswh;
179   Runtime::Current()->AddSystemWeakHolder(&cswh);
180 
181   ScopedObjectAccess soa(Thread::Current());
182 
183   StackHandleScope<1> hs(soa.Self());
184 
185   // We use Strings because they are very easy to allocate.
186   Handle<mirror::String> s(hs.NewHandle(mirror::String::AllocFromModifiedUtf8(soa.Self(), "ABC")));
187   cswh.Set(GcRoot<mirror::Object>(s.Get()));
188 
189   // Trigger a GC.
190   Runtime::Current()->GetHeap()->CollectGarbage(/* clear_soft_references */ false);
191 
192   // Expect the holder to have been called.
193   ASSERT_EQ(CollectorDoesAllowOrBroadcast() ? 1U : 0U, cswh.allow_count_);
194   ASSERT_EQ(CollectorDoesDisallow() ? 1U : 0U, cswh.disallow_count_);
195   ASSERT_EQ(1U, cswh.sweep_count_);
196 
197   // Expect the weak to not be cleared.
198   ASSERT_FALSE(cswh.Get().IsNull());
199   ASSERT_EQ(cswh.Get().Read(), s.Get());
200 
201   // Remove the holder.
202   Runtime::Current()->RemoveSystemWeakHolder(&cswh);
203 
204   // Trigger another GC.
205   Runtime::Current()->GetHeap()->CollectGarbage(/* clear_soft_references */ false);
206 
207   // Expectation: no change in the numbers.
208   EXPECT_EQ(CollectorDoesAllowOrBroadcast() ? 1U : 0U, cswh.allow_count_);
209   EXPECT_EQ(CollectorDoesDisallow() ? 1U : 0U, cswh.disallow_count_);
210   EXPECT_EQ(1U, cswh.sweep_count_);
211 }
212 
213 }  // namespace gc
214 }  // namespace art
215