blob: dfbbd2a0c1e7579751c819b6ddfe62ee850aaf98 [file] [log] [blame]
Andreas Gampefda57142016-09-08 20:29:18 -07001/*
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"
Andreas Gampec15a2f42017-04-21 12:09:39 -070026#include "gc_root-inl.h"
Andreas Gampefda57142016-09-08 20:29:18 -070027#include "handle_scope-inl.h"
28#include "heap.h"
29#include "mirror/string.h"
Mathieu Chartier0795f232016-09-27 18:43:30 -070030#include "scoped_thread_state_change-inl.h"
Andreas Gampefda57142016-09-08 20:29:18 -070031#include "thread_list.h"
32
33namespace art {
34namespace gc {
35
36class SystemWeakTest : public CommonRuntimeTest {
37};
38
39struct CountingSystemWeakHolder : public SystemWeakHolder {
40 CountingSystemWeakHolder()
41 : SystemWeakHolder(kAllocTrackerLock),
42 allow_count_(0),
43 disallow_count_(0),
44 sweep_count_(0) {}
45
46 void Allow() OVERRIDE
47 REQUIRES_SHARED(Locks::mutator_lock_)
48 REQUIRES(!allow_disallow_lock_) {
49 SystemWeakHolder::Allow();
50
51 allow_count_++;
52 }
53
54 void Disallow() OVERRIDE
55 REQUIRES_SHARED(Locks::mutator_lock_)
56 REQUIRES(!allow_disallow_lock_) {
57 SystemWeakHolder::Disallow();
58
59 disallow_count_++;
60 }
61
Hiroshi Yamauchi30493242016-11-03 13:06:52 -070062 void Broadcast(bool broadcast_for_checkpoint) OVERRIDE
Andreas Gampefda57142016-09-08 20:29:18 -070063 REQUIRES(!allow_disallow_lock_) {
Hiroshi Yamauchi30493242016-11-03 13:06:52 -070064 SystemWeakHolder::Broadcast(broadcast_for_checkpoint);
Andreas Gampefda57142016-09-08 20:29:18 -070065
Hiroshi Yamauchi30493242016-11-03 13:06:52 -070066 if (!broadcast_for_checkpoint) {
67 // Don't count the broadcasts for running checkpoints.
68 allow_count_++;
69 }
Andreas Gampefda57142016-09-08 20:29:18 -070070 }
71
72 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
83 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
93 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
109static 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
121static 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
132TEST_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(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
157TEST_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(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
177TEST_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(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(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