1 /*
2 * Copyright (C) 2015 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 "space_test.h"
18
19 #include "dlmalloc_space.h"
20 #include "rosalloc_space.h"
21 #include "scoped_thread_state_change-inl.h"
22
23 namespace art HIDDEN {
24 namespace gc {
25 namespace space {
26
27 enum MallocSpaceType {
28 kMallocSpaceDlMalloc,
29 kMallocSpaceRosAlloc,
30 };
31
32 class SpaceCreateTest : public SpaceTest<CommonRuntimeTestWithParam<MallocSpaceType>> {
33 public:
CreateSpace(const std::string & name,size_t initial_size,size_t growth_limit,size_t capacity)34 MallocSpace* CreateSpace(const std::string& name,
35 size_t initial_size,
36 size_t growth_limit,
37 size_t capacity) {
38 const MallocSpaceType type = GetParam();
39 if (type == kMallocSpaceDlMalloc) {
40 return DlMallocSpace::Create(name,
41 initial_size,
42 growth_limit,
43 capacity,
44 /*can_move_objects=*/ false);
45 }
46 DCHECK_EQ(static_cast<uint32_t>(type), static_cast<uint32_t>(kMallocSpaceRosAlloc));
47 return RosAllocSpace::Create(name,
48 initial_size,
49 growth_limit,
50 capacity,
51 Runtime::Current()->GetHeap()->IsLowMemoryMode(),
52 /*can_move_objects=*/ false);
53 }
54 };
55
TEST_P(SpaceCreateTest,InitTestBody)56 TEST_P(SpaceCreateTest, InitTestBody) {
57 // This will lead to error messages in the log.
58 ScopedLogSeverity sls(LogSeverity::FATAL);
59
60 {
61 // Init < max == growth
62 std::unique_ptr<Space> space(CreateSpace("test", 16 * MB, 32 * MB, 32 * MB));
63 EXPECT_TRUE(space != nullptr);
64 // Init == max == growth
65 space.reset(CreateSpace("test", 16 * MB, 16 * MB, 16 * MB));
66 EXPECT_TRUE(space != nullptr);
67 // Init > max == growth
68 space.reset(CreateSpace("test", 32 * MB, 16 * MB, 16 * MB));
69 EXPECT_TRUE(space == nullptr);
70 // Growth == init < max
71 space.reset(CreateSpace("test", 16 * MB, 16 * MB, 32 * MB));
72 EXPECT_TRUE(space != nullptr);
73 // Growth < init < max
74 space.reset(CreateSpace("test", 16 * MB, 8 * MB, 32 * MB));
75 EXPECT_TRUE(space == nullptr);
76 // Init < growth < max
77 space.reset(CreateSpace("test", 8 * MB, 16 * MB, 32 * MB));
78 EXPECT_TRUE(space != nullptr);
79 // Init < max < growth
80 space.reset(CreateSpace("test", 8 * MB, 32 * MB, 16 * MB));
81 EXPECT_TRUE(space == nullptr);
82 }
83 }
84
85 // TODO: This test is not very good, we should improve it.
86 // The test should do more allocations before the creation of the ZygoteSpace, and then do
87 // allocations after the ZygoteSpace is created. The test should also do some GCs to ensure that
88 // the GC works with the ZygoteSpace.
TEST_P(SpaceCreateTest,ZygoteSpaceTestBody)89 TEST_P(SpaceCreateTest, ZygoteSpaceTestBody) {
90 size_t unused;
91 MallocSpace* space(CreateSpace("test", 4 * MB, 16 * MB, 16 * MB));
92 ASSERT_TRUE(space != nullptr);
93
94 // Make space findable to the heap, will also delete space when runtime is cleaned up
95 AddSpace(space);
96 Thread* self = Thread::Current();
97 ScopedObjectAccess soa(self);
98
99 // Succeeds, fits without adjusting the footprint limit.
100 size_t ptr1_bytes_allocated, ptr1_usable_size, ptr1_bytes_tl_bulk_allocated;
101 StackHandleScope<3> hs(soa.Self());
102 MutableHandle<mirror::Object> ptr1(hs.NewHandle(Alloc(space,
103 self,
104 1 * MB,
105 &ptr1_bytes_allocated,
106 &ptr1_usable_size,
107 &ptr1_bytes_tl_bulk_allocated)));
108 EXPECT_TRUE(ptr1 != nullptr);
109 EXPECT_LE(1U * MB, ptr1_bytes_allocated);
110 EXPECT_LE(1U * MB, ptr1_usable_size);
111 EXPECT_LE(ptr1_usable_size, ptr1_bytes_allocated);
112 EXPECT_EQ(ptr1_bytes_tl_bulk_allocated, ptr1_bytes_allocated);
113
114 // Fails, requires a higher footprint limit.
115 mirror::Object* ptr2 = Alloc(space, self, 8 * MB, &unused, nullptr, &unused);
116 EXPECT_TRUE(ptr2 == nullptr);
117
118 // Succeeds, adjusts the footprint.
119 size_t ptr3_bytes_allocated, ptr3_usable_size, ptr3_bytes_tl_bulk_allocated;
120 MutableHandle<mirror::Object> ptr3(hs.NewHandle(AllocWithGrowth(space,
121 self,
122 8 * MB,
123 &ptr3_bytes_allocated,
124 &ptr3_usable_size,
125 &ptr3_bytes_tl_bulk_allocated)));
126 EXPECT_TRUE(ptr3 != nullptr);
127 EXPECT_LE(8U * MB, ptr3_bytes_allocated);
128 EXPECT_LE(8U * MB, ptr3_usable_size);
129 EXPECT_LE(ptr3_usable_size, ptr3_bytes_allocated);
130 EXPECT_EQ(ptr3_bytes_tl_bulk_allocated, ptr3_bytes_allocated);
131
132 // Fails, requires a higher footprint limit.
133 mirror::Object* ptr4 = space->Alloc(self, 8 * MB, &unused, nullptr, &unused);
134 EXPECT_TRUE(ptr4 == nullptr);
135
136 // Also fails, requires a higher allowed footprint.
137 mirror::Object* ptr5 = space->AllocWithGrowth(self, 8 * MB, &unused, nullptr, &unused);
138 EXPECT_TRUE(ptr5 == nullptr);
139
140 // Release some memory.
141 size_t free3 = space->AllocationSize(ptr3.Get(), nullptr);
142 EXPECT_EQ(free3, ptr3_bytes_allocated);
143 EXPECT_EQ(free3, space->Free(self, ptr3.Assign(nullptr)));
144 EXPECT_LE(8U * MB, free3);
145
146 // Succeeds, now that memory has been freed.
147 size_t ptr6_bytes_allocated, ptr6_usable_size, ptr6_bytes_tl_bulk_allocated;
148 Handle<mirror::Object> ptr6(hs.NewHandle(AllocWithGrowth(space,
149 self,
150 9 * MB,
151 &ptr6_bytes_allocated,
152 &ptr6_usable_size,
153 &ptr6_bytes_tl_bulk_allocated)));
154 EXPECT_TRUE(ptr6 != nullptr);
155 EXPECT_LE(9U * MB, ptr6_bytes_allocated);
156 EXPECT_LE(9U * MB, ptr6_usable_size);
157 EXPECT_LE(ptr6_usable_size, ptr6_bytes_allocated);
158 EXPECT_EQ(ptr6_bytes_tl_bulk_allocated, ptr6_bytes_allocated);
159
160 // Final clean up.
161 size_t free1 = space->AllocationSize(ptr1.Get(), nullptr);
162 space->Free(self, ptr1.Assign(nullptr));
163 EXPECT_LE(1U * MB, free1);
164
165 // Make sure that the zygote space isn't directly at the start of the space.
166 EXPECT_TRUE(space->Alloc(self, 1U * MB, &unused, nullptr, &unused) != nullptr);
167
168 gc::Heap* heap = Runtime::Current()->GetHeap();
169 space::Space* old_space = space;
170 {
171 ScopedThreadSuspension sts(self, ThreadState::kSuspended);
172 ScopedSuspendAll ssa("Add image space");
173 heap->RemoveSpace(old_space);
174 }
175 heap->RevokeAllThreadLocalBuffers();
176 space::ZygoteSpace* zygote_space = space->CreateZygoteSpace("alloc space",
177 heap->IsLowMemoryMode(),
178 &space);
179 delete old_space;
180 // Add the zygote space.
181 AddSpace(zygote_space, false);
182
183 // Make space findable to the heap, will also delete space when runtime is cleaned up
184 AddSpace(space, false);
185
186 // Succeeds, fits without adjusting the footprint limit.
187 ptr1.Assign(Alloc(space,
188 self,
189 1 * MB,
190 &ptr1_bytes_allocated,
191 &ptr1_usable_size,
192 &ptr1_bytes_tl_bulk_allocated));
193 EXPECT_TRUE(ptr1 != nullptr);
194 EXPECT_LE(1U * MB, ptr1_bytes_allocated);
195 EXPECT_LE(1U * MB, ptr1_usable_size);
196 EXPECT_LE(ptr1_usable_size, ptr1_bytes_allocated);
197 EXPECT_EQ(ptr1_bytes_tl_bulk_allocated, ptr1_bytes_allocated);
198
199 // Fails, requires a higher footprint limit.
200 ptr2 = Alloc(space, self, 8 * MB, &unused, nullptr, &unused);
201 EXPECT_TRUE(ptr2 == nullptr);
202
203 // Succeeds, adjusts the footprint.
204 ptr3.Assign(AllocWithGrowth(space,
205 self,
206 2 * MB,
207 &ptr3_bytes_allocated,
208 &ptr3_usable_size,
209 &ptr3_bytes_tl_bulk_allocated));
210 EXPECT_TRUE(ptr3 != nullptr);
211 EXPECT_LE(2U * MB, ptr3_bytes_allocated);
212 EXPECT_LE(2U * MB, ptr3_usable_size);
213 EXPECT_LE(ptr3_usable_size, ptr3_bytes_allocated);
214 EXPECT_EQ(ptr3_bytes_tl_bulk_allocated, ptr3_bytes_allocated);
215 space->Free(self, ptr3.Assign(nullptr));
216
217 // Final clean up.
218 free1 = space->AllocationSize(ptr1.Get(), nullptr);
219 space->Free(self, ptr1.Assign(nullptr));
220 EXPECT_LE(1U * MB, free1);
221 }
222
TEST_P(SpaceCreateTest,AllocAndFreeTestBody)223 TEST_P(SpaceCreateTest, AllocAndFreeTestBody) {
224 size_t unused = 0;
225 MallocSpace* space(CreateSpace("test", 4 * MB, 16 * MB, 16 * MB));
226 ASSERT_TRUE(space != nullptr);
227 Thread* self = Thread::Current();
228 ScopedObjectAccess soa(self);
229
230 // Make space findable to the heap, will also delete space when runtime is cleaned up
231 AddSpace(space);
232
233 // Succeeds, fits without adjusting the footprint limit.
234 size_t ptr1_bytes_allocated, ptr1_usable_size, ptr1_bytes_tl_bulk_allocated;
235 StackHandleScope<3> hs(soa.Self());
236 MutableHandle<mirror::Object> ptr1(hs.NewHandle(Alloc(space,
237 self,
238 1 * MB,
239 &ptr1_bytes_allocated,
240 &ptr1_usable_size,
241 &ptr1_bytes_tl_bulk_allocated)));
242 EXPECT_TRUE(ptr1 != nullptr);
243 EXPECT_LE(1U * MB, ptr1_bytes_allocated);
244 EXPECT_LE(1U * MB, ptr1_usable_size);
245 EXPECT_LE(ptr1_usable_size, ptr1_bytes_allocated);
246 EXPECT_EQ(ptr1_bytes_tl_bulk_allocated, ptr1_bytes_allocated);
247
248 // Fails, requires a higher footprint limit.
249 mirror::Object* ptr2 = Alloc(space, self, 8 * MB, &unused, nullptr, &unused);
250 EXPECT_TRUE(ptr2 == nullptr);
251
252 // Succeeds, adjusts the footprint.
253 size_t ptr3_bytes_allocated, ptr3_usable_size, ptr3_bytes_tl_bulk_allocated;
254 MutableHandle<mirror::Object> ptr3(hs.NewHandle(AllocWithGrowth(space,
255 self,
256 8 * MB,
257 &ptr3_bytes_allocated,
258 &ptr3_usable_size,
259 &ptr3_bytes_tl_bulk_allocated)));
260 EXPECT_TRUE(ptr3 != nullptr);
261 EXPECT_LE(8U * MB, ptr3_bytes_allocated);
262 EXPECT_LE(8U * MB, ptr3_usable_size);
263 EXPECT_LE(ptr3_usable_size, ptr3_bytes_allocated);
264 EXPECT_EQ(ptr3_bytes_tl_bulk_allocated, ptr3_bytes_allocated);
265
266 // Fails, requires a higher footprint limit.
267 mirror::Object* ptr4 = Alloc(space, self, 8 * MB, &unused, nullptr, &unused);
268 EXPECT_TRUE(ptr4 == nullptr);
269
270 // Also fails, requires a higher allowed footprint.
271 mirror::Object* ptr5 = AllocWithGrowth(space, self, 8 * MB, &unused, nullptr, &unused);
272 EXPECT_TRUE(ptr5 == nullptr);
273
274 // Release some memory.
275 size_t free3 = space->AllocationSize(ptr3.Get(), nullptr);
276 EXPECT_EQ(free3, ptr3_bytes_allocated);
277 space->Free(self, ptr3.Assign(nullptr));
278 EXPECT_LE(8U * MB, free3);
279
280 // Succeeds, now that memory has been freed.
281 size_t ptr6_bytes_allocated, ptr6_usable_size, ptr6_bytes_tl_bulk_allocated;
282 Handle<mirror::Object> ptr6(hs.NewHandle(AllocWithGrowth(space,
283 self,
284 9 * MB,
285 &ptr6_bytes_allocated,
286 &ptr6_usable_size,
287 &ptr6_bytes_tl_bulk_allocated)));
288 EXPECT_TRUE(ptr6 != nullptr);
289 EXPECT_LE(9U * MB, ptr6_bytes_allocated);
290 EXPECT_LE(9U * MB, ptr6_usable_size);
291 EXPECT_LE(ptr6_usable_size, ptr6_bytes_allocated);
292 EXPECT_EQ(ptr6_bytes_tl_bulk_allocated, ptr6_bytes_allocated);
293
294 // Final clean up.
295 size_t free1 = space->AllocationSize(ptr1.Get(), nullptr);
296 space->Free(self, ptr1.Assign(nullptr));
297 EXPECT_LE(1U * MB, free1);
298 }
299
TEST_P(SpaceCreateTest,AllocAndFreeListTestBody)300 TEST_P(SpaceCreateTest, AllocAndFreeListTestBody) {
301 MallocSpace* space(CreateSpace("test", 4 * MB, 16 * MB, 16 * MB));
302 ASSERT_TRUE(space != nullptr);
303
304 // Make space findable to the heap, will also delete space when runtime is cleaned up
305 AddSpace(space);
306 Thread* self = Thread::Current();
307 ScopedObjectAccess soa(self);
308
309 // Succeeds, fits without adjusting the max allowed footprint.
310 mirror::Object* lots_of_objects[1024];
311 for (size_t i = 0; i < arraysize(lots_of_objects); i++) {
312 size_t allocation_size, usable_size, bytes_tl_bulk_allocated;
313 size_t size_of_zero_length_byte_array = SizeOfZeroLengthByteArray();
314 lots_of_objects[i] = Alloc(space,
315 self,
316 size_of_zero_length_byte_array,
317 &allocation_size,
318 &usable_size,
319 &bytes_tl_bulk_allocated);
320 EXPECT_TRUE(lots_of_objects[i] != nullptr);
321 size_t computed_usable_size;
322 EXPECT_EQ(allocation_size, space->AllocationSize(lots_of_objects[i], &computed_usable_size));
323 EXPECT_EQ(usable_size, computed_usable_size);
324 EXPECT_TRUE(bytes_tl_bulk_allocated == 0 ||
325 bytes_tl_bulk_allocated >= allocation_size);
326 }
327
328 // Release memory.
329 space->FreeList(self, arraysize(lots_of_objects), lots_of_objects);
330
331 // Succeeds, fits by adjusting the max allowed footprint.
332 for (size_t i = 0; i < arraysize(lots_of_objects); i++) {
333 size_t allocation_size, usable_size, bytes_tl_bulk_allocated;
334 lots_of_objects[i] = AllocWithGrowth(space,
335 self,
336 1024,
337 &allocation_size,
338 &usable_size,
339 &bytes_tl_bulk_allocated);
340 EXPECT_TRUE(lots_of_objects[i] != nullptr);
341 size_t computed_usable_size;
342 EXPECT_EQ(allocation_size, space->AllocationSize(lots_of_objects[i], &computed_usable_size));
343 EXPECT_EQ(usable_size, computed_usable_size);
344 EXPECT_TRUE(bytes_tl_bulk_allocated == 0 ||
345 bytes_tl_bulk_allocated >= allocation_size);
346 }
347
348 // Release memory.
349 space->FreeList(self, arraysize(lots_of_objects), lots_of_objects);
350 }
351
352 INSTANTIATE_TEST_CASE_P(CreateRosAllocSpace,
353 SpaceCreateTest,
354 testing::Values(kMallocSpaceRosAlloc));
355 INSTANTIATE_TEST_CASE_P(CreateDlMallocSpace,
356 SpaceCreateTest,
357 testing::Values(kMallocSpaceDlMalloc));
358
359 } // namespace space
360 } // namespace gc
361 } // namespace art
362