blob: dc9950c862155f2479f90f37c72b1a0935880096 [file] [log] [blame]
Carl Shapiro69759ea2011-07-21 18:13:35 -07001// Copyright 2011 Google Inc. All Rights Reserved.
2// Author: cshapiro@google.com (Carl Shapiro)
3
4#include "src/space.h"
5
6#include <sys/mman.h>
7
8#include "src/logging.h"
9#include "src/mspace.h"
10#include "src/scoped_ptr.h"
11#include "src/utils.h"
12
13namespace art {
14
15Space* Space::Create(size_t startup_size, size_t maximum_size) {
16 scoped_ptr<Space> space(new Space(startup_size, maximum_size));
17 bool success = space->Init();
18 if (!success) {
19 return NULL;
20 } else {
21 return space.release();
22 }
23}
24
25void* Space::CreateMallocSpace(void* base,
26 size_t startup_size,
27 size_t maximum_size) {
28 errno = 0;
29 bool is_locked = false;
30 size_t commit_size = startup_size / 2;
31 void* msp = create_contiguous_mspace_with_base(commit_size, maximum_size,
32 is_locked, base);
33 if (msp != NULL) {
34 // Do not permit the heap grow past the starting size without our
35 // intervention.
36 mspace_set_max_allowed_footprint(msp, startup_size);
37 } else {
38 // There is no guarantee that errno has meaning when the call
39 // fails, but it often does.
40 PLOG(ERROR) << "create_contiguous_mspace_with_base failed";
41 }
42 return msp;
43}
44
45bool Space::Init() {
46 if (!(startup_size_ <= maximum_size_)) {
47 return false;
48 }
49 size_t length = RoundUp(maximum_size_, 4096);
50 int prot = PROT_READ | PROT_WRITE;
51 int flags = MAP_PRIVATE | MAP_ANONYMOUS;
52 void* base = mmap(NULL, length, prot, flags, -1, 0);
53 if (base == MAP_FAILED) {
54 PLOG(ERROR) << "mmap failed";
55 return false;
56 }
57 base_ = static_cast<byte*>(base);
58 limit_ = base_ + length;
59 mspace_ = CreateMallocSpace(base, startup_size_, maximum_size_);
60 if (mspace_ == NULL) {
61 munmap(base_, length);
62 return false;
63 }
64 return true;
65}
66
67Space::~Space() {
68 if (base_ == NULL) {
69 return;
70 }
71 int result = munmap(base_, limit_ - base_);
72 if (result == -1) {
73 PLOG(WARNING) << "munmap failed";
74 }
75}
76
77Object* Space::AllocWithoutGrowth(size_t num_bytes) {
78 return reinterpret_cast<Object*>(mspace_calloc(mspace_, 1, num_bytes));
79}
80
81Object* Space::AllocWithGrowth(size_t num_bytes) {
82 // Grow as much as possible within the mspace.
83 size_t max_allowed = maximum_size_;
84 mspace_set_max_allowed_footprint(mspace_, max_allowed);
85 // Try the allocation.
86 void* ptr = AllocWithoutGrowth(num_bytes);
87 // Shrink back down as small as possible.
88 size_t footprint = mspace_footprint(mspace_);
89 mspace_set_max_allowed_footprint(mspace_, footprint);
90 // Return the new allocation or NULL.
91 return reinterpret_cast<Object*>(ptr);
92}
93
94size_t Space::Free(void* ptr) {
95 DCHECK(ptr != NULL);
96 size_t num_bytes = mspace_usable_size(mspace_, ptr);
97 mspace_free(mspace_, ptr);
98 return num_bytes;
99}
100
101void Space::DontNeed(void* start, void* end, void* num_bytes) {
102 start = (void*)RoundUp((uintptr_t)start, 4096);
103 end = (void*)RoundDown((uintptr_t)end, 4096);
104 if (start >= end) {
105 return;
106 }
107 size_t length = reinterpret_cast<byte*>(end) - reinterpret_cast<byte*>(start);
108 int result = madvise(start, length, MADV_DONTNEED);
109 if (result == -1) {
110 PLOG(WARNING) << "madvise failed";
111 } else {
112 *reinterpret_cast<size_t*>(num_bytes) += length;
113 }
114}
115
116void Space::Trim() {
117 CHECK(mspace_ != NULL);
118 mspace_trim(mspace_, 0);
119 size_t num_bytes_released = 0;
120 mspace_walk_free_pages(mspace_, DontNeed, &num_bytes_released);
121}
122
123size_t Space::MaxAllowedFootprint() {
124 return mspace_max_allowed_footprint(mspace_);
125}
126
127void Space::Grow(size_t new_size) {
128}
129
130} // namespace art