Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(134)

Side by Side Diff: third_party/tcmalloc/page_heap.cc

Issue 242088: Decommits when coallescing spans instead of committing. (Closed)
Patch Set: Rebase past the DEFER_DECOMMIT rollback Created 11 years, 2 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « third_party/tcmalloc/page_heap.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2008, Google Inc. 1 // Copyright (c) 2008, Google Inc.
2 // All rights reserved. 2 // All rights reserved.
3 // 3 //
4 // Redistribution and use in source and binary forms, with or without 4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are 5 // modification, are permitted provided that the following conditions are
6 // met: 6 // met:
7 // 7 //
8 // * Redistributions of source code must retain the above copyright 8 // * Redistributions of source code must retain the above copyright
9 // notice, this list of conditions and the following disclaimer. 9 // notice, this list of conditions and the following disclaimer.
10 // * Redistributions in binary form must reproduce the above 10 // * Redistributions in binary form must reproduce the above
(...skipping 139 matching lines...) Expand 10 before | Expand all | Expand 10 after
150 return leftover; 150 return leftover;
151 } 151 }
152 152
153 void PageHeap::CommitSpan(Span* span) { 153 void PageHeap::CommitSpan(Span* span) {
154 TCMalloc_SystemCommit( 154 TCMalloc_SystemCommit(
155 reinterpret_cast<void*>(span->start << kPageShift), 155 reinterpret_cast<void*>(span->start << kPageShift),
156 static_cast<size_t>(span->length << kPageShift) 156 static_cast<size_t>(span->length << kPageShift)
157 ); 157 );
158 } 158 }
159 159
160 void PageHeap::DecommitSpan(Span* span) {
161 TCMalloc_SystemRelease(reinterpret_cast<void*>(span->start << kPageShift),
162 static_cast<size_t>(span->length << kPageShift));
163 }
164
160 Span* PageHeap::Carve(Span* span, Length n) { 165 Span* PageHeap::Carve(Span* span, Length n) {
161 ASSERT(n > 0); 166 ASSERT(n > 0);
162 ASSERT(span->location != Span::IN_USE); 167 ASSERT(span->location != Span::IN_USE);
163 const int old_location = span->location; 168 const int old_location = span->location;
164 DLL_Remove(span); 169 DLL_Remove(span);
165 span->location = Span::IN_USE; 170 span->location = Span::IN_USE;
166 Event(span, 'A', n); 171 Event(span, 'A', n);
167 172
168 const int extra = span->length - n; 173 const int extra = span->length - n;
169 ASSERT(extra >= 0); 174 ASSERT(extra >= 0);
(...skipping 24 matching lines...) Expand all
194 } 199 }
195 200
196 void PageHeap::Delete(Span* span) { 201 void PageHeap::Delete(Span* span) {
197 ASSERT(Check()); 202 ASSERT(Check());
198 ASSERT(span->location == Span::IN_USE); 203 ASSERT(span->location == Span::IN_USE);
199 ASSERT(span->length > 0); 204 ASSERT(span->length > 0);
200 ASSERT(GetDescriptor(span->start) == span); 205 ASSERT(GetDescriptor(span->start) == span);
201 ASSERT(GetDescriptor(span->start + span->length - 1) == span); 206 ASSERT(GetDescriptor(span->start + span->length - 1) == span);
202 span->sizeclass = 0; 207 span->sizeclass = 0;
203 span->sample = 0; 208 span->sample = 0;
209 DecommitSpan(span);
204 210
205 // Coalesce -- we guarantee that "p" != 0, so no bounds checking 211 // Coalesce -- we guarantee that "p" != 0, so no bounds checking
206 // necessary. We do not bother resetting the stale pagemap 212 // necessary. We do not bother resetting the stale pagemap
207 // entries for the pieces we are merging together because we only 213 // entries for the pieces we are merging together because we only
208 // care about the pagemap entries for the boundaries. 214 // care about the pagemap entries for the boundaries.
209 // 215 //
210 // Note that the spans we merge into "span" may come out of 216 // Note that the spans we merge into "span" may come out of
211 // a "returned" list. We move those into "normal" list 217 // a "normal" or "returned" list. We move those into the
212 // as for 'immediate' operations we favour committing over 218 // "returned" list in order to favor decommitting over committing.
213 // decommitting (decommitting is performed offline).
214 const PageID p = span->start; 219 const PageID p = span->start;
215 const Length n = span->length; 220 const Length n = span->length;
216 Span* prev = GetDescriptor(p-1); 221 Span* prev = GetDescriptor(p-1);
217 if (prev != NULL && prev->location != Span::IN_USE) { 222 if (prev != NULL && prev->location != Span::IN_USE) {
218 // Merge preceding span into this span 223 // Merge preceding span into this span
219 ASSERT(prev->start + prev->length == p); 224 ASSERT(prev->start + prev->length == p);
220 const Length len = prev->length; 225 const Length len = prev->length;
221 if (prev->location == Span::ON_RETURNED_FREELIST) { 226 if (prev->location == Span::ON_NORMAL_FREELIST) {
222 CommitSpan(prev); 227 DecommitSpan(prev);
223 } 228 }
224 DLL_Remove(prev); 229 DLL_Remove(prev);
225 DeleteSpan(prev); 230 DeleteSpan(prev);
226 span->start -= len; 231 span->start -= len;
227 span->length += len; 232 span->length += len;
228 pagemap_.set(span->start, span); 233 pagemap_.set(span->start, span);
229 Event(span, 'L', len); 234 Event(span, 'L', len);
230 } 235 }
231 Span* next = GetDescriptor(p+n); 236 Span* next = GetDescriptor(p+n);
232 if (next != NULL && next->location != Span::IN_USE) { 237 if (next != NULL && next->location != Span::IN_USE) {
233 // Merge next span into this span 238 // Merge next span into this span
234 ASSERT(next->start == p+n); 239 ASSERT(next->start == p+n);
235 const Length len = next->length; 240 const Length len = next->length;
236 if (next->location == Span::ON_RETURNED_FREELIST) { 241 if (next->location == Span::ON_NORMAL_FREELIST) {
237 CommitSpan(next); 242 DecommitSpan(next);
238 } 243 }
239 DLL_Remove(next); 244 DLL_Remove(next);
240 DeleteSpan(next); 245 DeleteSpan(next);
241 span->length += len; 246 span->length += len;
242 pagemap_.set(span->start + span->length - 1, span); 247 pagemap_.set(span->start + span->length - 1, span);
243 Event(span, 'R', len); 248 Event(span, 'R', len);
244 } 249 }
245 250
246 Event(span, 'D', span->length); 251 Event(span, 'D', span->length);
antonm 2009/10/01 11:59:27 why you don't decommit the span itself? is it by
247 span->location = Span::ON_NORMAL_FREELIST; 252 span->location = Span::ON_RETURNED_FREELIST;
248 if (span->length < kMaxPages) { 253 if (span->length < kMaxPages) {
249 DLL_Prepend(&free_[span->length].normal, span); 254 DLL_Prepend(&free_[span->length].normal, span);
Mike Belshe 2009/10/01 02:39:48 I think this is a bug - DLL_Prepend(&free_[span->
antonm 2009/10/01 11:59:27 That's definitely a bug and I would expect immedia
250 } else { 255 } else {
251 DLL_Prepend(&large_.normal, span); 256 DLL_Prepend(&large_.normal, span);
Mike Belshe 2009/10/01 02:39:48 ditto
252 } 257 }
253 free_pages_ += n; 258 free_pages_ += n;
254 259
255 IncrementalScavenge(n); 260 IncrementalScavenge(n);
256 ASSERT(Check()); 261 ASSERT(Check());
257 } 262 }
258 263
259 void PageHeap::IncrementalScavenge(Length n) { 264 void PageHeap::IncrementalScavenge(Length n) {
260 // Fast path; not yet time to release memory 265 // Fast path; not yet time to release memory
261 scavenge_counter_ -= n; 266 scavenge_counter_ -= n;
(...skipping 235 matching lines...) Expand 10 before | Expand all | Expand 10 after
497 502
498 void PageHeap::ReleaseFreePages() { 503 void PageHeap::ReleaseFreePages() {
499 for (Length s = 0; s < kMaxPages; s++) { 504 for (Length s = 0; s < kMaxPages; s++) {
500 ReleaseFreeList(&free_[s].normal, &free_[s].returned); 505 ReleaseFreeList(&free_[s].normal, &free_[s].returned);
501 } 506 }
502 ReleaseFreeList(&large_.normal, &large_.returned); 507 ReleaseFreeList(&large_.normal, &large_.returned);
503 ASSERT(Check()); 508 ASSERT(Check());
504 } 509 }
505 510
506 } // namespace tcmalloc 511 } // namespace tcmalloc
OLDNEW
« no previous file with comments | « third_party/tcmalloc/page_heap.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698