annotate src/share/vm/gc_implementation/concurrentMarkSweep/compactibleFreeListSpace.cpp @ 517:e9be0e04635a

6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off Summary: Added safe_object_iterate() for use by JMapPerm. Reviewed-by: tonyp
author jmasa
date Tue, 06 Jan 2009 07:05:05 -0800
parents 5d254928c888
children 0af8b0718fc9
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1 /*
196
d1605aabd0a1 6719955: Update copyright year
xdono
parents: 187
diff changeset
2 * Copyright 2001-2008 Sun Microsystems, Inc. All Rights Reserved.
0
a61af66fc99e Initial load
duke
parents:
diff changeset
3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
a61af66fc99e Initial load
duke
parents:
diff changeset
4 *
a61af66fc99e Initial load
duke
parents:
diff changeset
5 * This code is free software; you can redistribute it and/or modify it
a61af66fc99e Initial load
duke
parents:
diff changeset
6 * under the terms of the GNU General Public License version 2 only, as
a61af66fc99e Initial load
duke
parents:
diff changeset
7 * published by the Free Software Foundation.
a61af66fc99e Initial load
duke
parents:
diff changeset
8 *
a61af66fc99e Initial load
duke
parents:
diff changeset
9 * This code is distributed in the hope that it will be useful, but WITHOUT
a61af66fc99e Initial load
duke
parents:
diff changeset
10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
a61af66fc99e Initial load
duke
parents:
diff changeset
11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
a61af66fc99e Initial load
duke
parents:
diff changeset
12 * version 2 for more details (a copy is included in the LICENSE file that
a61af66fc99e Initial load
duke
parents:
diff changeset
13 * accompanied this code).
a61af66fc99e Initial load
duke
parents:
diff changeset
14 *
a61af66fc99e Initial load
duke
parents:
diff changeset
15 * You should have received a copy of the GNU General Public License version
a61af66fc99e Initial load
duke
parents:
diff changeset
16 * 2 along with this work; if not, write to the Free Software Foundation,
a61af66fc99e Initial load
duke
parents:
diff changeset
17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
a61af66fc99e Initial load
duke
parents:
diff changeset
18 *
a61af66fc99e Initial load
duke
parents:
diff changeset
19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
a61af66fc99e Initial load
duke
parents:
diff changeset
20 * CA 95054 USA or visit www.sun.com if you need additional information or
a61af66fc99e Initial load
duke
parents:
diff changeset
21 * have any questions.
a61af66fc99e Initial load
duke
parents:
diff changeset
22 *
a61af66fc99e Initial load
duke
parents:
diff changeset
23 */
a61af66fc99e Initial load
duke
parents:
diff changeset
24
a61af66fc99e Initial load
duke
parents:
diff changeset
25 # include "incls/_precompiled.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
26 # include "incls/_compactibleFreeListSpace.cpp.incl"
a61af66fc99e Initial load
duke
parents:
diff changeset
27
a61af66fc99e Initial load
duke
parents:
diff changeset
28 /////////////////////////////////////////////////////////////////////////
a61af66fc99e Initial load
duke
parents:
diff changeset
29 //// CompactibleFreeListSpace
a61af66fc99e Initial load
duke
parents:
diff changeset
30 /////////////////////////////////////////////////////////////////////////
a61af66fc99e Initial load
duke
parents:
diff changeset
31
a61af66fc99e Initial load
duke
parents:
diff changeset
32 // highest ranked free list lock rank
a61af66fc99e Initial load
duke
parents:
diff changeset
33 int CompactibleFreeListSpace::_lockRank = Mutex::leaf + 3;
a61af66fc99e Initial load
duke
parents:
diff changeset
34
a61af66fc99e Initial load
duke
parents:
diff changeset
35 // Constructor
a61af66fc99e Initial load
duke
parents:
diff changeset
36 CompactibleFreeListSpace::CompactibleFreeListSpace(BlockOffsetSharedArray* bs,
a61af66fc99e Initial load
duke
parents:
diff changeset
37 MemRegion mr, bool use_adaptive_freelists,
a61af66fc99e Initial load
duke
parents:
diff changeset
38 FreeBlockDictionary::DictionaryChoice dictionaryChoice) :
a61af66fc99e Initial load
duke
parents:
diff changeset
39 _dictionaryChoice(dictionaryChoice),
a61af66fc99e Initial load
duke
parents:
diff changeset
40 _adaptive_freelists(use_adaptive_freelists),
a61af66fc99e Initial load
duke
parents:
diff changeset
41 _bt(bs, mr),
a61af66fc99e Initial load
duke
parents:
diff changeset
42 // free list locks are in the range of values taken by _lockRank
a61af66fc99e Initial load
duke
parents:
diff changeset
43 // This range currently is [_leaf+2, _leaf+3]
a61af66fc99e Initial load
duke
parents:
diff changeset
44 // Note: this requires that CFLspace c'tors
a61af66fc99e Initial load
duke
parents:
diff changeset
45 // are called serially in the order in which the locks are
a61af66fc99e Initial load
duke
parents:
diff changeset
46 // are acquired in the program text. This is true today.
a61af66fc99e Initial load
duke
parents:
diff changeset
47 _freelistLock(_lockRank--, "CompactibleFreeListSpace._lock", true),
a61af66fc99e Initial load
duke
parents:
diff changeset
48 _parDictionaryAllocLock(Mutex::leaf - 1, // == rank(ExpandHeap_lock) - 1
a61af66fc99e Initial load
duke
parents:
diff changeset
49 "CompactibleFreeListSpace._dict_par_lock", true),
a61af66fc99e Initial load
duke
parents:
diff changeset
50 _rescan_task_size(CardTableModRefBS::card_size_in_words * BitsPerWord *
a61af66fc99e Initial load
duke
parents:
diff changeset
51 CMSRescanMultiple),
a61af66fc99e Initial load
duke
parents:
diff changeset
52 _marking_task_size(CardTableModRefBS::card_size_in_words * BitsPerWord *
a61af66fc99e Initial load
duke
parents:
diff changeset
53 CMSConcMarkMultiple),
a61af66fc99e Initial load
duke
parents:
diff changeset
54 _collector(NULL)
a61af66fc99e Initial load
duke
parents:
diff changeset
55 {
a61af66fc99e Initial load
duke
parents:
diff changeset
56 _bt.set_space(this);
263
12eea04c8b06 6672698: mangle_unused_area() should not remangle the entire heap at each collection.
jmasa
parents: 187
diff changeset
57 initialize(mr, SpaceDecorator::Clear, SpaceDecorator::Mangle);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
58 // We have all of "mr", all of which we place in the dictionary
a61af66fc99e Initial load
duke
parents:
diff changeset
59 // as one big chunk. We'll need to decide here which of several
a61af66fc99e Initial load
duke
parents:
diff changeset
60 // possible alternative dictionary implementations to use. For
a61af66fc99e Initial load
duke
parents:
diff changeset
61 // now the choice is easy, since we have only one working
a61af66fc99e Initial load
duke
parents:
diff changeset
62 // implementation, namely, the simple binary tree (splaying
a61af66fc99e Initial load
duke
parents:
diff changeset
63 // temporarily disabled).
a61af66fc99e Initial load
duke
parents:
diff changeset
64 switch (dictionaryChoice) {
a61af66fc99e Initial load
duke
parents:
diff changeset
65 case FreeBlockDictionary::dictionaryBinaryTree:
a61af66fc99e Initial load
duke
parents:
diff changeset
66 _dictionary = new BinaryTreeDictionary(mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
67 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
68 case FreeBlockDictionary::dictionarySplayTree:
a61af66fc99e Initial load
duke
parents:
diff changeset
69 case FreeBlockDictionary::dictionarySkipList:
a61af66fc99e Initial load
duke
parents:
diff changeset
70 default:
a61af66fc99e Initial load
duke
parents:
diff changeset
71 warning("dictionaryChoice: selected option not understood; using"
a61af66fc99e Initial load
duke
parents:
diff changeset
72 " default BinaryTreeDictionary implementation instead.");
a61af66fc99e Initial load
duke
parents:
diff changeset
73 _dictionary = new BinaryTreeDictionary(mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
74 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
75 }
a61af66fc99e Initial load
duke
parents:
diff changeset
76 splitBirth(mr.word_size());
a61af66fc99e Initial load
duke
parents:
diff changeset
77 assert(_dictionary != NULL, "CMS dictionary initialization");
a61af66fc99e Initial load
duke
parents:
diff changeset
78 // The indexed free lists are initially all empty and are lazily
a61af66fc99e Initial load
duke
parents:
diff changeset
79 // filled in on demand. Initialize the array elements to NULL.
a61af66fc99e Initial load
duke
parents:
diff changeset
80 initializeIndexedFreeListArray();
a61af66fc99e Initial load
duke
parents:
diff changeset
81
a61af66fc99e Initial load
duke
parents:
diff changeset
82 // Not using adaptive free lists assumes that allocation is first
a61af66fc99e Initial load
duke
parents:
diff changeset
83 // from the linAB's. Also a cms perm gen which can be compacted
a61af66fc99e Initial load
duke
parents:
diff changeset
84 // has to have the klass's klassKlass allocated at a lower
a61af66fc99e Initial load
duke
parents:
diff changeset
85 // address in the heap than the klass so that the klassKlass is
a61af66fc99e Initial load
duke
parents:
diff changeset
86 // moved to its new location before the klass is moved.
a61af66fc99e Initial load
duke
parents:
diff changeset
87 // Set the _refillSize for the linear allocation blocks
a61af66fc99e Initial load
duke
parents:
diff changeset
88 if (!use_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
89 FreeChunk* fc = _dictionary->getChunk(mr.word_size());
a61af66fc99e Initial load
duke
parents:
diff changeset
90 // The small linAB initially has all the space and will allocate
a61af66fc99e Initial load
duke
parents:
diff changeset
91 // a chunk of any size.
a61af66fc99e Initial load
duke
parents:
diff changeset
92 HeapWord* addr = (HeapWord*) fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
93 _smallLinearAllocBlock.set(addr, fc->size() ,
a61af66fc99e Initial load
duke
parents:
diff changeset
94 1024*SmallForLinearAlloc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
95 // Note that _unallocated_block is not updated here.
a61af66fc99e Initial load
duke
parents:
diff changeset
96 // Allocations from the linear allocation block should
a61af66fc99e Initial load
duke
parents:
diff changeset
97 // update it.
a61af66fc99e Initial load
duke
parents:
diff changeset
98 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
99 _smallLinearAllocBlock.set(0, 0, 1024*SmallForLinearAlloc,
a61af66fc99e Initial load
duke
parents:
diff changeset
100 SmallForLinearAlloc);
a61af66fc99e Initial load
duke
parents:
diff changeset
101 }
a61af66fc99e Initial load
duke
parents:
diff changeset
102 // CMSIndexedFreeListReplenish should be at least 1
a61af66fc99e Initial load
duke
parents:
diff changeset
103 CMSIndexedFreeListReplenish = MAX2((uintx)1, CMSIndexedFreeListReplenish);
a61af66fc99e Initial load
duke
parents:
diff changeset
104 _promoInfo.setSpace(this);
a61af66fc99e Initial load
duke
parents:
diff changeset
105 if (UseCMSBestFit) {
a61af66fc99e Initial load
duke
parents:
diff changeset
106 _fitStrategy = FreeBlockBestFitFirst;
a61af66fc99e Initial load
duke
parents:
diff changeset
107 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
108 _fitStrategy = FreeBlockStrategyNone;
a61af66fc99e Initial load
duke
parents:
diff changeset
109 }
a61af66fc99e Initial load
duke
parents:
diff changeset
110 checkFreeListConsistency();
a61af66fc99e Initial load
duke
parents:
diff changeset
111
a61af66fc99e Initial load
duke
parents:
diff changeset
112 // Initialize locks for parallel case.
a61af66fc99e Initial load
duke
parents:
diff changeset
113 if (ParallelGCThreads > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
114 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
115 _indexedFreeListParLocks[i] = new Mutex(Mutex::leaf - 1, // == ExpandHeap_lock - 1
a61af66fc99e Initial load
duke
parents:
diff changeset
116 "a freelist par lock",
a61af66fc99e Initial load
duke
parents:
diff changeset
117 true);
a61af66fc99e Initial load
duke
parents:
diff changeset
118 if (_indexedFreeListParLocks[i] == NULL)
a61af66fc99e Initial load
duke
parents:
diff changeset
119 vm_exit_during_initialization("Could not allocate a par lock");
a61af66fc99e Initial load
duke
parents:
diff changeset
120 DEBUG_ONLY(
a61af66fc99e Initial load
duke
parents:
diff changeset
121 _indexedFreeList[i].set_protecting_lock(_indexedFreeListParLocks[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
122 )
a61af66fc99e Initial load
duke
parents:
diff changeset
123 }
a61af66fc99e Initial load
duke
parents:
diff changeset
124 _dictionary->set_par_lock(&_parDictionaryAllocLock);
a61af66fc99e Initial load
duke
parents:
diff changeset
125 }
a61af66fc99e Initial load
duke
parents:
diff changeset
126 }
a61af66fc99e Initial load
duke
parents:
diff changeset
127
a61af66fc99e Initial load
duke
parents:
diff changeset
128 // Like CompactibleSpace forward() but always calls cross_threshold() to
a61af66fc99e Initial load
duke
parents:
diff changeset
129 // update the block offset table. Removed initialize_threshold call because
a61af66fc99e Initial load
duke
parents:
diff changeset
130 // CFLS does not use a block offset array for contiguous spaces.
a61af66fc99e Initial load
duke
parents:
diff changeset
131 HeapWord* CompactibleFreeListSpace::forward(oop q, size_t size,
a61af66fc99e Initial load
duke
parents:
diff changeset
132 CompactPoint* cp, HeapWord* compact_top) {
a61af66fc99e Initial load
duke
parents:
diff changeset
133 // q is alive
a61af66fc99e Initial load
duke
parents:
diff changeset
134 // First check if we should switch compaction space
a61af66fc99e Initial load
duke
parents:
diff changeset
135 assert(this == cp->space, "'this' should be current compaction space.");
a61af66fc99e Initial load
duke
parents:
diff changeset
136 size_t compaction_max_size = pointer_delta(end(), compact_top);
a61af66fc99e Initial load
duke
parents:
diff changeset
137 assert(adjustObjectSize(size) == cp->space->adjust_object_size_v(size),
a61af66fc99e Initial load
duke
parents:
diff changeset
138 "virtual adjustObjectSize_v() method is not correct");
a61af66fc99e Initial load
duke
parents:
diff changeset
139 size_t adjusted_size = adjustObjectSize(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
140 assert(compaction_max_size >= MinChunkSize || compaction_max_size == 0,
a61af66fc99e Initial load
duke
parents:
diff changeset
141 "no small fragments allowed");
a61af66fc99e Initial load
duke
parents:
diff changeset
142 assert(minimum_free_block_size() == MinChunkSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
143 "for de-virtualized reference below");
a61af66fc99e Initial load
duke
parents:
diff changeset
144 // Can't leave a nonzero size, residual fragment smaller than MinChunkSize
a61af66fc99e Initial load
duke
parents:
diff changeset
145 if (adjusted_size + MinChunkSize > compaction_max_size &&
a61af66fc99e Initial load
duke
parents:
diff changeset
146 adjusted_size != compaction_max_size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
147 do {
a61af66fc99e Initial load
duke
parents:
diff changeset
148 // switch to next compaction space
a61af66fc99e Initial load
duke
parents:
diff changeset
149 cp->space->set_compaction_top(compact_top);
a61af66fc99e Initial load
duke
parents:
diff changeset
150 cp->space = cp->space->next_compaction_space();
a61af66fc99e Initial load
duke
parents:
diff changeset
151 if (cp->space == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
152 cp->gen = GenCollectedHeap::heap()->prev_gen(cp->gen);
a61af66fc99e Initial load
duke
parents:
diff changeset
153 assert(cp->gen != NULL, "compaction must succeed");
a61af66fc99e Initial load
duke
parents:
diff changeset
154 cp->space = cp->gen->first_compaction_space();
a61af66fc99e Initial load
duke
parents:
diff changeset
155 assert(cp->space != NULL, "generation must have a first compaction space");
a61af66fc99e Initial load
duke
parents:
diff changeset
156 }
a61af66fc99e Initial load
duke
parents:
diff changeset
157 compact_top = cp->space->bottom();
a61af66fc99e Initial load
duke
parents:
diff changeset
158 cp->space->set_compaction_top(compact_top);
a61af66fc99e Initial load
duke
parents:
diff changeset
159 // The correct adjusted_size may not be the same as that for this method
a61af66fc99e Initial load
duke
parents:
diff changeset
160 // (i.e., cp->space may no longer be "this" so adjust the size again.
a61af66fc99e Initial load
duke
parents:
diff changeset
161 // Use the virtual method which is not used above to save the virtual
a61af66fc99e Initial load
duke
parents:
diff changeset
162 // dispatch.
a61af66fc99e Initial load
duke
parents:
diff changeset
163 adjusted_size = cp->space->adjust_object_size_v(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
164 compaction_max_size = pointer_delta(cp->space->end(), compact_top);
a61af66fc99e Initial load
duke
parents:
diff changeset
165 assert(cp->space->minimum_free_block_size() == 0, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
166 } while (adjusted_size > compaction_max_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
167 }
a61af66fc99e Initial load
duke
parents:
diff changeset
168
a61af66fc99e Initial load
duke
parents:
diff changeset
169 // store the forwarding pointer into the mark word
a61af66fc99e Initial load
duke
parents:
diff changeset
170 if ((HeapWord*)q != compact_top) {
a61af66fc99e Initial load
duke
parents:
diff changeset
171 q->forward_to(oop(compact_top));
a61af66fc99e Initial load
duke
parents:
diff changeset
172 assert(q->is_gc_marked(), "encoding the pointer should preserve the mark");
a61af66fc99e Initial load
duke
parents:
diff changeset
173 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
174 // if the object isn't moving we can just set the mark to the default
a61af66fc99e Initial load
duke
parents:
diff changeset
175 // mark and handle it specially later on.
a61af66fc99e Initial load
duke
parents:
diff changeset
176 q->init_mark();
a61af66fc99e Initial load
duke
parents:
diff changeset
177 assert(q->forwardee() == NULL, "should be forwarded to NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
178 }
a61af66fc99e Initial load
duke
parents:
diff changeset
179
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
180 VALIDATE_MARK_SWEEP_ONLY(MarkSweep::register_live_oop(q, adjusted_size));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
181 compact_top += adjusted_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
182
a61af66fc99e Initial load
duke
parents:
diff changeset
183 // we need to update the offset table so that the beginnings of objects can be
a61af66fc99e Initial load
duke
parents:
diff changeset
184 // found during scavenge. Note that we are updating the offset table based on
a61af66fc99e Initial load
duke
parents:
diff changeset
185 // where the object will be once the compaction phase finishes.
a61af66fc99e Initial load
duke
parents:
diff changeset
186
a61af66fc99e Initial load
duke
parents:
diff changeset
187 // Always call cross_threshold(). A contiguous space can only call it when
a61af66fc99e Initial load
duke
parents:
diff changeset
188 // the compaction_top exceeds the current threshold but not for an
a61af66fc99e Initial load
duke
parents:
diff changeset
189 // non-contiguous space.
a61af66fc99e Initial load
duke
parents:
diff changeset
190 cp->threshold =
a61af66fc99e Initial load
duke
parents:
diff changeset
191 cp->space->cross_threshold(compact_top - adjusted_size, compact_top);
a61af66fc99e Initial load
duke
parents:
diff changeset
192 return compact_top;
a61af66fc99e Initial load
duke
parents:
diff changeset
193 }
a61af66fc99e Initial load
duke
parents:
diff changeset
194
a61af66fc99e Initial load
duke
parents:
diff changeset
195 // A modified copy of OffsetTableContigSpace::cross_threshold() with _offsets -> _bt
a61af66fc99e Initial load
duke
parents:
diff changeset
196 // and use of single_block instead of alloc_block. The name here is not really
a61af66fc99e Initial load
duke
parents:
diff changeset
197 // appropriate - maybe a more general name could be invented for both the
a61af66fc99e Initial load
duke
parents:
diff changeset
198 // contiguous and noncontiguous spaces.
a61af66fc99e Initial load
duke
parents:
diff changeset
199
a61af66fc99e Initial load
duke
parents:
diff changeset
200 HeapWord* CompactibleFreeListSpace::cross_threshold(HeapWord* start, HeapWord* the_end) {
a61af66fc99e Initial load
duke
parents:
diff changeset
201 _bt.single_block(start, the_end);
a61af66fc99e Initial load
duke
parents:
diff changeset
202 return end();
a61af66fc99e Initial load
duke
parents:
diff changeset
203 }
a61af66fc99e Initial load
duke
parents:
diff changeset
204
a61af66fc99e Initial load
duke
parents:
diff changeset
205 // Initialize them to NULL.
a61af66fc99e Initial load
duke
parents:
diff changeset
206 void CompactibleFreeListSpace::initializeIndexedFreeListArray() {
a61af66fc99e Initial load
duke
parents:
diff changeset
207 for (size_t i = 0; i < IndexSetSize; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
208 // Note that on platforms where objects are double word aligned,
a61af66fc99e Initial load
duke
parents:
diff changeset
209 // the odd array elements are not used. It is convenient, however,
a61af66fc99e Initial load
duke
parents:
diff changeset
210 // to map directly from the object size to the array element.
a61af66fc99e Initial load
duke
parents:
diff changeset
211 _indexedFreeList[i].reset(IndexSetSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
212 _indexedFreeList[i].set_size(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
213 assert(_indexedFreeList[i].count() == 0, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
214 assert(_indexedFreeList[i].head() == NULL, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
215 assert(_indexedFreeList[i].tail() == NULL, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
216 assert(_indexedFreeList[i].hint() == IndexSetSize, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
217 }
a61af66fc99e Initial load
duke
parents:
diff changeset
218 }
a61af66fc99e Initial load
duke
parents:
diff changeset
219
a61af66fc99e Initial load
duke
parents:
diff changeset
220 void CompactibleFreeListSpace::resetIndexedFreeListArray() {
a61af66fc99e Initial load
duke
parents:
diff changeset
221 for (int i = 1; i < IndexSetSize; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
222 assert(_indexedFreeList[i].size() == (size_t) i,
a61af66fc99e Initial load
duke
parents:
diff changeset
223 "Indexed free list sizes are incorrect");
a61af66fc99e Initial load
duke
parents:
diff changeset
224 _indexedFreeList[i].reset(IndexSetSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
225 assert(_indexedFreeList[i].count() == 0, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
226 assert(_indexedFreeList[i].head() == NULL, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
227 assert(_indexedFreeList[i].tail() == NULL, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
228 assert(_indexedFreeList[i].hint() == IndexSetSize, "reset check failed");
a61af66fc99e Initial load
duke
parents:
diff changeset
229 }
a61af66fc99e Initial load
duke
parents:
diff changeset
230 }
a61af66fc99e Initial load
duke
parents:
diff changeset
231
a61af66fc99e Initial load
duke
parents:
diff changeset
232 void CompactibleFreeListSpace::reset(MemRegion mr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
233 resetIndexedFreeListArray();
a61af66fc99e Initial load
duke
parents:
diff changeset
234 dictionary()->reset();
a61af66fc99e Initial load
duke
parents:
diff changeset
235 if (BlockOffsetArrayUseUnallocatedBlock) {
a61af66fc99e Initial load
duke
parents:
diff changeset
236 assert(end() == mr.end(), "We are compacting to the bottom of CMS gen");
a61af66fc99e Initial load
duke
parents:
diff changeset
237 // Everything's allocated until proven otherwise.
a61af66fc99e Initial load
duke
parents:
diff changeset
238 _bt.set_unallocated_block(end());
a61af66fc99e Initial load
duke
parents:
diff changeset
239 }
a61af66fc99e Initial load
duke
parents:
diff changeset
240 if (!mr.is_empty()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
241 assert(mr.word_size() >= MinChunkSize, "Chunk size is too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
242 _bt.single_block(mr.start(), mr.word_size());
a61af66fc99e Initial load
duke
parents:
diff changeset
243 FreeChunk* fc = (FreeChunk*) mr.start();
a61af66fc99e Initial load
duke
parents:
diff changeset
244 fc->setSize(mr.word_size());
a61af66fc99e Initial load
duke
parents:
diff changeset
245 if (mr.word_size() >= IndexSetSize ) {
a61af66fc99e Initial load
duke
parents:
diff changeset
246 returnChunkToDictionary(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
247 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
248 _bt.verify_not_unallocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
249 _indexedFreeList[mr.word_size()].returnChunkAtHead(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
250 }
a61af66fc99e Initial load
duke
parents:
diff changeset
251 }
a61af66fc99e Initial load
duke
parents:
diff changeset
252 _promoInfo.reset();
a61af66fc99e Initial load
duke
parents:
diff changeset
253 _smallLinearAllocBlock._ptr = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
254 _smallLinearAllocBlock._word_size = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
255 }
a61af66fc99e Initial load
duke
parents:
diff changeset
256
a61af66fc99e Initial load
duke
parents:
diff changeset
257 void CompactibleFreeListSpace::reset_after_compaction() {
a61af66fc99e Initial load
duke
parents:
diff changeset
258 // Reset the space to the new reality - one free chunk.
a61af66fc99e Initial load
duke
parents:
diff changeset
259 MemRegion mr(compaction_top(), end());
a61af66fc99e Initial load
duke
parents:
diff changeset
260 reset(mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
261 // Now refill the linear allocation block(s) if possible.
a61af66fc99e Initial load
duke
parents:
diff changeset
262 if (_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
263 refillLinearAllocBlocksIfNeeded();
a61af66fc99e Initial load
duke
parents:
diff changeset
264 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
265 // Place as much of mr in the linAB as we can get,
a61af66fc99e Initial load
duke
parents:
diff changeset
266 // provided it was big enough to go into the dictionary.
a61af66fc99e Initial load
duke
parents:
diff changeset
267 FreeChunk* fc = dictionary()->findLargestDict();
a61af66fc99e Initial load
duke
parents:
diff changeset
268 if (fc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
269 assert(fc->size() == mr.word_size(),
a61af66fc99e Initial load
duke
parents:
diff changeset
270 "Why was the chunk broken up?");
a61af66fc99e Initial load
duke
parents:
diff changeset
271 removeChunkFromDictionary(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
272 HeapWord* addr = (HeapWord*) fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
273 _smallLinearAllocBlock.set(addr, fc->size() ,
a61af66fc99e Initial load
duke
parents:
diff changeset
274 1024*SmallForLinearAlloc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
275 // Note that _unallocated_block is not updated here.
a61af66fc99e Initial load
duke
parents:
diff changeset
276 }
a61af66fc99e Initial load
duke
parents:
diff changeset
277 }
a61af66fc99e Initial load
duke
parents:
diff changeset
278 }
a61af66fc99e Initial load
duke
parents:
diff changeset
279
a61af66fc99e Initial load
duke
parents:
diff changeset
280 // Walks the entire dictionary, returning a coterminal
a61af66fc99e Initial load
duke
parents:
diff changeset
281 // chunk, if it exists. Use with caution since it involves
a61af66fc99e Initial load
duke
parents:
diff changeset
282 // a potentially complete walk of a potentially large tree.
a61af66fc99e Initial load
duke
parents:
diff changeset
283 FreeChunk* CompactibleFreeListSpace::find_chunk_at_end() {
a61af66fc99e Initial load
duke
parents:
diff changeset
284
a61af66fc99e Initial load
duke
parents:
diff changeset
285 assert_lock_strong(&_freelistLock);
a61af66fc99e Initial load
duke
parents:
diff changeset
286
a61af66fc99e Initial load
duke
parents:
diff changeset
287 return dictionary()->find_chunk_ends_at(end());
a61af66fc99e Initial load
duke
parents:
diff changeset
288 }
a61af66fc99e Initial load
duke
parents:
diff changeset
289
a61af66fc99e Initial load
duke
parents:
diff changeset
290
a61af66fc99e Initial load
duke
parents:
diff changeset
291 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
292 void CompactibleFreeListSpace::initializeIndexedFreeListArrayReturnedBytes() {
a61af66fc99e Initial load
duke
parents:
diff changeset
293 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
294 _indexedFreeList[i].allocation_stats()->set_returnedBytes(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
295 }
a61af66fc99e Initial load
duke
parents:
diff changeset
296 }
a61af66fc99e Initial load
duke
parents:
diff changeset
297
a61af66fc99e Initial load
duke
parents:
diff changeset
298 size_t CompactibleFreeListSpace::sumIndexedFreeListArrayReturnedBytes() {
a61af66fc99e Initial load
duke
parents:
diff changeset
299 size_t sum = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
300 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
301 sum += _indexedFreeList[i].allocation_stats()->returnedBytes();
a61af66fc99e Initial load
duke
parents:
diff changeset
302 }
a61af66fc99e Initial load
duke
parents:
diff changeset
303 return sum;
a61af66fc99e Initial load
duke
parents:
diff changeset
304 }
a61af66fc99e Initial load
duke
parents:
diff changeset
305
a61af66fc99e Initial load
duke
parents:
diff changeset
306 size_t CompactibleFreeListSpace::totalCountInIndexedFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
307 size_t count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
308 for (int i = MinChunkSize; i < IndexSetSize; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
309 debug_only(
a61af66fc99e Initial load
duke
parents:
diff changeset
310 ssize_t total_list_count = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
311 for (FreeChunk* fc = _indexedFreeList[i].head(); fc != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
312 fc = fc->next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
313 total_list_count++;
a61af66fc99e Initial load
duke
parents:
diff changeset
314 }
a61af66fc99e Initial load
duke
parents:
diff changeset
315 assert(total_list_count == _indexedFreeList[i].count(),
a61af66fc99e Initial load
duke
parents:
diff changeset
316 "Count in list is incorrect");
a61af66fc99e Initial load
duke
parents:
diff changeset
317 )
a61af66fc99e Initial load
duke
parents:
diff changeset
318 count += _indexedFreeList[i].count();
a61af66fc99e Initial load
duke
parents:
diff changeset
319 }
a61af66fc99e Initial load
duke
parents:
diff changeset
320 return count;
a61af66fc99e Initial load
duke
parents:
diff changeset
321 }
a61af66fc99e Initial load
duke
parents:
diff changeset
322
a61af66fc99e Initial load
duke
parents:
diff changeset
323 size_t CompactibleFreeListSpace::totalCount() {
a61af66fc99e Initial load
duke
parents:
diff changeset
324 size_t num = totalCountInIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
325 num += dictionary()->totalCount();
a61af66fc99e Initial load
duke
parents:
diff changeset
326 if (_smallLinearAllocBlock._word_size != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
327 num++;
a61af66fc99e Initial load
duke
parents:
diff changeset
328 }
a61af66fc99e Initial load
duke
parents:
diff changeset
329 return num;
a61af66fc99e Initial load
duke
parents:
diff changeset
330 }
a61af66fc99e Initial load
duke
parents:
diff changeset
331 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
332
a61af66fc99e Initial load
duke
parents:
diff changeset
333 bool CompactibleFreeListSpace::is_free_block(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
334 FreeChunk* fc = (FreeChunk*) p;
a61af66fc99e Initial load
duke
parents:
diff changeset
335 return fc->isFree();
a61af66fc99e Initial load
duke
parents:
diff changeset
336 }
a61af66fc99e Initial load
duke
parents:
diff changeset
337
a61af66fc99e Initial load
duke
parents:
diff changeset
338 size_t CompactibleFreeListSpace::used() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
339 return capacity() - free();
a61af66fc99e Initial load
duke
parents:
diff changeset
340 }
a61af66fc99e Initial load
duke
parents:
diff changeset
341
a61af66fc99e Initial load
duke
parents:
diff changeset
342 size_t CompactibleFreeListSpace::free() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
343 // "MT-safe, but not MT-precise"(TM), if you will: i.e.
a61af66fc99e Initial load
duke
parents:
diff changeset
344 // if you do this while the structures are in flux you
a61af66fc99e Initial load
duke
parents:
diff changeset
345 // may get an approximate answer only; for instance
a61af66fc99e Initial load
duke
parents:
diff changeset
346 // because there is concurrent allocation either
a61af66fc99e Initial load
duke
parents:
diff changeset
347 // directly by mutators or for promotion during a GC.
a61af66fc99e Initial load
duke
parents:
diff changeset
348 // It's "MT-safe", however, in the sense that you are guaranteed
a61af66fc99e Initial load
duke
parents:
diff changeset
349 // not to crash and burn, for instance, because of walking
a61af66fc99e Initial load
duke
parents:
diff changeset
350 // pointers that could disappear as you were walking them.
a61af66fc99e Initial load
duke
parents:
diff changeset
351 // The approximation is because the various components
a61af66fc99e Initial load
duke
parents:
diff changeset
352 // that are read below are not read atomically (and
a61af66fc99e Initial load
duke
parents:
diff changeset
353 // further the computation of totalSizeInIndexedFreeLists()
a61af66fc99e Initial load
duke
parents:
diff changeset
354 // is itself a non-atomic computation. The normal use of
a61af66fc99e Initial load
duke
parents:
diff changeset
355 // this is during a resize operation at the end of GC
a61af66fc99e Initial load
duke
parents:
diff changeset
356 // and at that time you are guaranteed to get the
a61af66fc99e Initial load
duke
parents:
diff changeset
357 // correct actual value. However, for instance, this is
a61af66fc99e Initial load
duke
parents:
diff changeset
358 // also read completely asynchronously by the "perf-sampler"
a61af66fc99e Initial load
duke
parents:
diff changeset
359 // that supports jvmstat, and you are apt to see the values
a61af66fc99e Initial load
duke
parents:
diff changeset
360 // flicker in such cases.
a61af66fc99e Initial load
duke
parents:
diff changeset
361 assert(_dictionary != NULL, "No _dictionary?");
a61af66fc99e Initial load
duke
parents:
diff changeset
362 return (_dictionary->totalChunkSize(DEBUG_ONLY(freelistLock())) +
a61af66fc99e Initial load
duke
parents:
diff changeset
363 totalSizeInIndexedFreeLists() +
a61af66fc99e Initial load
duke
parents:
diff changeset
364 _smallLinearAllocBlock._word_size) * HeapWordSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
365 }
a61af66fc99e Initial load
duke
parents:
diff changeset
366
a61af66fc99e Initial load
duke
parents:
diff changeset
367 size_t CompactibleFreeListSpace::max_alloc_in_words() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
368 assert(_dictionary != NULL, "No _dictionary?");
a61af66fc99e Initial load
duke
parents:
diff changeset
369 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
370 size_t res = _dictionary->maxChunkSize();
a61af66fc99e Initial load
duke
parents:
diff changeset
371 res = MAX2(res, MIN2(_smallLinearAllocBlock._word_size,
a61af66fc99e Initial load
duke
parents:
diff changeset
372 (size_t) SmallForLinearAlloc - 1));
a61af66fc99e Initial load
duke
parents:
diff changeset
373 // XXX the following could potentially be pretty slow;
a61af66fc99e Initial load
duke
parents:
diff changeset
374 // should one, pesimally for the rare cases when res
a61af66fc99e Initial load
duke
parents:
diff changeset
375 // caclulated above is less than IndexSetSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
376 // just return res calculated above? My reasoning was that
a61af66fc99e Initial load
duke
parents:
diff changeset
377 // those cases will be so rare that the extra time spent doesn't
a61af66fc99e Initial load
duke
parents:
diff changeset
378 // really matter....
a61af66fc99e Initial load
duke
parents:
diff changeset
379 // Note: do not change the loop test i >= res + IndexSetStride
a61af66fc99e Initial load
duke
parents:
diff changeset
380 // to i > res below, because i is unsigned and res may be zero.
a61af66fc99e Initial load
duke
parents:
diff changeset
381 for (size_t i = IndexSetSize - 1; i >= res + IndexSetStride;
a61af66fc99e Initial load
duke
parents:
diff changeset
382 i -= IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
383 if (_indexedFreeList[i].head() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
384 assert(_indexedFreeList[i].count() != 0, "Inconsistent FreeList");
a61af66fc99e Initial load
duke
parents:
diff changeset
385 return i;
a61af66fc99e Initial load
duke
parents:
diff changeset
386 }
a61af66fc99e Initial load
duke
parents:
diff changeset
387 }
a61af66fc99e Initial load
duke
parents:
diff changeset
388 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
389 }
a61af66fc99e Initial load
duke
parents:
diff changeset
390
a61af66fc99e Initial load
duke
parents:
diff changeset
391 void CompactibleFreeListSpace::reportFreeListStatistics() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
392 assert_lock_strong(&_freelistLock);
a61af66fc99e Initial load
duke
parents:
diff changeset
393 assert(PrintFLSStatistics != 0, "Reporting error");
a61af66fc99e Initial load
duke
parents:
diff changeset
394 _dictionary->reportStatistics();
a61af66fc99e Initial load
duke
parents:
diff changeset
395 if (PrintFLSStatistics > 1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
396 reportIndexedFreeListStatistics();
a61af66fc99e Initial load
duke
parents:
diff changeset
397 size_t totalSize = totalSizeInIndexedFreeLists() +
a61af66fc99e Initial load
duke
parents:
diff changeset
398 _dictionary->totalChunkSize(DEBUG_ONLY(freelistLock()));
a61af66fc99e Initial load
duke
parents:
diff changeset
399 gclog_or_tty->print(" free=%ld frag=%1.4f\n", totalSize, flsFrag());
a61af66fc99e Initial load
duke
parents:
diff changeset
400 }
a61af66fc99e Initial load
duke
parents:
diff changeset
401 }
a61af66fc99e Initial load
duke
parents:
diff changeset
402
a61af66fc99e Initial load
duke
parents:
diff changeset
403 void CompactibleFreeListSpace::reportIndexedFreeListStatistics() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
404 assert_lock_strong(&_freelistLock);
a61af66fc99e Initial load
duke
parents:
diff changeset
405 gclog_or_tty->print("Statistics for IndexedFreeLists:\n"
a61af66fc99e Initial load
duke
parents:
diff changeset
406 "--------------------------------\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
407 size_t totalSize = totalSizeInIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
408 size_t freeBlocks = numFreeBlocksInIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
409 gclog_or_tty->print("Total Free Space: %d\n", totalSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
410 gclog_or_tty->print("Max Chunk Size: %d\n", maxChunkSizeInIndexedFreeLists());
a61af66fc99e Initial load
duke
parents:
diff changeset
411 gclog_or_tty->print("Number of Blocks: %d\n", freeBlocks);
a61af66fc99e Initial load
duke
parents:
diff changeset
412 if (freeBlocks != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
413 gclog_or_tty->print("Av. Block Size: %d\n", totalSize/freeBlocks);
a61af66fc99e Initial load
duke
parents:
diff changeset
414 }
a61af66fc99e Initial load
duke
parents:
diff changeset
415 }
a61af66fc99e Initial load
duke
parents:
diff changeset
416
a61af66fc99e Initial load
duke
parents:
diff changeset
417 size_t CompactibleFreeListSpace::numFreeBlocksInIndexedFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
418 size_t res = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
419 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
420 debug_only(
a61af66fc99e Initial load
duke
parents:
diff changeset
421 ssize_t recount = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
422 for (FreeChunk* fc = _indexedFreeList[i].head(); fc != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
423 fc = fc->next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
424 recount += 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
425 }
a61af66fc99e Initial load
duke
parents:
diff changeset
426 assert(recount == _indexedFreeList[i].count(),
a61af66fc99e Initial load
duke
parents:
diff changeset
427 "Incorrect count in list");
a61af66fc99e Initial load
duke
parents:
diff changeset
428 )
a61af66fc99e Initial load
duke
parents:
diff changeset
429 res += _indexedFreeList[i].count();
a61af66fc99e Initial load
duke
parents:
diff changeset
430 }
a61af66fc99e Initial load
duke
parents:
diff changeset
431 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
432 }
a61af66fc99e Initial load
duke
parents:
diff changeset
433
a61af66fc99e Initial load
duke
parents:
diff changeset
434 size_t CompactibleFreeListSpace::maxChunkSizeInIndexedFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
435 for (size_t i = IndexSetSize - 1; i != 0; i -= IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
436 if (_indexedFreeList[i].head() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
437 assert(_indexedFreeList[i].count() != 0, "Inconsistent FreeList");
a61af66fc99e Initial load
duke
parents:
diff changeset
438 return (size_t)i;
a61af66fc99e Initial load
duke
parents:
diff changeset
439 }
a61af66fc99e Initial load
duke
parents:
diff changeset
440 }
a61af66fc99e Initial load
duke
parents:
diff changeset
441 return 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
442 }
a61af66fc99e Initial load
duke
parents:
diff changeset
443
a61af66fc99e Initial load
duke
parents:
diff changeset
444 void CompactibleFreeListSpace::set_end(HeapWord* value) {
a61af66fc99e Initial load
duke
parents:
diff changeset
445 HeapWord* prevEnd = end();
a61af66fc99e Initial load
duke
parents:
diff changeset
446 assert(prevEnd != value, "unnecessary set_end call");
a61af66fc99e Initial load
duke
parents:
diff changeset
447 assert(prevEnd == NULL || value >= unallocated_block(), "New end is below unallocated block");
a61af66fc99e Initial load
duke
parents:
diff changeset
448 _end = value;
a61af66fc99e Initial load
duke
parents:
diff changeset
449 if (prevEnd != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
450 // Resize the underlying block offset table.
a61af66fc99e Initial load
duke
parents:
diff changeset
451 _bt.resize(pointer_delta(value, bottom()));
a61af66fc99e Initial load
duke
parents:
diff changeset
452 if (value <= prevEnd) {
a61af66fc99e Initial load
duke
parents:
diff changeset
453 assert(value >= unallocated_block(), "New end is below unallocated block");
a61af66fc99e Initial load
duke
parents:
diff changeset
454 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
455 // Now, take this new chunk and add it to the free blocks.
a61af66fc99e Initial load
duke
parents:
diff changeset
456 // Note that the BOT has not yet been updated for this block.
a61af66fc99e Initial load
duke
parents:
diff changeset
457 size_t newFcSize = pointer_delta(value, prevEnd);
a61af66fc99e Initial load
duke
parents:
diff changeset
458 // XXX This is REALLY UGLY and should be fixed up. XXX
a61af66fc99e Initial load
duke
parents:
diff changeset
459 if (!_adaptive_freelists && _smallLinearAllocBlock._ptr == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
460 // Mark the boundary of the new block in BOT
a61af66fc99e Initial load
duke
parents:
diff changeset
461 _bt.mark_block(prevEnd, value);
a61af66fc99e Initial load
duke
parents:
diff changeset
462 // put it all in the linAB
a61af66fc99e Initial load
duke
parents:
diff changeset
463 if (ParallelGCThreads == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
464 _smallLinearAllocBlock._ptr = prevEnd;
a61af66fc99e Initial load
duke
parents:
diff changeset
465 _smallLinearAllocBlock._word_size = newFcSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
466 repairLinearAllocBlock(&_smallLinearAllocBlock);
a61af66fc99e Initial load
duke
parents:
diff changeset
467 } else { // ParallelGCThreads > 0
a61af66fc99e Initial load
duke
parents:
diff changeset
468 MutexLockerEx x(parDictionaryAllocLock(),
a61af66fc99e Initial load
duke
parents:
diff changeset
469 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
470 _smallLinearAllocBlock._ptr = prevEnd;
a61af66fc99e Initial load
duke
parents:
diff changeset
471 _smallLinearAllocBlock._word_size = newFcSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
472 repairLinearAllocBlock(&_smallLinearAllocBlock);
a61af66fc99e Initial load
duke
parents:
diff changeset
473 }
a61af66fc99e Initial load
duke
parents:
diff changeset
474 // Births of chunks put into a LinAB are not recorded. Births
a61af66fc99e Initial load
duke
parents:
diff changeset
475 // of chunks as they are allocated out of a LinAB are.
a61af66fc99e Initial load
duke
parents:
diff changeset
476 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
477 // Add the block to the free lists, if possible coalescing it
a61af66fc99e Initial load
duke
parents:
diff changeset
478 // with the last free block, and update the BOT and census data.
a61af66fc99e Initial load
duke
parents:
diff changeset
479 addChunkToFreeListsAtEndRecordingStats(prevEnd, newFcSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
480 }
a61af66fc99e Initial load
duke
parents:
diff changeset
481 }
a61af66fc99e Initial load
duke
parents:
diff changeset
482 }
a61af66fc99e Initial load
duke
parents:
diff changeset
483 }
a61af66fc99e Initial load
duke
parents:
diff changeset
484
a61af66fc99e Initial load
duke
parents:
diff changeset
485 class FreeListSpace_DCTOC : public Filtering_DCTOC {
a61af66fc99e Initial load
duke
parents:
diff changeset
486 CompactibleFreeListSpace* _cfls;
a61af66fc99e Initial load
duke
parents:
diff changeset
487 CMSCollector* _collector;
a61af66fc99e Initial load
duke
parents:
diff changeset
488 protected:
a61af66fc99e Initial load
duke
parents:
diff changeset
489 // Override.
a61af66fc99e Initial load
duke
parents:
diff changeset
490 #define walk_mem_region_with_cl_DECL(ClosureType) \
a61af66fc99e Initial load
duke
parents:
diff changeset
491 virtual void walk_mem_region_with_cl(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
492 HeapWord* bottom, HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
493 ClosureType* cl); \
a61af66fc99e Initial load
duke
parents:
diff changeset
494 void walk_mem_region_with_cl_par(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
495 HeapWord* bottom, HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
496 ClosureType* cl); \
a61af66fc99e Initial load
duke
parents:
diff changeset
497 void walk_mem_region_with_cl_nopar(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
498 HeapWord* bottom, HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
499 ClosureType* cl)
a61af66fc99e Initial load
duke
parents:
diff changeset
500 walk_mem_region_with_cl_DECL(OopClosure);
a61af66fc99e Initial load
duke
parents:
diff changeset
501 walk_mem_region_with_cl_DECL(FilteringClosure);
a61af66fc99e Initial load
duke
parents:
diff changeset
502
a61af66fc99e Initial load
duke
parents:
diff changeset
503 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
504 FreeListSpace_DCTOC(CompactibleFreeListSpace* sp,
a61af66fc99e Initial load
duke
parents:
diff changeset
505 CMSCollector* collector,
a61af66fc99e Initial load
duke
parents:
diff changeset
506 OopClosure* cl,
a61af66fc99e Initial load
duke
parents:
diff changeset
507 CardTableModRefBS::PrecisionStyle precision,
a61af66fc99e Initial load
duke
parents:
diff changeset
508 HeapWord* boundary) :
a61af66fc99e Initial load
duke
parents:
diff changeset
509 Filtering_DCTOC(sp, cl, precision, boundary),
a61af66fc99e Initial load
duke
parents:
diff changeset
510 _cfls(sp), _collector(collector) {}
a61af66fc99e Initial load
duke
parents:
diff changeset
511 };
a61af66fc99e Initial load
duke
parents:
diff changeset
512
a61af66fc99e Initial load
duke
parents:
diff changeset
513 // We de-virtualize the block-related calls below, since we know that our
a61af66fc99e Initial load
duke
parents:
diff changeset
514 // space is a CompactibleFreeListSpace.
a61af66fc99e Initial load
duke
parents:
diff changeset
515 #define FreeListSpace_DCTOC__walk_mem_region_with_cl_DEFN(ClosureType) \
a61af66fc99e Initial load
duke
parents:
diff changeset
516 void FreeListSpace_DCTOC::walk_mem_region_with_cl(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
517 HeapWord* bottom, \
a61af66fc99e Initial load
duke
parents:
diff changeset
518 HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
519 ClosureType* cl) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
520 if (SharedHeap::heap()->n_par_threads() > 0) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
521 walk_mem_region_with_cl_par(mr, bottom, top, cl); \
a61af66fc99e Initial load
duke
parents:
diff changeset
522 } else { \
a61af66fc99e Initial load
duke
parents:
diff changeset
523 walk_mem_region_with_cl_nopar(mr, bottom, top, cl); \
a61af66fc99e Initial load
duke
parents:
diff changeset
524 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
525 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
526 void FreeListSpace_DCTOC::walk_mem_region_with_cl_par(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
527 HeapWord* bottom, \
a61af66fc99e Initial load
duke
parents:
diff changeset
528 HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
529 ClosureType* cl) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
530 /* Skip parts that are before "mr", in case "block_start" sent us \
a61af66fc99e Initial load
duke
parents:
diff changeset
531 back too far. */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
532 HeapWord* mr_start = mr.start(); \
a61af66fc99e Initial load
duke
parents:
diff changeset
533 size_t bot_size = _cfls->CompactibleFreeListSpace::block_size(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
534 HeapWord* next = bottom + bot_size; \
a61af66fc99e Initial load
duke
parents:
diff changeset
535 while (next < mr_start) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
536 bottom = next; \
a61af66fc99e Initial load
duke
parents:
diff changeset
537 bot_size = _cfls->CompactibleFreeListSpace::block_size(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
538 next = bottom + bot_size; \
a61af66fc99e Initial load
duke
parents:
diff changeset
539 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
540 \
a61af66fc99e Initial load
duke
parents:
diff changeset
541 while (bottom < top) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
542 if (_cfls->CompactibleFreeListSpace::block_is_obj(bottom) && \
a61af66fc99e Initial load
duke
parents:
diff changeset
543 !_cfls->CompactibleFreeListSpace::obj_allocated_since_save_marks( \
a61af66fc99e Initial load
duke
parents:
diff changeset
544 oop(bottom)) && \
a61af66fc99e Initial load
duke
parents:
diff changeset
545 !_collector->CMSCollector::is_dead_obj(oop(bottom))) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
546 size_t word_sz = oop(bottom)->oop_iterate(cl, mr); \
a61af66fc99e Initial load
duke
parents:
diff changeset
547 bottom += _cfls->adjustObjectSize(word_sz); \
a61af66fc99e Initial load
duke
parents:
diff changeset
548 } else { \
a61af66fc99e Initial load
duke
parents:
diff changeset
549 bottom += _cfls->CompactibleFreeListSpace::block_size(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
550 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
551 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
552 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
553 void FreeListSpace_DCTOC::walk_mem_region_with_cl_nopar(MemRegion mr, \
a61af66fc99e Initial load
duke
parents:
diff changeset
554 HeapWord* bottom, \
a61af66fc99e Initial load
duke
parents:
diff changeset
555 HeapWord* top, \
a61af66fc99e Initial load
duke
parents:
diff changeset
556 ClosureType* cl) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
557 /* Skip parts that are before "mr", in case "block_start" sent us \
a61af66fc99e Initial load
duke
parents:
diff changeset
558 back too far. */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
559 HeapWord* mr_start = mr.start(); \
a61af66fc99e Initial load
duke
parents:
diff changeset
560 size_t bot_size = _cfls->CompactibleFreeListSpace::block_size_nopar(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
561 HeapWord* next = bottom + bot_size; \
a61af66fc99e Initial load
duke
parents:
diff changeset
562 while (next < mr_start) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
563 bottom = next; \
a61af66fc99e Initial load
duke
parents:
diff changeset
564 bot_size = _cfls->CompactibleFreeListSpace::block_size_nopar(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
565 next = bottom + bot_size; \
a61af66fc99e Initial load
duke
parents:
diff changeset
566 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
567 \
a61af66fc99e Initial load
duke
parents:
diff changeset
568 while (bottom < top) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
569 if (_cfls->CompactibleFreeListSpace::block_is_obj_nopar(bottom) && \
a61af66fc99e Initial load
duke
parents:
diff changeset
570 !_cfls->CompactibleFreeListSpace::obj_allocated_since_save_marks( \
a61af66fc99e Initial load
duke
parents:
diff changeset
571 oop(bottom)) && \
a61af66fc99e Initial load
duke
parents:
diff changeset
572 !_collector->CMSCollector::is_dead_obj(oop(bottom))) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
573 size_t word_sz = oop(bottom)->oop_iterate(cl, mr); \
a61af66fc99e Initial load
duke
parents:
diff changeset
574 bottom += _cfls->adjustObjectSize(word_sz); \
a61af66fc99e Initial load
duke
parents:
diff changeset
575 } else { \
a61af66fc99e Initial load
duke
parents:
diff changeset
576 bottom += _cfls->CompactibleFreeListSpace::block_size_nopar(bottom); \
a61af66fc99e Initial load
duke
parents:
diff changeset
577 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
578 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
579 }
a61af66fc99e Initial load
duke
parents:
diff changeset
580
a61af66fc99e Initial load
duke
parents:
diff changeset
581 // (There are only two of these, rather than N, because the split is due
a61af66fc99e Initial load
duke
parents:
diff changeset
582 // only to the introduction of the FilteringClosure, a local part of the
a61af66fc99e Initial load
duke
parents:
diff changeset
583 // impl of this abstraction.)
a61af66fc99e Initial load
duke
parents:
diff changeset
584 FreeListSpace_DCTOC__walk_mem_region_with_cl_DEFN(OopClosure)
a61af66fc99e Initial load
duke
parents:
diff changeset
585 FreeListSpace_DCTOC__walk_mem_region_with_cl_DEFN(FilteringClosure)
a61af66fc99e Initial load
duke
parents:
diff changeset
586
a61af66fc99e Initial load
duke
parents:
diff changeset
587 DirtyCardToOopClosure*
a61af66fc99e Initial load
duke
parents:
diff changeset
588 CompactibleFreeListSpace::new_dcto_cl(OopClosure* cl,
a61af66fc99e Initial load
duke
parents:
diff changeset
589 CardTableModRefBS::PrecisionStyle precision,
a61af66fc99e Initial load
duke
parents:
diff changeset
590 HeapWord* boundary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
591 return new FreeListSpace_DCTOC(this, _collector, cl, precision, boundary);
a61af66fc99e Initial load
duke
parents:
diff changeset
592 }
a61af66fc99e Initial load
duke
parents:
diff changeset
593
a61af66fc99e Initial load
duke
parents:
diff changeset
594
a61af66fc99e Initial load
duke
parents:
diff changeset
595 // Note on locking for the space iteration functions:
a61af66fc99e Initial load
duke
parents:
diff changeset
596 // since the collector's iteration activities are concurrent with
a61af66fc99e Initial load
duke
parents:
diff changeset
597 // allocation activities by mutators, absent a suitable mutual exclusion
a61af66fc99e Initial load
duke
parents:
diff changeset
598 // mechanism the iterators may go awry. For instace a block being iterated
a61af66fc99e Initial load
duke
parents:
diff changeset
599 // may suddenly be allocated or divided up and part of it allocated and
a61af66fc99e Initial load
duke
parents:
diff changeset
600 // so on.
a61af66fc99e Initial load
duke
parents:
diff changeset
601
a61af66fc99e Initial load
duke
parents:
diff changeset
602 // Apply the given closure to each block in the space.
a61af66fc99e Initial load
duke
parents:
diff changeset
603 void CompactibleFreeListSpace::blk_iterate_careful(BlkClosureCareful* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
604 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
605 HeapWord *cur, *limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
606 for (cur = bottom(), limit = end(); cur < limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
607 cur += cl->do_blk_careful(cur));
a61af66fc99e Initial load
duke
parents:
diff changeset
608 }
a61af66fc99e Initial load
duke
parents:
diff changeset
609
a61af66fc99e Initial load
duke
parents:
diff changeset
610 // Apply the given closure to each block in the space.
a61af66fc99e Initial load
duke
parents:
diff changeset
611 void CompactibleFreeListSpace::blk_iterate(BlkClosure* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
612 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
613 HeapWord *cur, *limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
614 for (cur = bottom(), limit = end(); cur < limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
615 cur += cl->do_blk(cur));
a61af66fc99e Initial load
duke
parents:
diff changeset
616 }
a61af66fc99e Initial load
duke
parents:
diff changeset
617
a61af66fc99e Initial load
duke
parents:
diff changeset
618 // Apply the given closure to each oop in the space.
a61af66fc99e Initial load
duke
parents:
diff changeset
619 void CompactibleFreeListSpace::oop_iterate(OopClosure* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
620 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
621 HeapWord *cur, *limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
622 size_t curSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
623 for (cur = bottom(), limit = end(); cur < limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
624 cur += curSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
625 curSize = block_size(cur);
a61af66fc99e Initial load
duke
parents:
diff changeset
626 if (block_is_obj(cur)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
627 oop(cur)->oop_iterate(cl);
a61af66fc99e Initial load
duke
parents:
diff changeset
628 }
a61af66fc99e Initial load
duke
parents:
diff changeset
629 }
a61af66fc99e Initial load
duke
parents:
diff changeset
630 }
a61af66fc99e Initial load
duke
parents:
diff changeset
631
a61af66fc99e Initial load
duke
parents:
diff changeset
632 // Apply the given closure to each oop in the space \intersect memory region.
a61af66fc99e Initial load
duke
parents:
diff changeset
633 void CompactibleFreeListSpace::oop_iterate(MemRegion mr, OopClosure* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
634 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
635 if (is_empty()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
636 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
637 }
a61af66fc99e Initial load
duke
parents:
diff changeset
638 MemRegion cur = MemRegion(bottom(), end());
a61af66fc99e Initial load
duke
parents:
diff changeset
639 mr = mr.intersection(cur);
a61af66fc99e Initial load
duke
parents:
diff changeset
640 if (mr.is_empty()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
641 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
642 }
a61af66fc99e Initial load
duke
parents:
diff changeset
643 if (mr.equals(cur)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
644 oop_iterate(cl);
a61af66fc99e Initial load
duke
parents:
diff changeset
645 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
646 }
a61af66fc99e Initial load
duke
parents:
diff changeset
647 assert(mr.end() <= end(), "just took an intersection above");
a61af66fc99e Initial load
duke
parents:
diff changeset
648 HeapWord* obj_addr = block_start(mr.start());
a61af66fc99e Initial load
duke
parents:
diff changeset
649 HeapWord* t = mr.end();
a61af66fc99e Initial load
duke
parents:
diff changeset
650
a61af66fc99e Initial load
duke
parents:
diff changeset
651 SpaceMemRegionOopsIterClosure smr_blk(cl, mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
652 if (block_is_obj(obj_addr)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
653 // Handle first object specially.
a61af66fc99e Initial load
duke
parents:
diff changeset
654 oop obj = oop(obj_addr);
a61af66fc99e Initial load
duke
parents:
diff changeset
655 obj_addr += adjustObjectSize(obj->oop_iterate(&smr_blk));
a61af66fc99e Initial load
duke
parents:
diff changeset
656 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
657 FreeChunk* fc = (FreeChunk*)obj_addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
658 obj_addr += fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
659 }
a61af66fc99e Initial load
duke
parents:
diff changeset
660 while (obj_addr < t) {
a61af66fc99e Initial load
duke
parents:
diff changeset
661 HeapWord* obj = obj_addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
662 obj_addr += block_size(obj_addr);
a61af66fc99e Initial load
duke
parents:
diff changeset
663 // If "obj_addr" is not greater than top, then the
a61af66fc99e Initial load
duke
parents:
diff changeset
664 // entire object "obj" is within the region.
a61af66fc99e Initial load
duke
parents:
diff changeset
665 if (obj_addr <= t) {
a61af66fc99e Initial load
duke
parents:
diff changeset
666 if (block_is_obj(obj)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
667 oop(obj)->oop_iterate(cl);
a61af66fc99e Initial load
duke
parents:
diff changeset
668 }
a61af66fc99e Initial load
duke
parents:
diff changeset
669 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
670 // "obj" extends beyond end of region
a61af66fc99e Initial load
duke
parents:
diff changeset
671 if (block_is_obj(obj)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
672 oop(obj)->oop_iterate(&smr_blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
673 }
a61af66fc99e Initial load
duke
parents:
diff changeset
674 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
675 }
a61af66fc99e Initial load
duke
parents:
diff changeset
676 }
a61af66fc99e Initial load
duke
parents:
diff changeset
677 }
a61af66fc99e Initial load
duke
parents:
diff changeset
678
a61af66fc99e Initial load
duke
parents:
diff changeset
679 // NOTE: In the following methods, in order to safely be able to
a61af66fc99e Initial load
duke
parents:
diff changeset
680 // apply the closure to an object, we need to be sure that the
a61af66fc99e Initial load
duke
parents:
diff changeset
681 // object has been initialized. We are guaranteed that an object
a61af66fc99e Initial load
duke
parents:
diff changeset
682 // is initialized if we are holding the Heap_lock with the
a61af66fc99e Initial load
duke
parents:
diff changeset
683 // world stopped.
a61af66fc99e Initial load
duke
parents:
diff changeset
684 void CompactibleFreeListSpace::verify_objects_initialized() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
685 if (is_init_completed()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
686 assert_locked_or_safepoint(Heap_lock);
a61af66fc99e Initial load
duke
parents:
diff changeset
687 if (Universe::is_fully_initialized()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
688 guarantee(SafepointSynchronize::is_at_safepoint(),
a61af66fc99e Initial load
duke
parents:
diff changeset
689 "Required for objects to be initialized");
a61af66fc99e Initial load
duke
parents:
diff changeset
690 }
a61af66fc99e Initial load
duke
parents:
diff changeset
691 } // else make a concession at vm start-up
a61af66fc99e Initial load
duke
parents:
diff changeset
692 }
a61af66fc99e Initial load
duke
parents:
diff changeset
693
a61af66fc99e Initial load
duke
parents:
diff changeset
694 // Apply the given closure to each object in the space
a61af66fc99e Initial load
duke
parents:
diff changeset
695 void CompactibleFreeListSpace::object_iterate(ObjectClosure* blk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
696 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
697 NOT_PRODUCT(verify_objects_initialized());
a61af66fc99e Initial load
duke
parents:
diff changeset
698 HeapWord *cur, *limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
699 size_t curSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
700 for (cur = bottom(), limit = end(); cur < limit;
a61af66fc99e Initial load
duke
parents:
diff changeset
701 cur += curSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
702 curSize = block_size(cur);
a61af66fc99e Initial load
duke
parents:
diff changeset
703 if (block_is_obj(cur)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
704 blk->do_object(oop(cur));
a61af66fc99e Initial load
duke
parents:
diff changeset
705 }
a61af66fc99e Initial load
duke
parents:
diff changeset
706 }
a61af66fc99e Initial load
duke
parents:
diff changeset
707 }
a61af66fc99e Initial load
duke
parents:
diff changeset
708
517
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
709 // Apply the given closure to each live object in the space
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
710 // The usage of CompactibleFreeListSpace
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
711 // by the ConcurrentMarkSweepGeneration for concurrent GC's allows
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
712 // objects in the space with references to objects that are no longer
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
713 // valid. For example, an object may reference another object
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
714 // that has already been sweep up (collected). This method uses
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
715 // obj_is_alive() to determine whether it is safe to apply the closure to
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
716 // an object. See obj_is_alive() for details on how liveness of an
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
717 // object is decided.
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
718
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
719 void CompactibleFreeListSpace::safe_object_iterate(ObjectClosure* blk) {
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
720 assert_lock_strong(freelistLock());
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
721 NOT_PRODUCT(verify_objects_initialized());
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
722 HeapWord *cur, *limit;
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
723 size_t curSize;
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
724 for (cur = bottom(), limit = end(); cur < limit;
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
725 cur += curSize) {
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
726 curSize = block_size(cur);
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
727 if (block_is_obj(cur) && obj_is_alive(cur)) {
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
728 blk->do_object(oop(cur));
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
729 }
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
730 }
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
731 }
e9be0e04635a 6689653: JMapPerm fails with UseConcMarkSweepIncGC and compressed oops off
jmasa
parents: 360
diff changeset
732
0
a61af66fc99e Initial load
duke
parents:
diff changeset
733 void CompactibleFreeListSpace::object_iterate_mem(MemRegion mr,
a61af66fc99e Initial load
duke
parents:
diff changeset
734 UpwardsObjectClosure* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
735 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
736 NOT_PRODUCT(verify_objects_initialized());
a61af66fc99e Initial load
duke
parents:
diff changeset
737 Space::object_iterate_mem(mr, cl);
a61af66fc99e Initial load
duke
parents:
diff changeset
738 }
a61af66fc99e Initial load
duke
parents:
diff changeset
739
a61af66fc99e Initial load
duke
parents:
diff changeset
740 // Callers of this iterator beware: The closure application should
a61af66fc99e Initial load
duke
parents:
diff changeset
741 // be robust in the face of uninitialized objects and should (always)
a61af66fc99e Initial load
duke
parents:
diff changeset
742 // return a correct size so that the next addr + size below gives us a
a61af66fc99e Initial load
duke
parents:
diff changeset
743 // valid block boundary. [See for instance,
a61af66fc99e Initial load
duke
parents:
diff changeset
744 // ScanMarkedObjectsAgainCarefullyClosure::do_object_careful()
a61af66fc99e Initial load
duke
parents:
diff changeset
745 // in ConcurrentMarkSweepGeneration.cpp.]
a61af66fc99e Initial load
duke
parents:
diff changeset
746 HeapWord*
a61af66fc99e Initial load
duke
parents:
diff changeset
747 CompactibleFreeListSpace::object_iterate_careful(ObjectClosureCareful* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
748 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
749 HeapWord *addr, *last;
a61af66fc99e Initial load
duke
parents:
diff changeset
750 size_t size;
a61af66fc99e Initial load
duke
parents:
diff changeset
751 for (addr = bottom(), last = end();
a61af66fc99e Initial load
duke
parents:
diff changeset
752 addr < last; addr += size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
753 FreeChunk* fc = (FreeChunk*)addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
754 if (fc->isFree()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
755 // Since we hold the free list lock, which protects direct
a61af66fc99e Initial load
duke
parents:
diff changeset
756 // allocation in this generation by mutators, a free object
a61af66fc99e Initial load
duke
parents:
diff changeset
757 // will remain free throughout this iteration code.
a61af66fc99e Initial load
duke
parents:
diff changeset
758 size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
759 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
760 // Note that the object need not necessarily be initialized,
a61af66fc99e Initial load
duke
parents:
diff changeset
761 // because (for instance) the free list lock does NOT protect
a61af66fc99e Initial load
duke
parents:
diff changeset
762 // object initialization. The closure application below must
a61af66fc99e Initial load
duke
parents:
diff changeset
763 // therefore be correct in the face of uninitialized objects.
a61af66fc99e Initial load
duke
parents:
diff changeset
764 size = cl->do_object_careful(oop(addr));
a61af66fc99e Initial load
duke
parents:
diff changeset
765 if (size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
766 // An unparsable object found. Signal early termination.
a61af66fc99e Initial load
duke
parents:
diff changeset
767 return addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
768 }
a61af66fc99e Initial load
duke
parents:
diff changeset
769 }
a61af66fc99e Initial load
duke
parents:
diff changeset
770 }
a61af66fc99e Initial load
duke
parents:
diff changeset
771 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
772 }
a61af66fc99e Initial load
duke
parents:
diff changeset
773
a61af66fc99e Initial load
duke
parents:
diff changeset
774 // Callers of this iterator beware: The closure application should
a61af66fc99e Initial load
duke
parents:
diff changeset
775 // be robust in the face of uninitialized objects and should (always)
a61af66fc99e Initial load
duke
parents:
diff changeset
776 // return a correct size so that the next addr + size below gives us a
a61af66fc99e Initial load
duke
parents:
diff changeset
777 // valid block boundary. [See for instance,
a61af66fc99e Initial load
duke
parents:
diff changeset
778 // ScanMarkedObjectsAgainCarefullyClosure::do_object_careful()
a61af66fc99e Initial load
duke
parents:
diff changeset
779 // in ConcurrentMarkSweepGeneration.cpp.]
a61af66fc99e Initial load
duke
parents:
diff changeset
780 HeapWord*
a61af66fc99e Initial load
duke
parents:
diff changeset
781 CompactibleFreeListSpace::object_iterate_careful_m(MemRegion mr,
a61af66fc99e Initial load
duke
parents:
diff changeset
782 ObjectClosureCareful* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
783 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
784 // Can't use used_region() below because it may not necessarily
a61af66fc99e Initial load
duke
parents:
diff changeset
785 // be the same as [bottom(),end()); although we could
a61af66fc99e Initial load
duke
parents:
diff changeset
786 // use [used_region().start(),round_to(used_region().end(),CardSize)),
a61af66fc99e Initial load
duke
parents:
diff changeset
787 // that appears too cumbersome, so we just do the simpler check
a61af66fc99e Initial load
duke
parents:
diff changeset
788 // in the assertion below.
a61af66fc99e Initial load
duke
parents:
diff changeset
789 assert(!mr.is_empty() && MemRegion(bottom(),end()).contains(mr),
a61af66fc99e Initial load
duke
parents:
diff changeset
790 "mr should be non-empty and within used space");
a61af66fc99e Initial load
duke
parents:
diff changeset
791 HeapWord *addr, *end;
a61af66fc99e Initial load
duke
parents:
diff changeset
792 size_t size;
a61af66fc99e Initial load
duke
parents:
diff changeset
793 for (addr = block_start_careful(mr.start()), end = mr.end();
a61af66fc99e Initial load
duke
parents:
diff changeset
794 addr < end; addr += size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
795 FreeChunk* fc = (FreeChunk*)addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
796 if (fc->isFree()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
797 // Since we hold the free list lock, which protects direct
a61af66fc99e Initial load
duke
parents:
diff changeset
798 // allocation in this generation by mutators, a free object
a61af66fc99e Initial load
duke
parents:
diff changeset
799 // will remain free throughout this iteration code.
a61af66fc99e Initial load
duke
parents:
diff changeset
800 size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
801 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
802 // Note that the object need not necessarily be initialized,
a61af66fc99e Initial load
duke
parents:
diff changeset
803 // because (for instance) the free list lock does NOT protect
a61af66fc99e Initial load
duke
parents:
diff changeset
804 // object initialization. The closure application below must
a61af66fc99e Initial load
duke
parents:
diff changeset
805 // therefore be correct in the face of uninitialized objects.
a61af66fc99e Initial load
duke
parents:
diff changeset
806 size = cl->do_object_careful_m(oop(addr), mr);
a61af66fc99e Initial load
duke
parents:
diff changeset
807 if (size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
808 // An unparsable object found. Signal early termination.
a61af66fc99e Initial load
duke
parents:
diff changeset
809 return addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
810 }
a61af66fc99e Initial load
duke
parents:
diff changeset
811 }
a61af66fc99e Initial load
duke
parents:
diff changeset
812 }
a61af66fc99e Initial load
duke
parents:
diff changeset
813 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
814 }
a61af66fc99e Initial load
duke
parents:
diff changeset
815
a61af66fc99e Initial load
duke
parents:
diff changeset
816
342
37f87013dfd8 6711316: Open source the Garbage-First garbage collector
ysr
parents: 113
diff changeset
817 HeapWord* CompactibleFreeListSpace::block_start_const(const void* p) const {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
818 NOT_PRODUCT(verify_objects_initialized());
a61af66fc99e Initial load
duke
parents:
diff changeset
819 return _bt.block_start(p);
a61af66fc99e Initial load
duke
parents:
diff changeset
820 }
a61af66fc99e Initial load
duke
parents:
diff changeset
821
a61af66fc99e Initial load
duke
parents:
diff changeset
822 HeapWord* CompactibleFreeListSpace::block_start_careful(const void* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
823 return _bt.block_start_careful(p);
a61af66fc99e Initial load
duke
parents:
diff changeset
824 }
a61af66fc99e Initial load
duke
parents:
diff changeset
825
a61af66fc99e Initial load
duke
parents:
diff changeset
826 size_t CompactibleFreeListSpace::block_size(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
827 NOT_PRODUCT(verify_objects_initialized());
a61af66fc99e Initial load
duke
parents:
diff changeset
828 assert(MemRegion(bottom(), end()).contains(p), "p not in space");
a61af66fc99e Initial load
duke
parents:
diff changeset
829 // This must be volatile, or else there is a danger that the compiler
a61af66fc99e Initial load
duke
parents:
diff changeset
830 // will compile the code below into a sometimes-infinite loop, by keeping
a61af66fc99e Initial load
duke
parents:
diff changeset
831 // the value read the first time in a register.
a61af66fc99e Initial load
duke
parents:
diff changeset
832 while (true) {
a61af66fc99e Initial load
duke
parents:
diff changeset
833 // We must do this until we get a consistent view of the object.
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
834 if (FreeChunk::indicatesFreeChunk(p)) {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
835 volatile FreeChunk* fc = (volatile FreeChunk*)p;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
836 size_t res = fc->size();
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
837 // If the object is still a free chunk, return the size, else it
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
838 // has been allocated so try again.
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
839 if (FreeChunk::indicatesFreeChunk(p)) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
840 assert(res != 0, "Block size should not be 0");
a61af66fc99e Initial load
duke
parents:
diff changeset
841 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
842 }
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
843 } else {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
844 // must read from what 'p' points to in each loop.
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
845 klassOop k = ((volatile oopDesc*)p)->klass_or_null();
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
846 if (k != NULL) {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
847 assert(k->is_oop(true /* ignore mark word */), "Should really be klass oop.");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
848 oop o = (oop)p;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
849 assert(o->is_parsable(), "Should be parsable");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
850 assert(o->is_oop(true /* ignore mark word */), "Should be an oop.");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
851 size_t res = o->size_given_klass(k->klass_part());
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
852 res = adjustObjectSize(res);
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
853 assert(res != 0, "Block size should not be 0");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
854 return res;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
855 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
856 }
a61af66fc99e Initial load
duke
parents:
diff changeset
857 }
a61af66fc99e Initial load
duke
parents:
diff changeset
858 }
a61af66fc99e Initial load
duke
parents:
diff changeset
859
a61af66fc99e Initial load
duke
parents:
diff changeset
860 // A variant of the above that uses the Printezis bits for
a61af66fc99e Initial load
duke
parents:
diff changeset
861 // unparsable but allocated objects. This avoids any possible
a61af66fc99e Initial load
duke
parents:
diff changeset
862 // stalls waiting for mutators to initialize objects, and is
a61af66fc99e Initial load
duke
parents:
diff changeset
863 // thus potentially faster than the variant above. However,
a61af66fc99e Initial load
duke
parents:
diff changeset
864 // this variant may return a zero size for a block that is
a61af66fc99e Initial load
duke
parents:
diff changeset
865 // under mutation and for which a consistent size cannot be
a61af66fc99e Initial load
duke
parents:
diff changeset
866 // inferred without stalling; see CMSCollector::block_size_if_printezis_bits().
a61af66fc99e Initial load
duke
parents:
diff changeset
867 size_t CompactibleFreeListSpace::block_size_no_stall(HeapWord* p,
a61af66fc99e Initial load
duke
parents:
diff changeset
868 const CMSCollector* c)
a61af66fc99e Initial load
duke
parents:
diff changeset
869 const {
a61af66fc99e Initial load
duke
parents:
diff changeset
870 assert(MemRegion(bottom(), end()).contains(p), "p not in space");
a61af66fc99e Initial load
duke
parents:
diff changeset
871 // This must be volatile, or else there is a danger that the compiler
a61af66fc99e Initial load
duke
parents:
diff changeset
872 // will compile the code below into a sometimes-infinite loop, by keeping
a61af66fc99e Initial load
duke
parents:
diff changeset
873 // the value read the first time in a register.
a61af66fc99e Initial load
duke
parents:
diff changeset
874 DEBUG_ONLY(uint loops = 0;)
a61af66fc99e Initial load
duke
parents:
diff changeset
875 while (true) {
a61af66fc99e Initial load
duke
parents:
diff changeset
876 // We must do this until we get a consistent view of the object.
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
877 if (FreeChunk::indicatesFreeChunk(p)) {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
878 volatile FreeChunk* fc = (volatile FreeChunk*)p;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
879 size_t res = fc->size();
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
880 if (FreeChunk::indicatesFreeChunk(p)) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
881 assert(res != 0, "Block size should not be 0");
a61af66fc99e Initial load
duke
parents:
diff changeset
882 assert(loops == 0, "Should be 0");
a61af66fc99e Initial load
duke
parents:
diff changeset
883 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
884 }
a61af66fc99e Initial load
duke
parents:
diff changeset
885 } else {
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
886 // must read from what 'p' points to in each loop.
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
887 klassOop k = ((volatile oopDesc*)p)->klass_or_null();
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
888 if (k != NULL && ((oopDesc*)p)->is_parsable()) {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
889 assert(k->is_oop(), "Should really be klass oop.");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
890 oop o = (oop)p;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
891 assert(o->is_oop(), "Should be an oop");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
892 size_t res = o->size_given_klass(k->klass_part());
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
893 res = adjustObjectSize(res);
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
894 assert(res != 0, "Block size should not be 0");
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
895 return res;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
896 } else {
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
897 return c->block_size_if_printezis_bits(p);
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
898 }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
899 }
a61af66fc99e Initial load
duke
parents:
diff changeset
900 assert(loops == 0, "Can loop at most once");
a61af66fc99e Initial load
duke
parents:
diff changeset
901 DEBUG_ONLY(loops++;)
a61af66fc99e Initial load
duke
parents:
diff changeset
902 }
a61af66fc99e Initial load
duke
parents:
diff changeset
903 }
a61af66fc99e Initial load
duke
parents:
diff changeset
904
a61af66fc99e Initial load
duke
parents:
diff changeset
905 size_t CompactibleFreeListSpace::block_size_nopar(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
906 NOT_PRODUCT(verify_objects_initialized());
a61af66fc99e Initial load
duke
parents:
diff changeset
907 assert(MemRegion(bottom(), end()).contains(p), "p not in space");
a61af66fc99e Initial load
duke
parents:
diff changeset
908 FreeChunk* fc = (FreeChunk*)p;
a61af66fc99e Initial load
duke
parents:
diff changeset
909 if (fc->isFree()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
910 return fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
911 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
912 // Ignore mark word because this may be a recently promoted
a61af66fc99e Initial load
duke
parents:
diff changeset
913 // object whose mark word is used to chain together grey
a61af66fc99e Initial load
duke
parents:
diff changeset
914 // objects (the last one would have a null value).
a61af66fc99e Initial load
duke
parents:
diff changeset
915 assert(oop(p)->is_oop(true), "Should be an oop");
a61af66fc99e Initial load
duke
parents:
diff changeset
916 return adjustObjectSize(oop(p)->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
917 }
a61af66fc99e Initial load
duke
parents:
diff changeset
918 }
a61af66fc99e Initial load
duke
parents:
diff changeset
919
a61af66fc99e Initial load
duke
parents:
diff changeset
920 // This implementation assumes that the property of "being an object" is
a61af66fc99e Initial load
duke
parents:
diff changeset
921 // stable. But being a free chunk may not be (because of parallel
a61af66fc99e Initial load
duke
parents:
diff changeset
922 // promotion.)
a61af66fc99e Initial load
duke
parents:
diff changeset
923 bool CompactibleFreeListSpace::block_is_obj(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
924 FreeChunk* fc = (FreeChunk*)p;
a61af66fc99e Initial load
duke
parents:
diff changeset
925 assert(is_in_reserved(p), "Should be in space");
a61af66fc99e Initial load
duke
parents:
diff changeset
926 // When doing a mark-sweep-compact of the CMS generation, this
a61af66fc99e Initial load
duke
parents:
diff changeset
927 // assertion may fail because prepare_for_compaction() uses
a61af66fc99e Initial load
duke
parents:
diff changeset
928 // space that is garbage to maintain information on ranges of
a61af66fc99e Initial load
duke
parents:
diff changeset
929 // live objects so that these live ranges can be moved as a whole.
a61af66fc99e Initial load
duke
parents:
diff changeset
930 // Comment out this assertion until that problem can be solved
a61af66fc99e Initial load
duke
parents:
diff changeset
931 // (i.e., that the block start calculation may look at objects
a61af66fc99e Initial load
duke
parents:
diff changeset
932 // at address below "p" in finding the object that contains "p"
a61af66fc99e Initial load
duke
parents:
diff changeset
933 // and those objects (if garbage) may have been modified to hold
a61af66fc99e Initial load
duke
parents:
diff changeset
934 // live range information.
a61af66fc99e Initial load
duke
parents:
diff changeset
935 // assert(ParallelGCThreads > 0 || _bt.block_start(p) == p, "Should be a block boundary");
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
936 if (FreeChunk::indicatesFreeChunk(p)) return false;
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
937 klassOop k = oop(p)->klass_or_null();
0
a61af66fc99e Initial load
duke
parents:
diff changeset
938 if (k != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
939 // Ignore mark word because it may have been used to
a61af66fc99e Initial load
duke
parents:
diff changeset
940 // chain together promoted objects (the last one
a61af66fc99e Initial load
duke
parents:
diff changeset
941 // would have a null value).
a61af66fc99e Initial load
duke
parents:
diff changeset
942 assert(oop(p)->is_oop(true), "Should be an oop");
a61af66fc99e Initial load
duke
parents:
diff changeset
943 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
944 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
945 return false; // Was not an object at the start of collection.
a61af66fc99e Initial load
duke
parents:
diff changeset
946 }
a61af66fc99e Initial load
duke
parents:
diff changeset
947 }
a61af66fc99e Initial load
duke
parents:
diff changeset
948
a61af66fc99e Initial load
duke
parents:
diff changeset
949 // Check if the object is alive. This fact is checked either by consulting
a61af66fc99e Initial load
duke
parents:
diff changeset
950 // the main marking bitmap in the sweeping phase or, if it's a permanent
a61af66fc99e Initial load
duke
parents:
diff changeset
951 // generation and we're not in the sweeping phase, by checking the
a61af66fc99e Initial load
duke
parents:
diff changeset
952 // perm_gen_verify_bit_map where we store the "deadness" information if
a61af66fc99e Initial load
duke
parents:
diff changeset
953 // we did not sweep the perm gen in the most recent previous GC cycle.
a61af66fc99e Initial load
duke
parents:
diff changeset
954 bool CompactibleFreeListSpace::obj_is_alive(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
955 assert (block_is_obj(p), "The address should point to an object");
a61af66fc99e Initial load
duke
parents:
diff changeset
956
a61af66fc99e Initial load
duke
parents:
diff changeset
957 // If we're sweeping, we use object liveness information from the main bit map
a61af66fc99e Initial load
duke
parents:
diff changeset
958 // for both perm gen and old gen.
a61af66fc99e Initial load
duke
parents:
diff changeset
959 // We don't need to lock the bitmap (live_map or dead_map below), because
a61af66fc99e Initial load
duke
parents:
diff changeset
960 // EITHER we are in the middle of the sweeping phase, and the
a61af66fc99e Initial load
duke
parents:
diff changeset
961 // main marking bit map (live_map below) is locked,
a61af66fc99e Initial load
duke
parents:
diff changeset
962 // OR we're in other phases and perm_gen_verify_bit_map (dead_map below)
a61af66fc99e Initial load
duke
parents:
diff changeset
963 // is stable, because it's mutated only in the sweeping phase.
a61af66fc99e Initial load
duke
parents:
diff changeset
964 if (_collector->abstract_state() == CMSCollector::Sweeping) {
a61af66fc99e Initial load
duke
parents:
diff changeset
965 CMSBitMap* live_map = _collector->markBitMap();
a61af66fc99e Initial load
duke
parents:
diff changeset
966 return live_map->isMarked((HeapWord*) p);
a61af66fc99e Initial load
duke
parents:
diff changeset
967 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
968 // If we're not currently sweeping and we haven't swept the perm gen in
a61af66fc99e Initial load
duke
parents:
diff changeset
969 // the previous concurrent cycle then we may have dead but unswept objects
a61af66fc99e Initial load
duke
parents:
diff changeset
970 // in the perm gen. In this case, we use the "deadness" information
a61af66fc99e Initial load
duke
parents:
diff changeset
971 // that we had saved in perm_gen_verify_bit_map at the last sweep.
a61af66fc99e Initial load
duke
parents:
diff changeset
972 if (!CMSClassUnloadingEnabled && _collector->_permGen->reserved().contains(p)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
973 if (_collector->verifying()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
974 CMSBitMap* dead_map = _collector->perm_gen_verify_bit_map();
a61af66fc99e Initial load
duke
parents:
diff changeset
975 // Object is marked in the dead_map bitmap at the previous sweep
a61af66fc99e Initial load
duke
parents:
diff changeset
976 // when we know that it's dead; if the bitmap is not allocated then
a61af66fc99e Initial load
duke
parents:
diff changeset
977 // the object is alive.
a61af66fc99e Initial load
duke
parents:
diff changeset
978 return (dead_map->sizeInBits() == 0) // bit_map has been allocated
a61af66fc99e Initial load
duke
parents:
diff changeset
979 || !dead_map->par_isMarked((HeapWord*) p);
a61af66fc99e Initial load
duke
parents:
diff changeset
980 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
981 return false; // We can't say for sure if it's live, so we say that it's dead.
a61af66fc99e Initial load
duke
parents:
diff changeset
982 }
a61af66fc99e Initial load
duke
parents:
diff changeset
983 }
a61af66fc99e Initial load
duke
parents:
diff changeset
984 }
a61af66fc99e Initial load
duke
parents:
diff changeset
985 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
986 }
a61af66fc99e Initial load
duke
parents:
diff changeset
987
a61af66fc99e Initial load
duke
parents:
diff changeset
988 bool CompactibleFreeListSpace::block_is_obj_nopar(const HeapWord* p) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
989 FreeChunk* fc = (FreeChunk*)p;
a61af66fc99e Initial load
duke
parents:
diff changeset
990 assert(is_in_reserved(p), "Should be in space");
a61af66fc99e Initial load
duke
parents:
diff changeset
991 assert(_bt.block_start(p) == p, "Should be a block boundary");
a61af66fc99e Initial load
duke
parents:
diff changeset
992 if (!fc->isFree()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
993 // Ignore mark word because it may have been used to
a61af66fc99e Initial load
duke
parents:
diff changeset
994 // chain together promoted objects (the last one
a61af66fc99e Initial load
duke
parents:
diff changeset
995 // would have a null value).
a61af66fc99e Initial load
duke
parents:
diff changeset
996 assert(oop(p)->is_oop(true), "Should be an oop");
a61af66fc99e Initial load
duke
parents:
diff changeset
997 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
998 }
a61af66fc99e Initial load
duke
parents:
diff changeset
999 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
1000 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1001
a61af66fc99e Initial load
duke
parents:
diff changeset
1002 // "MT-safe but not guaranteed MT-precise" (TM); you may get an
a61af66fc99e Initial load
duke
parents:
diff changeset
1003 // approximate answer if you don't hold the freelistlock when you call this.
a61af66fc99e Initial load
duke
parents:
diff changeset
1004 size_t CompactibleFreeListSpace::totalSizeInIndexedFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1005 size_t size = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1006 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1007 debug_only(
a61af66fc99e Initial load
duke
parents:
diff changeset
1008 // We may be calling here without the lock in which case we
a61af66fc99e Initial load
duke
parents:
diff changeset
1009 // won't do this modest sanity check.
a61af66fc99e Initial load
duke
parents:
diff changeset
1010 if (freelistLock()->owned_by_self()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1011 size_t total_list_size = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1012 for (FreeChunk* fc = _indexedFreeList[i].head(); fc != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1013 fc = fc->next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1014 total_list_size += i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1015 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1016 assert(total_list_size == i * _indexedFreeList[i].count(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1017 "Count in list is incorrect");
a61af66fc99e Initial load
duke
parents:
diff changeset
1018 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1019 )
a61af66fc99e Initial load
duke
parents:
diff changeset
1020 size += i * _indexedFreeList[i].count();
a61af66fc99e Initial load
duke
parents:
diff changeset
1021 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1022 return size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1023 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1024
a61af66fc99e Initial load
duke
parents:
diff changeset
1025 HeapWord* CompactibleFreeListSpace::par_allocate(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1026 MutexLockerEx x(freelistLock(), Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
1027 return allocate(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1028 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1029
a61af66fc99e Initial load
duke
parents:
diff changeset
1030 HeapWord*
a61af66fc99e Initial load
duke
parents:
diff changeset
1031 CompactibleFreeListSpace::getChunkFromSmallLinearAllocBlockRemainder(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1032 return getChunkFromLinearAllocBlockRemainder(&_smallLinearAllocBlock, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1033 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1034
a61af66fc99e Initial load
duke
parents:
diff changeset
1035 HeapWord* CompactibleFreeListSpace::allocate(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1036 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
1037 HeapWord* res = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1038 assert(size == adjustObjectSize(size),
a61af66fc99e Initial load
duke
parents:
diff changeset
1039 "use adjustObjectSize() before calling into allocate()");
a61af66fc99e Initial load
duke
parents:
diff changeset
1040
a61af66fc99e Initial load
duke
parents:
diff changeset
1041 if (_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1042 res = allocate_adaptive_freelists(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1043 } else { // non-adaptive free lists
a61af66fc99e Initial load
duke
parents:
diff changeset
1044 res = allocate_non_adaptive_freelists(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1045 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1046
a61af66fc99e Initial load
duke
parents:
diff changeset
1047 if (res != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1048 // check that res does lie in this space!
a61af66fc99e Initial load
duke
parents:
diff changeset
1049 assert(is_in_reserved(res), "Not in this space!");
a61af66fc99e Initial load
duke
parents:
diff changeset
1050 assert(is_aligned((void*)res), "alignment check");
a61af66fc99e Initial load
duke
parents:
diff changeset
1051
a61af66fc99e Initial load
duke
parents:
diff changeset
1052 FreeChunk* fc = (FreeChunk*)res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1053 fc->markNotFree();
a61af66fc99e Initial load
duke
parents:
diff changeset
1054 assert(!fc->isFree(), "shouldn't be marked free");
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
1055 assert(oop(fc)->klass_or_null() == NULL, "should look uninitialized");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1056 // Verify that the block offset table shows this to
a61af66fc99e Initial load
duke
parents:
diff changeset
1057 // be a single block, but not one which is unallocated.
a61af66fc99e Initial load
duke
parents:
diff changeset
1058 _bt.verify_single_block(res, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1059 _bt.verify_not_unallocated(res, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1060 // mangle a just allocated object with a distinct pattern.
a61af66fc99e Initial load
duke
parents:
diff changeset
1061 debug_only(fc->mangleAllocated(size));
a61af66fc99e Initial load
duke
parents:
diff changeset
1062 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1063
a61af66fc99e Initial load
duke
parents:
diff changeset
1064 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1065 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1066
a61af66fc99e Initial load
duke
parents:
diff changeset
1067 HeapWord* CompactibleFreeListSpace::allocate_non_adaptive_freelists(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1068 HeapWord* res = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1069 // try and use linear allocation for smaller blocks
a61af66fc99e Initial load
duke
parents:
diff changeset
1070 if (size < _smallLinearAllocBlock._allocation_size_limit) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1071 // if successful, the following also adjusts block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
1072 res = getChunkFromSmallLinearAllocBlock(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1073 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1074 // Else triage to indexed lists for smaller sizes
a61af66fc99e Initial load
duke
parents:
diff changeset
1075 if (res == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1076 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1077 res = (HeapWord*) getChunkFromIndexedFreeList(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1078 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1079 // else get it from the big dictionary; if even this doesn't
a61af66fc99e Initial load
duke
parents:
diff changeset
1080 // work we are out of luck.
a61af66fc99e Initial load
duke
parents:
diff changeset
1081 res = (HeapWord*)getChunkFromDictionaryExact(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1082 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1083 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1084
a61af66fc99e Initial load
duke
parents:
diff changeset
1085 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1086 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1087
a61af66fc99e Initial load
duke
parents:
diff changeset
1088 HeapWord* CompactibleFreeListSpace::allocate_adaptive_freelists(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1089 assert_lock_strong(freelistLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
1090 HeapWord* res = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1091 assert(size == adjustObjectSize(size),
a61af66fc99e Initial load
duke
parents:
diff changeset
1092 "use adjustObjectSize() before calling into allocate()");
a61af66fc99e Initial load
duke
parents:
diff changeset
1093
a61af66fc99e Initial load
duke
parents:
diff changeset
1094 // Strategy
a61af66fc99e Initial load
duke
parents:
diff changeset
1095 // if small
a61af66fc99e Initial load
duke
parents:
diff changeset
1096 // exact size from small object indexed list if small
a61af66fc99e Initial load
duke
parents:
diff changeset
1097 // small or large linear allocation block (linAB) as appropriate
a61af66fc99e Initial load
duke
parents:
diff changeset
1098 // take from lists of greater sized chunks
a61af66fc99e Initial load
duke
parents:
diff changeset
1099 // else
a61af66fc99e Initial load
duke
parents:
diff changeset
1100 // dictionary
a61af66fc99e Initial load
duke
parents:
diff changeset
1101 // small or large linear allocation block if it has the space
a61af66fc99e Initial load
duke
parents:
diff changeset
1102 // Try allocating exact size from indexTable first
a61af66fc99e Initial load
duke
parents:
diff changeset
1103 if (size < IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1104 res = (HeapWord*) getChunkFromIndexedFreeList(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1105 if(res != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1106 assert(res != (HeapWord*)_indexedFreeList[size].head(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1107 "Not removed from free list");
a61af66fc99e Initial load
duke
parents:
diff changeset
1108 // no block offset table adjustment is necessary on blocks in
a61af66fc99e Initial load
duke
parents:
diff changeset
1109 // the indexed lists.
a61af66fc99e Initial load
duke
parents:
diff changeset
1110
a61af66fc99e Initial load
duke
parents:
diff changeset
1111 // Try allocating from the small LinAB
a61af66fc99e Initial load
duke
parents:
diff changeset
1112 } else if (size < _smallLinearAllocBlock._allocation_size_limit &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1113 (res = getChunkFromSmallLinearAllocBlock(size)) != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1114 // if successful, the above also adjusts block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
1115 // Note that this call will refill the LinAB to
a61af66fc99e Initial load
duke
parents:
diff changeset
1116 // satisfy the request. This is different that
a61af66fc99e Initial load
duke
parents:
diff changeset
1117 // evm.
a61af66fc99e Initial load
duke
parents:
diff changeset
1118 // Don't record chunk off a LinAB? smallSplitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1119
a61af66fc99e Initial load
duke
parents:
diff changeset
1120 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1121 // Raid the exact free lists larger than size, even if they are not
a61af66fc99e Initial load
duke
parents:
diff changeset
1122 // overpopulated.
a61af66fc99e Initial load
duke
parents:
diff changeset
1123 res = (HeapWord*) getChunkFromGreater(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1124 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1125 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1126 // Big objects get allocated directly from the dictionary.
a61af66fc99e Initial load
duke
parents:
diff changeset
1127 res = (HeapWord*) getChunkFromDictionaryExact(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1128 if (res == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1129 // Try hard not to fail since an allocation failure will likely
a61af66fc99e Initial load
duke
parents:
diff changeset
1130 // trigger a synchronous GC. Try to get the space from the
a61af66fc99e Initial load
duke
parents:
diff changeset
1131 // allocation blocks.
a61af66fc99e Initial load
duke
parents:
diff changeset
1132 res = getChunkFromSmallLinearAllocBlockRemainder(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1133 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1134 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1135
a61af66fc99e Initial load
duke
parents:
diff changeset
1136 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1137 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1138
a61af66fc99e Initial load
duke
parents:
diff changeset
1139 // A worst-case estimate of the space required (in HeapWords) to expand the heap
a61af66fc99e Initial load
duke
parents:
diff changeset
1140 // when promoting obj.
a61af66fc99e Initial load
duke
parents:
diff changeset
1141 size_t CompactibleFreeListSpace::expansionSpaceRequired(size_t obj_size) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1142 // Depending on the object size, expansion may require refilling either a
a61af66fc99e Initial load
duke
parents:
diff changeset
1143 // bigLAB or a smallLAB plus refilling a PromotionInfo object. MinChunkSize
a61af66fc99e Initial load
duke
parents:
diff changeset
1144 // is added because the dictionary may over-allocate to avoid fragmentation.
a61af66fc99e Initial load
duke
parents:
diff changeset
1145 size_t space = obj_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1146 if (!_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1147 space = MAX2(space, _smallLinearAllocBlock._refillSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1148 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1149 space += _promoInfo.refillSize() + 2 * MinChunkSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
1150 return space;
a61af66fc99e Initial load
duke
parents:
diff changeset
1151 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1152
a61af66fc99e Initial load
duke
parents:
diff changeset
1153 FreeChunk* CompactibleFreeListSpace::getChunkFromGreater(size_t numWords) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1154 FreeChunk* ret;
a61af66fc99e Initial load
duke
parents:
diff changeset
1155
a61af66fc99e Initial load
duke
parents:
diff changeset
1156 assert(numWords >= MinChunkSize, "Size is less than minimum");
a61af66fc99e Initial load
duke
parents:
diff changeset
1157 assert(linearAllocationWouldFail() || bestFitFirst(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1158 "Should not be here");
a61af66fc99e Initial load
duke
parents:
diff changeset
1159
a61af66fc99e Initial load
duke
parents:
diff changeset
1160 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1161 size_t currSize = numWords + MinChunkSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
1162 assert(currSize % MinObjAlignment == 0, "currSize should be aligned");
a61af66fc99e Initial load
duke
parents:
diff changeset
1163 for (i = currSize; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1164 FreeList* fl = &_indexedFreeList[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
1165 if (fl->head()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1166 ret = getFromListGreater(fl, numWords);
a61af66fc99e Initial load
duke
parents:
diff changeset
1167 assert(ret == NULL || ret->isFree(), "Should be returning a free chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1168 return ret;
a61af66fc99e Initial load
duke
parents:
diff changeset
1169 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1170 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1171
a61af66fc99e Initial load
duke
parents:
diff changeset
1172 currSize = MAX2((size_t)SmallForDictionary,
a61af66fc99e Initial load
duke
parents:
diff changeset
1173 (size_t)(numWords + MinChunkSize));
a61af66fc99e Initial load
duke
parents:
diff changeset
1174
a61af66fc99e Initial load
duke
parents:
diff changeset
1175 /* Try to get a chunk that satisfies request, while avoiding
a61af66fc99e Initial load
duke
parents:
diff changeset
1176 fragmentation that can't be handled. */
a61af66fc99e Initial load
duke
parents:
diff changeset
1177 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1178 ret = dictionary()->getChunk(currSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1179 if (ret != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1180 assert(ret->size() - numWords >= MinChunkSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
1181 "Chunk is too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1182 _bt.allocated((HeapWord*)ret, ret->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1183 /* Carve returned chunk. */
a61af66fc99e Initial load
duke
parents:
diff changeset
1184 (void) splitChunkAndReturnRemainder(ret, numWords);
a61af66fc99e Initial load
duke
parents:
diff changeset
1185 /* Label this as no longer a free chunk. */
a61af66fc99e Initial load
duke
parents:
diff changeset
1186 assert(ret->isFree(), "This chunk should be free");
a61af66fc99e Initial load
duke
parents:
diff changeset
1187 ret->linkPrev(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1188 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1189 assert(ret == NULL || ret->isFree(), "Should be returning a free chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1190 return ret;
a61af66fc99e Initial load
duke
parents:
diff changeset
1191 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1192 ShouldNotReachHere();
a61af66fc99e Initial load
duke
parents:
diff changeset
1193 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1194
a61af66fc99e Initial load
duke
parents:
diff changeset
1195 bool CompactibleFreeListSpace::verifyChunkInIndexedFreeLists(FreeChunk* fc)
a61af66fc99e Initial load
duke
parents:
diff changeset
1196 const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1197 assert(fc->size() < IndexSetSize, "Size of chunk is too large");
a61af66fc99e Initial load
duke
parents:
diff changeset
1198 return _indexedFreeList[fc->size()].verifyChunkInFreeLists(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1199 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1200
a61af66fc99e Initial load
duke
parents:
diff changeset
1201 bool CompactibleFreeListSpace::verifyChunkInFreeLists(FreeChunk* fc) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1202 if (fc->size() >= IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1203 return dictionary()->verifyChunkInFreeLists(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1204 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1205 return verifyChunkInIndexedFreeLists(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1206 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1207 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1208
a61af66fc99e Initial load
duke
parents:
diff changeset
1209 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
1210 void CompactibleFreeListSpace::assert_locked() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1211 CMSLockVerifier::assert_locked(freelistLock(), parDictionaryAllocLock());
a61af66fc99e Initial load
duke
parents:
diff changeset
1212 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1213 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
1214
a61af66fc99e Initial load
duke
parents:
diff changeset
1215 FreeChunk* CompactibleFreeListSpace::allocateScratch(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1216 // In the parallel case, the main thread holds the free list lock
a61af66fc99e Initial load
duke
parents:
diff changeset
1217 // on behalf the parallel threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
1218 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1219 FreeChunk* fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1220 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1221 // If GC is parallel, this might be called by several threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
1222 // This should be rare enough that the locking overhead won't affect
a61af66fc99e Initial load
duke
parents:
diff changeset
1223 // the sequential code.
a61af66fc99e Initial load
duke
parents:
diff changeset
1224 MutexLockerEx x(parDictionaryAllocLock(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1225 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
1226 fc = getChunkFromDictionary(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1227 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1228 if (fc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1229 fc->dontCoalesce();
a61af66fc99e Initial load
duke
parents:
diff changeset
1230 assert(fc->isFree(), "Should be free, but not coalescable");
a61af66fc99e Initial load
duke
parents:
diff changeset
1231 // Verify that the block offset table shows this to
a61af66fc99e Initial load
duke
parents:
diff changeset
1232 // be a single block, but not one which is unallocated.
a61af66fc99e Initial load
duke
parents:
diff changeset
1233 _bt.verify_single_block((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1234 _bt.verify_not_unallocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1235 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1236 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1237 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1238
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
1239 oop CompactibleFreeListSpace::promote(oop obj, size_t obj_size) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1240 assert(obj_size == (size_t)obj->size(), "bad obj_size passed in");
a61af66fc99e Initial load
duke
parents:
diff changeset
1241 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1242
a61af66fc99e Initial load
duke
parents:
diff changeset
1243 // if we are tracking promotions, then first ensure space for
a61af66fc99e Initial load
duke
parents:
diff changeset
1244 // promotion (including spooling space for saving header if necessary).
a61af66fc99e Initial load
duke
parents:
diff changeset
1245 // then allocate and copy, then track promoted info if needed.
a61af66fc99e Initial load
duke
parents:
diff changeset
1246 // When tracking (see PromotionInfo::track()), the mark word may
a61af66fc99e Initial load
duke
parents:
diff changeset
1247 // be displaced and in this case restoration of the mark word
a61af66fc99e Initial load
duke
parents:
diff changeset
1248 // occurs in the (oop_since_save_marks_)iterate phase.
a61af66fc99e Initial load
duke
parents:
diff changeset
1249 if (_promoInfo.tracking() && !_promoInfo.ensure_spooling_space()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1250 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1251 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1252 // Call the allocate(size_t, bool) form directly to avoid the
a61af66fc99e Initial load
duke
parents:
diff changeset
1253 // additional call through the allocate(size_t) form. Having
a61af66fc99e Initial load
duke
parents:
diff changeset
1254 // the compile inline the call is problematic because allocate(size_t)
a61af66fc99e Initial load
duke
parents:
diff changeset
1255 // is a virtual method.
a61af66fc99e Initial load
duke
parents:
diff changeset
1256 HeapWord* res = allocate(adjustObjectSize(obj_size));
a61af66fc99e Initial load
duke
parents:
diff changeset
1257 if (res != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1258 Copy::aligned_disjoint_words((HeapWord*)obj, res, obj_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1259 // if we should be tracking promotions, do so.
a61af66fc99e Initial load
duke
parents:
diff changeset
1260 if (_promoInfo.tracking()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1261 _promoInfo.track((PromotedObject*)res);
a61af66fc99e Initial load
duke
parents:
diff changeset
1262 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1263 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1264 return oop(res);
a61af66fc99e Initial load
duke
parents:
diff changeset
1265 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1266
a61af66fc99e Initial load
duke
parents:
diff changeset
1267 HeapWord*
a61af66fc99e Initial load
duke
parents:
diff changeset
1268 CompactibleFreeListSpace::getChunkFromSmallLinearAllocBlock(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1269 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1270 assert(size >= MinChunkSize, "minimum chunk size");
a61af66fc99e Initial load
duke
parents:
diff changeset
1271 assert(size < _smallLinearAllocBlock._allocation_size_limit,
a61af66fc99e Initial load
duke
parents:
diff changeset
1272 "maximum from smallLinearAllocBlock");
a61af66fc99e Initial load
duke
parents:
diff changeset
1273 return getChunkFromLinearAllocBlock(&_smallLinearAllocBlock, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1274 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1275
a61af66fc99e Initial load
duke
parents:
diff changeset
1276 HeapWord*
a61af66fc99e Initial load
duke
parents:
diff changeset
1277 CompactibleFreeListSpace::getChunkFromLinearAllocBlock(LinearAllocBlock *blk,
a61af66fc99e Initial load
duke
parents:
diff changeset
1278 size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1279 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1280 assert(size >= MinChunkSize, "too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1281 HeapWord* res = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1282 // Try to do linear allocation from blk, making sure that
a61af66fc99e Initial load
duke
parents:
diff changeset
1283 if (blk->_word_size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1284 // We have probably been unable to fill this either in the prologue or
a61af66fc99e Initial load
duke
parents:
diff changeset
1285 // when it was exhausted at the last linear allocation. Bail out until
a61af66fc99e Initial load
duke
parents:
diff changeset
1286 // next time.
a61af66fc99e Initial load
duke
parents:
diff changeset
1287 assert(blk->_ptr == NULL, "consistency check");
a61af66fc99e Initial load
duke
parents:
diff changeset
1288 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1289 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1290 assert(blk->_word_size != 0 && blk->_ptr != NULL, "consistency check");
a61af66fc99e Initial load
duke
parents:
diff changeset
1291 res = getChunkFromLinearAllocBlockRemainder(blk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1292 if (res != NULL) return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1293
a61af66fc99e Initial load
duke
parents:
diff changeset
1294 // about to exhaust this linear allocation block
a61af66fc99e Initial load
duke
parents:
diff changeset
1295 if (blk->_word_size == size) { // exactly satisfied
a61af66fc99e Initial load
duke
parents:
diff changeset
1296 res = blk->_ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1297 _bt.allocated(res, blk->_word_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1298 } else if (size + MinChunkSize <= blk->_refillSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1299 // Update _unallocated_block if the size is such that chunk would be
a61af66fc99e Initial load
duke
parents:
diff changeset
1300 // returned to the indexed free list. All other chunks in the indexed
a61af66fc99e Initial load
duke
parents:
diff changeset
1301 // free lists are allocated from the dictionary so that _unallocated_block
a61af66fc99e Initial load
duke
parents:
diff changeset
1302 // has already been adjusted for them. Do it here so that the cost
a61af66fc99e Initial load
duke
parents:
diff changeset
1303 // for all chunks added back to the indexed free lists.
a61af66fc99e Initial load
duke
parents:
diff changeset
1304 if (blk->_word_size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1305 _bt.allocated(blk->_ptr, blk->_word_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1306 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1307 // Return the chunk that isn't big enough, and then refill below.
a61af66fc99e Initial load
duke
parents:
diff changeset
1308 addChunkToFreeLists(blk->_ptr, blk->_word_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1309 _bt.verify_single_block(blk->_ptr, (blk->_ptr + blk->_word_size));
a61af66fc99e Initial load
duke
parents:
diff changeset
1310 // Don't keep statistics on adding back chunk from a LinAB.
a61af66fc99e Initial load
duke
parents:
diff changeset
1311 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1312 // A refilled block would not satisfy the request.
a61af66fc99e Initial load
duke
parents:
diff changeset
1313 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1314 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1315
a61af66fc99e Initial load
duke
parents:
diff changeset
1316 blk->_ptr = NULL; blk->_word_size = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1317 refillLinearAllocBlock(blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1318 assert(blk->_ptr == NULL || blk->_word_size >= size + MinChunkSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
1319 "block was replenished");
a61af66fc99e Initial load
duke
parents:
diff changeset
1320 if (res != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1321 splitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1322 repairLinearAllocBlock(blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1323 } else if (blk->_ptr != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1324 res = blk->_ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1325 size_t blk_size = blk->_word_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1326 blk->_word_size -= size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1327 blk->_ptr += size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1328 splitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1329 repairLinearAllocBlock(blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1330 // Update BOT last so that other (parallel) GC threads see a consistent
a61af66fc99e Initial load
duke
parents:
diff changeset
1331 // view of the BOT and free blocks.
a61af66fc99e Initial load
duke
parents:
diff changeset
1332 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
1333 _bt.split_block(res, blk_size, size); // adjust block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
1334 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1335 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1336 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1337
a61af66fc99e Initial load
duke
parents:
diff changeset
1338 HeapWord* CompactibleFreeListSpace::getChunkFromLinearAllocBlockRemainder(
a61af66fc99e Initial load
duke
parents:
diff changeset
1339 LinearAllocBlock* blk,
a61af66fc99e Initial load
duke
parents:
diff changeset
1340 size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1341 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1342 assert(size >= MinChunkSize, "too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1343
a61af66fc99e Initial load
duke
parents:
diff changeset
1344 HeapWord* res = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1345 // This is the common case. Keep it simple.
a61af66fc99e Initial load
duke
parents:
diff changeset
1346 if (blk->_word_size >= size + MinChunkSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1347 assert(blk->_ptr != NULL, "consistency check");
a61af66fc99e Initial load
duke
parents:
diff changeset
1348 res = blk->_ptr;
a61af66fc99e Initial load
duke
parents:
diff changeset
1349 // Note that the BOT is up-to-date for the linAB before allocation. It
a61af66fc99e Initial load
duke
parents:
diff changeset
1350 // indicates the start of the linAB. The split_block() updates the
a61af66fc99e Initial load
duke
parents:
diff changeset
1351 // BOT for the linAB after the allocation (indicates the start of the
a61af66fc99e Initial load
duke
parents:
diff changeset
1352 // next chunk to be allocated).
a61af66fc99e Initial load
duke
parents:
diff changeset
1353 size_t blk_size = blk->_word_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1354 blk->_word_size -= size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1355 blk->_ptr += size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1356 splitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1357 repairLinearAllocBlock(blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1358 // Update BOT last so that other (parallel) GC threads see a consistent
a61af66fc99e Initial load
duke
parents:
diff changeset
1359 // view of the BOT and free blocks.
a61af66fc99e Initial load
duke
parents:
diff changeset
1360 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
1361 _bt.split_block(res, blk_size, size); // adjust block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
1362 _bt.allocated(res, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1363 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1364 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1365 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1366
a61af66fc99e Initial load
duke
parents:
diff changeset
1367 FreeChunk*
a61af66fc99e Initial load
duke
parents:
diff changeset
1368 CompactibleFreeListSpace::getChunkFromIndexedFreeList(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1369 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1370 assert(size < SmallForDictionary, "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
1371 FreeChunk* res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1372 res = _indexedFreeList[size].getChunkAtHead();
a61af66fc99e Initial load
duke
parents:
diff changeset
1373 if (res == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1374 res = getChunkFromIndexedFreeListHelper(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1375 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1376 _bt.verify_not_unallocated((HeapWord*) res, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1377 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1378 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1379
a61af66fc99e Initial load
duke
parents:
diff changeset
1380 FreeChunk*
a61af66fc99e Initial load
duke
parents:
diff changeset
1381 CompactibleFreeListSpace::getChunkFromIndexedFreeListHelper(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1382 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1383 FreeChunk* fc = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1384 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1385 assert(_indexedFreeList[size].head() == NULL ||
a61af66fc99e Initial load
duke
parents:
diff changeset
1386 _indexedFreeList[size].surplus() <= 0,
a61af66fc99e Initial load
duke
parents:
diff changeset
1387 "List for this size should be empty or under populated");
a61af66fc99e Initial load
duke
parents:
diff changeset
1388 // Try best fit in exact lists before replenishing the list
a61af66fc99e Initial load
duke
parents:
diff changeset
1389 if (!bestFitFirst() || (fc = bestFitSmall(size)) == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1390 // Replenish list.
a61af66fc99e Initial load
duke
parents:
diff changeset
1391 //
a61af66fc99e Initial load
duke
parents:
diff changeset
1392 // Things tried that failed.
a61af66fc99e Initial load
duke
parents:
diff changeset
1393 // Tried allocating out of the two LinAB's first before
a61af66fc99e Initial load
duke
parents:
diff changeset
1394 // replenishing lists.
a61af66fc99e Initial load
duke
parents:
diff changeset
1395 // Tried small linAB of size 256 (size in indexed list)
a61af66fc99e Initial load
duke
parents:
diff changeset
1396 // and replenishing indexed lists from the small linAB.
a61af66fc99e Initial load
duke
parents:
diff changeset
1397 //
a61af66fc99e Initial load
duke
parents:
diff changeset
1398 FreeChunk* newFc = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1399 size_t replenish_size = CMSIndexedFreeListReplenish * size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1400 if (replenish_size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1401 // Do not replenish from an underpopulated size.
a61af66fc99e Initial load
duke
parents:
diff changeset
1402 if (_indexedFreeList[replenish_size].surplus() > 0 &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1403 _indexedFreeList[replenish_size].head() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1404 newFc =
a61af66fc99e Initial load
duke
parents:
diff changeset
1405 _indexedFreeList[replenish_size].getChunkAtHead();
a61af66fc99e Initial load
duke
parents:
diff changeset
1406 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1407 newFc = bestFitSmall(replenish_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1408 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1409 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1410 if (newFc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1411 splitDeath(replenish_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1412 } else if (replenish_size > size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1413 assert(CMSIndexedFreeListReplenish > 1, "ctl pt invariant");
a61af66fc99e Initial load
duke
parents:
diff changeset
1414 newFc =
a61af66fc99e Initial load
duke
parents:
diff changeset
1415 getChunkFromIndexedFreeListHelper(replenish_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1416 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1417 if (newFc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1418 assert(newFc->size() == replenish_size, "Got wrong size");
a61af66fc99e Initial load
duke
parents:
diff changeset
1419 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1420 FreeChunk *curFc, *nextFc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1421 // carve up and link blocks 0, ..., CMSIndexedFreeListReplenish - 2
a61af66fc99e Initial load
duke
parents:
diff changeset
1422 // The last chunk is not added to the lists but is returned as the
a61af66fc99e Initial load
duke
parents:
diff changeset
1423 // free chunk.
a61af66fc99e Initial load
duke
parents:
diff changeset
1424 for (curFc = newFc, nextFc = (FreeChunk*)((HeapWord*)curFc + size),
a61af66fc99e Initial load
duke
parents:
diff changeset
1425 i = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1426 i < (CMSIndexedFreeListReplenish - 1);
a61af66fc99e Initial load
duke
parents:
diff changeset
1427 curFc = nextFc, nextFc = (FreeChunk*)((HeapWord*)nextFc + size),
a61af66fc99e Initial load
duke
parents:
diff changeset
1428 i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1429 curFc->setSize(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1430 // Don't record this as a return in order to try and
a61af66fc99e Initial load
duke
parents:
diff changeset
1431 // determine the "returns" from a GC.
a61af66fc99e Initial load
duke
parents:
diff changeset
1432 _bt.verify_not_unallocated((HeapWord*) fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1433 _indexedFreeList[size].returnChunkAtTail(curFc, false);
a61af66fc99e Initial load
duke
parents:
diff changeset
1434 _bt.mark_block((HeapWord*)curFc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1435 splitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1436 // Don't record the initial population of the indexed list
a61af66fc99e Initial load
duke
parents:
diff changeset
1437 // as a split birth.
a61af66fc99e Initial load
duke
parents:
diff changeset
1438 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1439
a61af66fc99e Initial load
duke
parents:
diff changeset
1440 // check that the arithmetic was OK above
a61af66fc99e Initial load
duke
parents:
diff changeset
1441 assert((HeapWord*)nextFc == (HeapWord*)newFc + replenish_size,
a61af66fc99e Initial load
duke
parents:
diff changeset
1442 "inconsistency in carving newFc");
a61af66fc99e Initial load
duke
parents:
diff changeset
1443 curFc->setSize(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1444 _bt.mark_block((HeapWord*)curFc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1445 splitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1446 return curFc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1447 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1448 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1449 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1450 // Get a free chunk from the free chunk dictionary to be returned to
a61af66fc99e Initial load
duke
parents:
diff changeset
1451 // replenish the indexed free list.
a61af66fc99e Initial load
duke
parents:
diff changeset
1452 fc = getChunkFromDictionaryExact(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1453 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1454 assert(fc == NULL || fc->isFree(), "Should be returning a free chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1455 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1456 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1457
a61af66fc99e Initial load
duke
parents:
diff changeset
1458 FreeChunk*
a61af66fc99e Initial load
duke
parents:
diff changeset
1459 CompactibleFreeListSpace::getChunkFromDictionary(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1460 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1461 FreeChunk* fc = _dictionary->getChunk(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1462 if (fc == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1463 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1464 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1465 _bt.allocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1466 if (fc->size() >= size + MinChunkSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1467 fc = splitChunkAndReturnRemainder(fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1468 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1469 assert(fc->size() >= size, "chunk too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1470 assert(fc->size() < size + MinChunkSize, "chunk too big");
a61af66fc99e Initial load
duke
parents:
diff changeset
1471 _bt.verify_single_block((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1472 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1473 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1474
a61af66fc99e Initial load
duke
parents:
diff changeset
1475 FreeChunk*
a61af66fc99e Initial load
duke
parents:
diff changeset
1476 CompactibleFreeListSpace::getChunkFromDictionaryExact(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1477 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1478 FreeChunk* fc = _dictionary->getChunk(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1479 if (fc == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1480 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1481 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1482 _bt.allocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1483 if (fc->size() == size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1484 _bt.verify_single_block((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1485 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1486 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1487 assert(fc->size() > size, "getChunk() guarantee");
a61af66fc99e Initial load
duke
parents:
diff changeset
1488 if (fc->size() < size + MinChunkSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1489 // Return the chunk to the dictionary and go get a bigger one.
a61af66fc99e Initial load
duke
parents:
diff changeset
1490 returnChunkToDictionary(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1491 fc = _dictionary->getChunk(size + MinChunkSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1492 if (fc == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1493 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1494 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1495 _bt.allocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
1496 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1497 assert(fc->size() >= size + MinChunkSize, "tautology");
a61af66fc99e Initial load
duke
parents:
diff changeset
1498 fc = splitChunkAndReturnRemainder(fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1499 assert(fc->size() == size, "chunk is wrong size");
a61af66fc99e Initial load
duke
parents:
diff changeset
1500 _bt.verify_single_block((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1501 return fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1502 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1503
a61af66fc99e Initial load
duke
parents:
diff changeset
1504 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1505 CompactibleFreeListSpace::returnChunkToDictionary(FreeChunk* chunk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1506 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1507
a61af66fc99e Initial load
duke
parents:
diff changeset
1508 size_t size = chunk->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1509 _bt.verify_single_block((HeapWord*)chunk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1510 // adjust _unallocated_block downward, as necessary
a61af66fc99e Initial load
duke
parents:
diff changeset
1511 _bt.freed((HeapWord*)chunk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1512 _dictionary->returnChunk(chunk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1513 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1514
a61af66fc99e Initial load
duke
parents:
diff changeset
1515 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1516 CompactibleFreeListSpace::returnChunkToFreeList(FreeChunk* fc) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1517 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1518 size_t size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1519 _bt.verify_single_block((HeapWord*) fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1520 _bt.verify_not_unallocated((HeapWord*) fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1521 if (_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1522 _indexedFreeList[size].returnChunkAtTail(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1523 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1524 _indexedFreeList[size].returnChunkAtHead(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1525 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1526 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1527
a61af66fc99e Initial load
duke
parents:
diff changeset
1528 // Add chunk to end of last block -- if it's the largest
a61af66fc99e Initial load
duke
parents:
diff changeset
1529 // block -- and update BOT and census data. We would
a61af66fc99e Initial load
duke
parents:
diff changeset
1530 // of course have preferred to coalesce it with the
a61af66fc99e Initial load
duke
parents:
diff changeset
1531 // last block, but it's currently less expensive to find the
a61af66fc99e Initial load
duke
parents:
diff changeset
1532 // largest block than it is to find the last.
a61af66fc99e Initial load
duke
parents:
diff changeset
1533 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1534 CompactibleFreeListSpace::addChunkToFreeListsAtEndRecordingStats(
a61af66fc99e Initial load
duke
parents:
diff changeset
1535 HeapWord* chunk, size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1536 // check that the chunk does lie in this space!
a61af66fc99e Initial load
duke
parents:
diff changeset
1537 assert(chunk != NULL && is_in_reserved(chunk), "Not in this space!");
a61af66fc99e Initial load
duke
parents:
diff changeset
1538 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1539 // One of the parallel gc task threads may be here
a61af66fc99e Initial load
duke
parents:
diff changeset
1540 // whilst others are allocating.
a61af66fc99e Initial load
duke
parents:
diff changeset
1541 Mutex* lock = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1542 if (ParallelGCThreads != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1543 lock = &_parDictionaryAllocLock;
a61af66fc99e Initial load
duke
parents:
diff changeset
1544 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1545 FreeChunk* ec;
a61af66fc99e Initial load
duke
parents:
diff changeset
1546 {
a61af66fc99e Initial load
duke
parents:
diff changeset
1547 MutexLockerEx x(lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
1548 ec = dictionary()->findLargestDict(); // get largest block
a61af66fc99e Initial load
duke
parents:
diff changeset
1549 if (ec != NULL && ec->end() == chunk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1550 // It's a coterminal block - we can coalesce.
a61af66fc99e Initial load
duke
parents:
diff changeset
1551 size_t old_size = ec->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1552 coalDeath(old_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1553 removeChunkFromDictionary(ec);
a61af66fc99e Initial load
duke
parents:
diff changeset
1554 size += old_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1555 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1556 ec = (FreeChunk*)chunk;
a61af66fc99e Initial load
duke
parents:
diff changeset
1557 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1558 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1559 ec->setSize(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1560 debug_only(ec->mangleFreed(size));
a61af66fc99e Initial load
duke
parents:
diff changeset
1561 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1562 lock = _indexedFreeListParLocks[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
1563 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1564 MutexLockerEx x(lock, Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
1565 addChunkAndRepairOffsetTable((HeapWord*)ec, size, true);
a61af66fc99e Initial load
duke
parents:
diff changeset
1566 // record the birth under the lock since the recording involves
a61af66fc99e Initial load
duke
parents:
diff changeset
1567 // manipulation of the list on which the chunk lives and
a61af66fc99e Initial load
duke
parents:
diff changeset
1568 // if the chunk is allocated and is the last on the list,
a61af66fc99e Initial load
duke
parents:
diff changeset
1569 // the list can go away.
a61af66fc99e Initial load
duke
parents:
diff changeset
1570 coalBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1571 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1572
a61af66fc99e Initial load
duke
parents:
diff changeset
1573 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1574 CompactibleFreeListSpace::addChunkToFreeLists(HeapWord* chunk,
a61af66fc99e Initial load
duke
parents:
diff changeset
1575 size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1576 // check that the chunk does lie in this space!
a61af66fc99e Initial load
duke
parents:
diff changeset
1577 assert(chunk != NULL && is_in_reserved(chunk), "Not in this space!");
a61af66fc99e Initial load
duke
parents:
diff changeset
1578 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1579 _bt.verify_single_block(chunk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1580
a61af66fc99e Initial load
duke
parents:
diff changeset
1581 FreeChunk* fc = (FreeChunk*) chunk;
a61af66fc99e Initial load
duke
parents:
diff changeset
1582 fc->setSize(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1583 debug_only(fc->mangleFreed(size));
a61af66fc99e Initial load
duke
parents:
diff changeset
1584 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1585 returnChunkToFreeList(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1586 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1587 returnChunkToDictionary(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1588 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1589 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1590
a61af66fc99e Initial load
duke
parents:
diff changeset
1591 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1592 CompactibleFreeListSpace::addChunkAndRepairOffsetTable(HeapWord* chunk,
a61af66fc99e Initial load
duke
parents:
diff changeset
1593 size_t size, bool coalesced) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1594 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1595 assert(chunk != NULL, "null chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1596 if (coalesced) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1597 // repair BOT
a61af66fc99e Initial load
duke
parents:
diff changeset
1598 _bt.single_block(chunk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1599 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1600 addChunkToFreeLists(chunk, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1601 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1602
a61af66fc99e Initial load
duke
parents:
diff changeset
1603 // We _must_ find the purported chunk on our free lists;
a61af66fc99e Initial load
duke
parents:
diff changeset
1604 // we assert if we don't.
a61af66fc99e Initial load
duke
parents:
diff changeset
1605 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1606 CompactibleFreeListSpace::removeFreeChunkFromFreeLists(FreeChunk* fc) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1607 size_t size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1608 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1609 debug_only(verifyFreeLists());
a61af66fc99e Initial load
duke
parents:
diff changeset
1610 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1611 removeChunkFromIndexedFreeList(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1612 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1613 removeChunkFromDictionary(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1614 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1615 _bt.verify_single_block((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1616 debug_only(verifyFreeLists());
a61af66fc99e Initial load
duke
parents:
diff changeset
1617 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1618
a61af66fc99e Initial load
duke
parents:
diff changeset
1619 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1620 CompactibleFreeListSpace::removeChunkFromDictionary(FreeChunk* fc) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1621 size_t size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1622 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1623 assert(fc != NULL, "null chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1624 _bt.verify_single_block((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1625 _dictionary->removeChunk(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1626 // adjust _unallocated_block upward, as necessary
a61af66fc99e Initial load
duke
parents:
diff changeset
1627 _bt.allocated((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1628 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1629
a61af66fc99e Initial load
duke
parents:
diff changeset
1630 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1631 CompactibleFreeListSpace::removeChunkFromIndexedFreeList(FreeChunk* fc) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1632 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1633 size_t size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1634 _bt.verify_single_block((HeapWord*)fc, size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1635 NOT_PRODUCT(
a61af66fc99e Initial load
duke
parents:
diff changeset
1636 if (FLSVerifyIndexTable) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1637 verifyIndexedFreeList(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1638 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1639 )
a61af66fc99e Initial load
duke
parents:
diff changeset
1640 _indexedFreeList[size].removeChunk(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1641 debug_only(fc->clearNext());
a61af66fc99e Initial load
duke
parents:
diff changeset
1642 debug_only(fc->clearPrev());
a61af66fc99e Initial load
duke
parents:
diff changeset
1643 NOT_PRODUCT(
a61af66fc99e Initial load
duke
parents:
diff changeset
1644 if (FLSVerifyIndexTable) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1645 verifyIndexedFreeList(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1646 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1647 )
a61af66fc99e Initial load
duke
parents:
diff changeset
1648 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1649
a61af66fc99e Initial load
duke
parents:
diff changeset
1650 FreeChunk* CompactibleFreeListSpace::bestFitSmall(size_t numWords) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1651 /* A hint is the next larger size that has a surplus.
a61af66fc99e Initial load
duke
parents:
diff changeset
1652 Start search at a size large enough to guarantee that
a61af66fc99e Initial load
duke
parents:
diff changeset
1653 the excess is >= MIN_CHUNK. */
a61af66fc99e Initial load
duke
parents:
diff changeset
1654 size_t start = align_object_size(numWords + MinChunkSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1655 if (start < IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1656 FreeList* it = _indexedFreeList;
a61af66fc99e Initial load
duke
parents:
diff changeset
1657 size_t hint = _indexedFreeList[start].hint();
a61af66fc99e Initial load
duke
parents:
diff changeset
1658 while (hint < IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1659 assert(hint % MinObjAlignment == 0, "hint should be aligned");
a61af66fc99e Initial load
duke
parents:
diff changeset
1660 FreeList *fl = &_indexedFreeList[hint];
a61af66fc99e Initial load
duke
parents:
diff changeset
1661 if (fl->surplus() > 0 && fl->head() != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1662 // Found a list with surplus, reset original hint
a61af66fc99e Initial load
duke
parents:
diff changeset
1663 // and split out a free chunk which is returned.
a61af66fc99e Initial load
duke
parents:
diff changeset
1664 _indexedFreeList[start].set_hint(hint);
a61af66fc99e Initial load
duke
parents:
diff changeset
1665 FreeChunk* res = getFromListGreater(fl, numWords);
a61af66fc99e Initial load
duke
parents:
diff changeset
1666 assert(res == NULL || res->isFree(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1667 "Should be returning a free chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1668 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
1669 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1670 hint = fl->hint(); /* keep looking */
a61af66fc99e Initial load
duke
parents:
diff changeset
1671 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1672 /* None found. */
a61af66fc99e Initial load
duke
parents:
diff changeset
1673 it[start].set_hint(IndexSetSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1674 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1675 return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
1676 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1677
a61af66fc99e Initial load
duke
parents:
diff changeset
1678 /* Requires fl->size >= numWords + MinChunkSize */
a61af66fc99e Initial load
duke
parents:
diff changeset
1679 FreeChunk* CompactibleFreeListSpace::getFromListGreater(FreeList* fl,
a61af66fc99e Initial load
duke
parents:
diff changeset
1680 size_t numWords) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1681 FreeChunk *curr = fl->head();
a61af66fc99e Initial load
duke
parents:
diff changeset
1682 size_t oldNumWords = curr->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1683 assert(numWords >= MinChunkSize, "Word size is too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1684 assert(curr != NULL, "List is empty");
a61af66fc99e Initial load
duke
parents:
diff changeset
1685 assert(oldNumWords >= numWords + MinChunkSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
1686 "Size of chunks in the list is too small");
a61af66fc99e Initial load
duke
parents:
diff changeset
1687
a61af66fc99e Initial load
duke
parents:
diff changeset
1688 fl->removeChunk(curr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1689 // recorded indirectly by splitChunkAndReturnRemainder -
a61af66fc99e Initial load
duke
parents:
diff changeset
1690 // smallSplit(oldNumWords, numWords);
a61af66fc99e Initial load
duke
parents:
diff changeset
1691 FreeChunk* new_chunk = splitChunkAndReturnRemainder(curr, numWords);
a61af66fc99e Initial load
duke
parents:
diff changeset
1692 // Does anything have to be done for the remainder in terms of
a61af66fc99e Initial load
duke
parents:
diff changeset
1693 // fixing the card table?
a61af66fc99e Initial load
duke
parents:
diff changeset
1694 assert(new_chunk == NULL || new_chunk->isFree(),
a61af66fc99e Initial load
duke
parents:
diff changeset
1695 "Should be returning a free chunk");
a61af66fc99e Initial load
duke
parents:
diff changeset
1696 return new_chunk;
a61af66fc99e Initial load
duke
parents:
diff changeset
1697 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1698
a61af66fc99e Initial load
duke
parents:
diff changeset
1699 FreeChunk*
a61af66fc99e Initial load
duke
parents:
diff changeset
1700 CompactibleFreeListSpace::splitChunkAndReturnRemainder(FreeChunk* chunk,
a61af66fc99e Initial load
duke
parents:
diff changeset
1701 size_t new_size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1702 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1703 size_t size = chunk->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1704 assert(size > new_size, "Split from a smaller block?");
a61af66fc99e Initial load
duke
parents:
diff changeset
1705 assert(is_aligned(chunk), "alignment problem");
a61af66fc99e Initial load
duke
parents:
diff changeset
1706 assert(size == adjustObjectSize(size), "alignment problem");
a61af66fc99e Initial load
duke
parents:
diff changeset
1707 size_t rem_size = size - new_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
1708 assert(rem_size == adjustObjectSize(rem_size), "alignment problem");
a61af66fc99e Initial load
duke
parents:
diff changeset
1709 assert(rem_size >= MinChunkSize, "Free chunk smaller than minimum");
a61af66fc99e Initial load
duke
parents:
diff changeset
1710 FreeChunk* ffc = (FreeChunk*)((HeapWord*)chunk + new_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1711 assert(is_aligned(ffc), "alignment problem");
a61af66fc99e Initial load
duke
parents:
diff changeset
1712 ffc->setSize(rem_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1713 ffc->linkNext(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
1714 ffc->linkPrev(NULL); // Mark as a free block for other (parallel) GC threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
1715 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
1716 // adjust block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
1717 _bt.split_block((HeapWord*)chunk, chunk->size(), new_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1718 if (rem_size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1719 bool is_par = (SharedHeap::heap()->n_par_threads() > 0);
a61af66fc99e Initial load
duke
parents:
diff changeset
1720 if (is_par) _indexedFreeListParLocks[rem_size]->lock();
a61af66fc99e Initial load
duke
parents:
diff changeset
1721 returnChunkToFreeList(ffc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1722 split(size, rem_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1723 if (is_par) _indexedFreeListParLocks[rem_size]->unlock();
a61af66fc99e Initial load
duke
parents:
diff changeset
1724 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1725 returnChunkToDictionary(ffc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1726 split(size ,rem_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1727 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1728 chunk->setSize(new_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1729 return chunk;
a61af66fc99e Initial load
duke
parents:
diff changeset
1730 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1731
a61af66fc99e Initial load
duke
parents:
diff changeset
1732 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1733 CompactibleFreeListSpace::sweep_completed() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1734 // Now that space is probably plentiful, refill linear
a61af66fc99e Initial load
duke
parents:
diff changeset
1735 // allocation blocks as needed.
a61af66fc99e Initial load
duke
parents:
diff changeset
1736 refillLinearAllocBlocksIfNeeded();
a61af66fc99e Initial load
duke
parents:
diff changeset
1737 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1738
a61af66fc99e Initial load
duke
parents:
diff changeset
1739 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1740 CompactibleFreeListSpace::gc_prologue() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1741 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1742 if (PrintFLSStatistics != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1743 gclog_or_tty->print("Before GC:\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
1744 reportFreeListStatistics();
a61af66fc99e Initial load
duke
parents:
diff changeset
1745 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1746 refillLinearAllocBlocksIfNeeded();
a61af66fc99e Initial load
duke
parents:
diff changeset
1747 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1748
a61af66fc99e Initial load
duke
parents:
diff changeset
1749 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1750 CompactibleFreeListSpace::gc_epilogue() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1751 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1752 if (PrintGCDetails && Verbose && !_adaptive_freelists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1753 if (_smallLinearAllocBlock._word_size == 0)
a61af66fc99e Initial load
duke
parents:
diff changeset
1754 warning("CompactibleFreeListSpace(epilogue):: Linear allocation failure");
a61af66fc99e Initial load
duke
parents:
diff changeset
1755 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1756 assert(_promoInfo.noPromotions(), "_promoInfo inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
1757 _promoInfo.stopTrackingPromotions();
a61af66fc99e Initial load
duke
parents:
diff changeset
1758 repairLinearAllocationBlocks();
a61af66fc99e Initial load
duke
parents:
diff changeset
1759 // Print Space's stats
a61af66fc99e Initial load
duke
parents:
diff changeset
1760 if (PrintFLSStatistics != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1761 gclog_or_tty->print("After GC:\n");
a61af66fc99e Initial load
duke
parents:
diff changeset
1762 reportFreeListStatistics();
a61af66fc99e Initial load
duke
parents:
diff changeset
1763 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1764 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1765
a61af66fc99e Initial load
duke
parents:
diff changeset
1766 // Iteration support, mostly delegated from a CMS generation
a61af66fc99e Initial load
duke
parents:
diff changeset
1767
a61af66fc99e Initial load
duke
parents:
diff changeset
1768 void CompactibleFreeListSpace::save_marks() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1769 // mark the "end" of the used space at the time of this call;
a61af66fc99e Initial load
duke
parents:
diff changeset
1770 // note, however, that promoted objects from this point
a61af66fc99e Initial load
duke
parents:
diff changeset
1771 // on are tracked in the _promoInfo below.
a61af66fc99e Initial load
duke
parents:
diff changeset
1772 set_saved_mark_word(BlockOffsetArrayUseUnallocatedBlock ?
a61af66fc99e Initial load
duke
parents:
diff changeset
1773 unallocated_block() : end());
a61af66fc99e Initial load
duke
parents:
diff changeset
1774 // inform allocator that promotions should be tracked.
a61af66fc99e Initial load
duke
parents:
diff changeset
1775 assert(_promoInfo.noPromotions(), "_promoInfo inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
1776 _promoInfo.startTrackingPromotions();
a61af66fc99e Initial load
duke
parents:
diff changeset
1777 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1778
a61af66fc99e Initial load
duke
parents:
diff changeset
1779 bool CompactibleFreeListSpace::no_allocs_since_save_marks() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1780 assert(_promoInfo.tracking(), "No preceding save_marks?");
a61af66fc99e Initial load
duke
parents:
diff changeset
1781 guarantee(SharedHeap::heap()->n_par_threads() == 0,
a61af66fc99e Initial load
duke
parents:
diff changeset
1782 "Shouldn't be called (yet) during parallel part of gc.");
a61af66fc99e Initial load
duke
parents:
diff changeset
1783 return _promoInfo.noPromotions();
a61af66fc99e Initial load
duke
parents:
diff changeset
1784 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1785
a61af66fc99e Initial load
duke
parents:
diff changeset
1786 #define CFLS_OOP_SINCE_SAVE_MARKS_DEFN(OopClosureType, nv_suffix) \
a61af66fc99e Initial load
duke
parents:
diff changeset
1787 \
a61af66fc99e Initial load
duke
parents:
diff changeset
1788 void CompactibleFreeListSpace:: \
a61af66fc99e Initial load
duke
parents:
diff changeset
1789 oop_since_save_marks_iterate##nv_suffix(OopClosureType* blk) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1790 assert(SharedHeap::heap()->n_par_threads() == 0, \
a61af66fc99e Initial load
duke
parents:
diff changeset
1791 "Shouldn't be called (yet) during parallel part of gc."); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1792 _promoInfo.promoted_oops_iterate##nv_suffix(blk); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1793 /* \
a61af66fc99e Initial load
duke
parents:
diff changeset
1794 * This also restores any displaced headers and removes the elements from \
a61af66fc99e Initial load
duke
parents:
diff changeset
1795 * the iteration set as they are processed, so that we have a clean slate \
a61af66fc99e Initial load
duke
parents:
diff changeset
1796 * at the end of the iteration. Note, thus, that if new objects are \
a61af66fc99e Initial load
duke
parents:
diff changeset
1797 * promoted as a result of the iteration they are iterated over as well. \
a61af66fc99e Initial load
duke
parents:
diff changeset
1798 */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1799 assert(_promoInfo.noPromotions(), "_promoInfo inconsistency"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1800 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1801
a61af66fc99e Initial load
duke
parents:
diff changeset
1802 ALL_SINCE_SAVE_MARKS_CLOSURES(CFLS_OOP_SINCE_SAVE_MARKS_DEFN)
a61af66fc99e Initial load
duke
parents:
diff changeset
1803
a61af66fc99e Initial load
duke
parents:
diff changeset
1804 //////////////////////////////////////////////////////////////////////////////
a61af66fc99e Initial load
duke
parents:
diff changeset
1805 // We go over the list of promoted objects, removing each from the list,
a61af66fc99e Initial load
duke
parents:
diff changeset
1806 // and applying the closure (this may, in turn, add more elements to
a61af66fc99e Initial load
duke
parents:
diff changeset
1807 // the tail of the promoted list, and these newly added objects will
a61af66fc99e Initial load
duke
parents:
diff changeset
1808 // also be processed) until the list is empty.
a61af66fc99e Initial load
duke
parents:
diff changeset
1809 // To aid verification and debugging, in the non-product builds
a61af66fc99e Initial load
duke
parents:
diff changeset
1810 // we actually forward _promoHead each time we process a promoted oop.
a61af66fc99e Initial load
duke
parents:
diff changeset
1811 // Note that this is not necessary in general (i.e. when we don't need to
a61af66fc99e Initial load
duke
parents:
diff changeset
1812 // call PromotionInfo::verify()) because oop_iterate can only add to the
a61af66fc99e Initial load
duke
parents:
diff changeset
1813 // end of _promoTail, and never needs to look at _promoHead.
a61af66fc99e Initial load
duke
parents:
diff changeset
1814
a61af66fc99e Initial load
duke
parents:
diff changeset
1815 #define PROMOTED_OOPS_ITERATE_DEFN(OopClosureType, nv_suffix) \
a61af66fc99e Initial load
duke
parents:
diff changeset
1816 \
a61af66fc99e Initial load
duke
parents:
diff changeset
1817 void PromotionInfo::promoted_oops_iterate##nv_suffix(OopClosureType* cl) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1818 NOT_PRODUCT(verify()); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1819 PromotedObject *curObj, *nextObj; \
a61af66fc99e Initial load
duke
parents:
diff changeset
1820 for (curObj = _promoHead; curObj != NULL; curObj = nextObj) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1821 if ((nextObj = curObj->next()) == NULL) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1822 /* protect ourselves against additions due to closure application \
a61af66fc99e Initial load
duke
parents:
diff changeset
1823 below by resetting the list. */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1824 assert(_promoTail == curObj, "Should have been the tail"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1825 _promoHead = _promoTail = NULL; \
a61af66fc99e Initial load
duke
parents:
diff changeset
1826 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
1827 if (curObj->hasDisplacedMark()) { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1828 /* restore displaced header */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1829 oop(curObj)->set_mark(nextDisplacedHeader()); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1830 } else { \
a61af66fc99e Initial load
duke
parents:
diff changeset
1831 /* restore prototypical header */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1832 oop(curObj)->init_mark(); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1833 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
1834 /* The "promoted_mark" should now not be set */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1835 assert(!curObj->hasPromotedMark(), \
a61af66fc99e Initial load
duke
parents:
diff changeset
1836 "Should have been cleared by restoring displaced mark-word"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1837 NOT_PRODUCT(_promoHead = nextObj); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1838 if (cl != NULL) oop(curObj)->oop_iterate(cl); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1839 if (nextObj == NULL) { /* start at head of list reset above */ \
a61af66fc99e Initial load
duke
parents:
diff changeset
1840 nextObj = _promoHead; \
a61af66fc99e Initial load
duke
parents:
diff changeset
1841 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
1842 } \
a61af66fc99e Initial load
duke
parents:
diff changeset
1843 assert(noPromotions(), "post-condition violation"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1844 assert(_promoHead == NULL && _promoTail == NULL, "emptied promoted list");\
a61af66fc99e Initial load
duke
parents:
diff changeset
1845 assert(_spoolHead == _spoolTail, "emptied spooling buffers"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1846 assert(_firstIndex == _nextIndex, "empty buffer"); \
a61af66fc99e Initial load
duke
parents:
diff changeset
1847 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1848
a61af66fc99e Initial load
duke
parents:
diff changeset
1849 // This should have been ALL_SINCE_...() just like the others,
a61af66fc99e Initial load
duke
parents:
diff changeset
1850 // but, because the body of the method above is somehwat longer,
a61af66fc99e Initial load
duke
parents:
diff changeset
1851 // the MSVC compiler cannot cope; as a workaround, we split the
a61af66fc99e Initial load
duke
parents:
diff changeset
1852 // macro into its 3 constituent parts below (see original macro
a61af66fc99e Initial load
duke
parents:
diff changeset
1853 // definition in specializedOopClosures.hpp).
a61af66fc99e Initial load
duke
parents:
diff changeset
1854 SPECIALIZED_SINCE_SAVE_MARKS_CLOSURES_YOUNG(PROMOTED_OOPS_ITERATE_DEFN)
a61af66fc99e Initial load
duke
parents:
diff changeset
1855 PROMOTED_OOPS_ITERATE_DEFN(OopsInGenClosure,_v)
a61af66fc99e Initial load
duke
parents:
diff changeset
1856
a61af66fc99e Initial load
duke
parents:
diff changeset
1857
a61af66fc99e Initial load
duke
parents:
diff changeset
1858 void CompactibleFreeListSpace::object_iterate_since_last_GC(ObjectClosure* cl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1859 // ugghh... how would one do this efficiently for a non-contiguous space?
a61af66fc99e Initial load
duke
parents:
diff changeset
1860 guarantee(false, "NYI");
a61af66fc99e Initial load
duke
parents:
diff changeset
1861 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1862
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1863 bool CompactibleFreeListSpace::linearAllocationWouldFail() const {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1864 return _smallLinearAllocBlock._word_size == 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1865 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1866
a61af66fc99e Initial load
duke
parents:
diff changeset
1867 void CompactibleFreeListSpace::repairLinearAllocationBlocks() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1868 // Fix up linear allocation blocks to look like free blocks
a61af66fc99e Initial load
duke
parents:
diff changeset
1869 repairLinearAllocBlock(&_smallLinearAllocBlock);
a61af66fc99e Initial load
duke
parents:
diff changeset
1870 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1871
a61af66fc99e Initial load
duke
parents:
diff changeset
1872 void CompactibleFreeListSpace::repairLinearAllocBlock(LinearAllocBlock* blk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1873 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1874 if (blk->_ptr != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1875 assert(blk->_word_size != 0 && blk->_word_size >= MinChunkSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
1876 "Minimum block size requirement");
a61af66fc99e Initial load
duke
parents:
diff changeset
1877 FreeChunk* fc = (FreeChunk*)(blk->_ptr);
a61af66fc99e Initial load
duke
parents:
diff changeset
1878 fc->setSize(blk->_word_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1879 fc->linkPrev(NULL); // mark as free
a61af66fc99e Initial load
duke
parents:
diff changeset
1880 fc->dontCoalesce();
a61af66fc99e Initial load
duke
parents:
diff changeset
1881 assert(fc->isFree(), "just marked it free");
a61af66fc99e Initial load
duke
parents:
diff changeset
1882 assert(fc->cantCoalesce(), "just marked it uncoalescable");
a61af66fc99e Initial load
duke
parents:
diff changeset
1883 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1884 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1885
a61af66fc99e Initial load
duke
parents:
diff changeset
1886 void CompactibleFreeListSpace::refillLinearAllocBlocksIfNeeded() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1887 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1888 if (_smallLinearAllocBlock._ptr == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1889 assert(_smallLinearAllocBlock._word_size == 0,
a61af66fc99e Initial load
duke
parents:
diff changeset
1890 "Size of linAB should be zero if the ptr is NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
1891 // Reset the linAB refill and allocation size limit.
a61af66fc99e Initial load
duke
parents:
diff changeset
1892 _smallLinearAllocBlock.set(0, 0, 1024*SmallForLinearAlloc, SmallForLinearAlloc);
a61af66fc99e Initial load
duke
parents:
diff changeset
1893 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1894 refillLinearAllocBlockIfNeeded(&_smallLinearAllocBlock);
a61af66fc99e Initial load
duke
parents:
diff changeset
1895 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1896
a61af66fc99e Initial load
duke
parents:
diff changeset
1897 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1898 CompactibleFreeListSpace::refillLinearAllocBlockIfNeeded(LinearAllocBlock* blk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1899 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1900 assert((blk->_ptr == NULL && blk->_word_size == 0) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
1901 (blk->_ptr != NULL && blk->_word_size >= MinChunkSize),
a61af66fc99e Initial load
duke
parents:
diff changeset
1902 "blk invariant");
a61af66fc99e Initial load
duke
parents:
diff changeset
1903 if (blk->_ptr == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1904 refillLinearAllocBlock(blk);
a61af66fc99e Initial load
duke
parents:
diff changeset
1905 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1906 if (PrintMiscellaneous && Verbose) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1907 if (blk->_word_size == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1908 warning("CompactibleFreeListSpace(prologue):: Linear allocation failure");
a61af66fc99e Initial load
duke
parents:
diff changeset
1909 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1910 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1911 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1912
a61af66fc99e Initial load
duke
parents:
diff changeset
1913 void
a61af66fc99e Initial load
duke
parents:
diff changeset
1914 CompactibleFreeListSpace::refillLinearAllocBlock(LinearAllocBlock* blk) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1915 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1916 assert(blk->_word_size == 0 && blk->_ptr == NULL,
a61af66fc99e Initial load
duke
parents:
diff changeset
1917 "linear allocation block should be empty");
a61af66fc99e Initial load
duke
parents:
diff changeset
1918 FreeChunk* fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1919 if (blk->_refillSize < SmallForDictionary &&
a61af66fc99e Initial load
duke
parents:
diff changeset
1920 (fc = getChunkFromIndexedFreeList(blk->_refillSize)) != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1921 // A linAB's strategy might be to use small sizes to reduce
a61af66fc99e Initial load
duke
parents:
diff changeset
1922 // fragmentation but still get the benefits of allocation from a
a61af66fc99e Initial load
duke
parents:
diff changeset
1923 // linAB.
a61af66fc99e Initial load
duke
parents:
diff changeset
1924 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1925 fc = getChunkFromDictionary(blk->_refillSize);
a61af66fc99e Initial load
duke
parents:
diff changeset
1926 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1927 if (fc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1928 blk->_ptr = (HeapWord*)fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
1929 blk->_word_size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
1930 fc->dontCoalesce(); // to prevent sweeper from sweeping us up
a61af66fc99e Initial load
duke
parents:
diff changeset
1931 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1932 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1933
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1934 // Support for concurrent collection policy decisions.
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1935 bool CompactibleFreeListSpace::should_concurrent_collect() const {
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1936 // In the future we might want to add in frgamentation stats --
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1937 // including erosion of the "mountain" into this decision as well.
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1938 return !adaptive_freelists() && linearAllocationWouldFail();
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1939 }
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
1940
0
a61af66fc99e Initial load
duke
parents:
diff changeset
1941 // Support for compaction
a61af66fc99e Initial load
duke
parents:
diff changeset
1942
a61af66fc99e Initial load
duke
parents:
diff changeset
1943 void CompactibleFreeListSpace::prepare_for_compaction(CompactPoint* cp) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1944 SCAN_AND_FORWARD(cp,end,block_is_obj,block_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1945 // prepare_for_compaction() uses the space between live objects
a61af66fc99e Initial load
duke
parents:
diff changeset
1946 // so that later phase can skip dead space quickly. So verification
a61af66fc99e Initial load
duke
parents:
diff changeset
1947 // of the free lists doesn't work after.
a61af66fc99e Initial load
duke
parents:
diff changeset
1948 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1949
a61af66fc99e Initial load
duke
parents:
diff changeset
1950 #define obj_size(q) adjustObjectSize(oop(q)->size())
a61af66fc99e Initial load
duke
parents:
diff changeset
1951 #define adjust_obj_size(s) adjustObjectSize(s)
a61af66fc99e Initial load
duke
parents:
diff changeset
1952
a61af66fc99e Initial load
duke
parents:
diff changeset
1953 void CompactibleFreeListSpace::adjust_pointers() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1954 // In other versions of adjust_pointers(), a bail out
a61af66fc99e Initial load
duke
parents:
diff changeset
1955 // based on the amount of live data in the generation
a61af66fc99e Initial load
duke
parents:
diff changeset
1956 // (i.e., if 0, bail out) may be used.
a61af66fc99e Initial load
duke
parents:
diff changeset
1957 // Cannot test used() == 0 here because the free lists have already
a61af66fc99e Initial load
duke
parents:
diff changeset
1958 // been mangled by the compaction.
a61af66fc99e Initial load
duke
parents:
diff changeset
1959
a61af66fc99e Initial load
duke
parents:
diff changeset
1960 SCAN_AND_ADJUST_POINTERS(adjust_obj_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1961 // See note about verification in prepare_for_compaction().
a61af66fc99e Initial load
duke
parents:
diff changeset
1962 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1963
a61af66fc99e Initial load
duke
parents:
diff changeset
1964 void CompactibleFreeListSpace::compact() {
a61af66fc99e Initial load
duke
parents:
diff changeset
1965 SCAN_AND_COMPACT(obj_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
1966 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1967
a61af66fc99e Initial load
duke
parents:
diff changeset
1968 // fragmentation_metric = 1 - [sum of (fbs**2) / (sum of fbs)**2]
a61af66fc99e Initial load
duke
parents:
diff changeset
1969 // where fbs is free block sizes
a61af66fc99e Initial load
duke
parents:
diff changeset
1970 double CompactibleFreeListSpace::flsFrag() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
1971 size_t itabFree = totalSizeInIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
1972 double frag = 0.0;
a61af66fc99e Initial load
duke
parents:
diff changeset
1973 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1974
a61af66fc99e Initial load
duke
parents:
diff changeset
1975 for (i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1976 double sz = i;
a61af66fc99e Initial load
duke
parents:
diff changeset
1977 frag += _indexedFreeList[i].count() * (sz * sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
1978 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1979
a61af66fc99e Initial load
duke
parents:
diff changeset
1980 double totFree = itabFree +
a61af66fc99e Initial load
duke
parents:
diff changeset
1981 _dictionary->totalChunkSize(DEBUG_ONLY(freelistLock()));
a61af66fc99e Initial load
duke
parents:
diff changeset
1982 if (totFree > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1983 frag = ((frag + _dictionary->sum_of_squared_block_sizes()) /
a61af66fc99e Initial load
duke
parents:
diff changeset
1984 (totFree * totFree));
a61af66fc99e Initial load
duke
parents:
diff changeset
1985 frag = (double)1.0 - frag;
a61af66fc99e Initial load
duke
parents:
diff changeset
1986 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
1987 assert(frag == 0.0, "Follows from totFree == 0");
a61af66fc99e Initial load
duke
parents:
diff changeset
1988 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1989 return frag;
a61af66fc99e Initial load
duke
parents:
diff changeset
1990 }
a61af66fc99e Initial load
duke
parents:
diff changeset
1991
a61af66fc99e Initial load
duke
parents:
diff changeset
1992 #define CoalSurplusPercent 1.05
a61af66fc99e Initial load
duke
parents:
diff changeset
1993 #define SplitSurplusPercent 1.10
a61af66fc99e Initial load
duke
parents:
diff changeset
1994
a61af66fc99e Initial load
duke
parents:
diff changeset
1995 void CompactibleFreeListSpace::beginSweepFLCensus(
a61af66fc99e Initial load
duke
parents:
diff changeset
1996 float inter_sweep_current,
a61af66fc99e Initial load
duke
parents:
diff changeset
1997 float inter_sweep_estimate) {
a61af66fc99e Initial load
duke
parents:
diff changeset
1998 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
1999 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
2000 for (i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2001 FreeList* fl = &_indexedFreeList[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
2002 fl->compute_desired(inter_sweep_current, inter_sweep_estimate);
a61af66fc99e Initial load
duke
parents:
diff changeset
2003 fl->set_coalDesired((ssize_t)((double)fl->desired() * CoalSurplusPercent));
a61af66fc99e Initial load
duke
parents:
diff changeset
2004 fl->set_beforeSweep(fl->count());
a61af66fc99e Initial load
duke
parents:
diff changeset
2005 fl->set_bfrSurp(fl->surplus());
a61af66fc99e Initial load
duke
parents:
diff changeset
2006 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2007 _dictionary->beginSweepDictCensus(CoalSurplusPercent,
a61af66fc99e Initial load
duke
parents:
diff changeset
2008 inter_sweep_current,
a61af66fc99e Initial load
duke
parents:
diff changeset
2009 inter_sweep_estimate);
a61af66fc99e Initial load
duke
parents:
diff changeset
2010 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2011
a61af66fc99e Initial load
duke
parents:
diff changeset
2012 void CompactibleFreeListSpace::setFLSurplus() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2013 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
2014 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
2015 for (i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2016 FreeList *fl = &_indexedFreeList[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
2017 fl->set_surplus(fl->count() -
a61af66fc99e Initial load
duke
parents:
diff changeset
2018 (ssize_t)((double)fl->desired() * SplitSurplusPercent));
a61af66fc99e Initial load
duke
parents:
diff changeset
2019 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2020 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2021
a61af66fc99e Initial load
duke
parents:
diff changeset
2022 void CompactibleFreeListSpace::setFLHints() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2023 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
2024 size_t i;
a61af66fc99e Initial load
duke
parents:
diff changeset
2025 size_t h = IndexSetSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
2026 for (i = IndexSetSize - 1; i != 0; i -= IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2027 FreeList *fl = &_indexedFreeList[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
2028 fl->set_hint(h);
a61af66fc99e Initial load
duke
parents:
diff changeset
2029 if (fl->surplus() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2030 h = i;
a61af66fc99e Initial load
duke
parents:
diff changeset
2031 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2032 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2033 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2034
a61af66fc99e Initial load
duke
parents:
diff changeset
2035 void CompactibleFreeListSpace::clearFLCensus() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2036 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
2037 int i;
a61af66fc99e Initial load
duke
parents:
diff changeset
2038 for (i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2039 FreeList *fl = &_indexedFreeList[i];
a61af66fc99e Initial load
duke
parents:
diff changeset
2040 fl->set_prevSweep(fl->count());
a61af66fc99e Initial load
duke
parents:
diff changeset
2041 fl->set_coalBirths(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
2042 fl->set_coalDeaths(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
2043 fl->set_splitBirths(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
2044 fl->set_splitDeaths(0);
a61af66fc99e Initial load
duke
parents:
diff changeset
2045 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2046 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2047
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2048 void CompactibleFreeListSpace::endSweepFLCensus(size_t sweep_count) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2049 setFLSurplus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2050 setFLHints();
a61af66fc99e Initial load
duke
parents:
diff changeset
2051 if (PrintGC && PrintFLSCensus > 0) {
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2052 printFLCensus(sweep_count);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2053 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2054 clearFLCensus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2055 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
2056 _dictionary->endSweepDictCensus(SplitSurplusPercent);
a61af66fc99e Initial load
duke
parents:
diff changeset
2057 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2058
a61af66fc99e Initial load
duke
parents:
diff changeset
2059 bool CompactibleFreeListSpace::coalOverPopulated(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2060 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2061 FreeList *fl = &_indexedFreeList[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
2062 return (fl->coalDesired() < 0) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2063 ((int)fl->count() > fl->coalDesired());
a61af66fc99e Initial load
duke
parents:
diff changeset
2064 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2065 return dictionary()->coalDictOverPopulated(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2066 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2067 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2068
a61af66fc99e Initial load
duke
parents:
diff changeset
2069 void CompactibleFreeListSpace::smallCoalBirth(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2070 assert(size < SmallForDictionary, "Size too large for indexed list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2071 FreeList *fl = &_indexedFreeList[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
2072 fl->increment_coalBirths();
a61af66fc99e Initial load
duke
parents:
diff changeset
2073 fl->increment_surplus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2074 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2075
a61af66fc99e Initial load
duke
parents:
diff changeset
2076 void CompactibleFreeListSpace::smallCoalDeath(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2077 assert(size < SmallForDictionary, "Size too large for indexed list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2078 FreeList *fl = &_indexedFreeList[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
2079 fl->increment_coalDeaths();
a61af66fc99e Initial load
duke
parents:
diff changeset
2080 fl->decrement_surplus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2081 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2082
a61af66fc99e Initial load
duke
parents:
diff changeset
2083 void CompactibleFreeListSpace::coalBirth(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2084 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2085 smallCoalBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2086 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2087 dictionary()->dictCensusUpdate(size,
a61af66fc99e Initial load
duke
parents:
diff changeset
2088 false /* split */,
a61af66fc99e Initial load
duke
parents:
diff changeset
2089 true /* birth */);
a61af66fc99e Initial load
duke
parents:
diff changeset
2090 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2091 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2092
a61af66fc99e Initial load
duke
parents:
diff changeset
2093 void CompactibleFreeListSpace::coalDeath(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2094 if(size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2095 smallCoalDeath(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2096 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2097 dictionary()->dictCensusUpdate(size,
a61af66fc99e Initial load
duke
parents:
diff changeset
2098 false /* split */,
a61af66fc99e Initial load
duke
parents:
diff changeset
2099 false /* birth */);
a61af66fc99e Initial load
duke
parents:
diff changeset
2100 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2101 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2102
a61af66fc99e Initial load
duke
parents:
diff changeset
2103 void CompactibleFreeListSpace::smallSplitBirth(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2104 assert(size < SmallForDictionary, "Size too large for indexed list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2105 FreeList *fl = &_indexedFreeList[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
2106 fl->increment_splitBirths();
a61af66fc99e Initial load
duke
parents:
diff changeset
2107 fl->increment_surplus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2108 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2109
a61af66fc99e Initial load
duke
parents:
diff changeset
2110 void CompactibleFreeListSpace::smallSplitDeath(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2111 assert(size < SmallForDictionary, "Size too large for indexed list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2112 FreeList *fl = &_indexedFreeList[size];
a61af66fc99e Initial load
duke
parents:
diff changeset
2113 fl->increment_splitDeaths();
a61af66fc99e Initial load
duke
parents:
diff changeset
2114 fl->decrement_surplus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2115 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2116
a61af66fc99e Initial load
duke
parents:
diff changeset
2117 void CompactibleFreeListSpace::splitBirth(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2118 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2119 smallSplitBirth(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2120 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2121 dictionary()->dictCensusUpdate(size,
a61af66fc99e Initial load
duke
parents:
diff changeset
2122 true /* split */,
a61af66fc99e Initial load
duke
parents:
diff changeset
2123 true /* birth */);
a61af66fc99e Initial load
duke
parents:
diff changeset
2124 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2125 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2126
a61af66fc99e Initial load
duke
parents:
diff changeset
2127 void CompactibleFreeListSpace::splitDeath(size_t size) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2128 if (size < SmallForDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2129 smallSplitDeath(size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2130 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2131 dictionary()->dictCensusUpdate(size,
a61af66fc99e Initial load
duke
parents:
diff changeset
2132 true /* split */,
a61af66fc99e Initial load
duke
parents:
diff changeset
2133 false /* birth */);
a61af66fc99e Initial load
duke
parents:
diff changeset
2134 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2135 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2136
a61af66fc99e Initial load
duke
parents:
diff changeset
2137 void CompactibleFreeListSpace::split(size_t from, size_t to1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2138 size_t to2 = from - to1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2139 splitDeath(from);
a61af66fc99e Initial load
duke
parents:
diff changeset
2140 splitBirth(to1);
a61af66fc99e Initial load
duke
parents:
diff changeset
2141 splitBirth(to2);
a61af66fc99e Initial load
duke
parents:
diff changeset
2142 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2143
a61af66fc99e Initial load
duke
parents:
diff changeset
2144 void CompactibleFreeListSpace::print() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2145 tty->print(" CompactibleFreeListSpace");
a61af66fc99e Initial load
duke
parents:
diff changeset
2146 Space::print();
a61af66fc99e Initial load
duke
parents:
diff changeset
2147 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2148
a61af66fc99e Initial load
duke
parents:
diff changeset
2149 void CompactibleFreeListSpace::prepare_for_verify() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2150 assert_locked();
a61af66fc99e Initial load
duke
parents:
diff changeset
2151 repairLinearAllocationBlocks();
a61af66fc99e Initial load
duke
parents:
diff changeset
2152 // Verify that the SpoolBlocks look like free blocks of
a61af66fc99e Initial load
duke
parents:
diff changeset
2153 // appropriate sizes... To be done ...
a61af66fc99e Initial load
duke
parents:
diff changeset
2154 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2155
a61af66fc99e Initial load
duke
parents:
diff changeset
2156 class VerifyAllBlksClosure: public BlkClosure {
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2157 private:
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2158 const CompactibleFreeListSpace* _sp;
a61af66fc99e Initial load
duke
parents:
diff changeset
2159 const MemRegion _span;
a61af66fc99e Initial load
duke
parents:
diff changeset
2160
a61af66fc99e Initial load
duke
parents:
diff changeset
2161 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
2162 VerifyAllBlksClosure(const CompactibleFreeListSpace* sp,
a61af66fc99e Initial load
duke
parents:
diff changeset
2163 MemRegion span) : _sp(sp), _span(span) { }
a61af66fc99e Initial load
duke
parents:
diff changeset
2164
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2165 virtual size_t do_blk(HeapWord* addr) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2166 size_t res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2167 if (_sp->block_is_obj(addr)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2168 oop p = oop(addr);
a61af66fc99e Initial load
duke
parents:
diff changeset
2169 guarantee(p->is_oop(), "Should be an oop");
a61af66fc99e Initial load
duke
parents:
diff changeset
2170 res = _sp->adjustObjectSize(p->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2171 if (_sp->obj_is_alive(addr)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2172 p->verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2173 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2174 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2175 FreeChunk* fc = (FreeChunk*)addr;
a61af66fc99e Initial load
duke
parents:
diff changeset
2176 res = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
2177 if (FLSVerifyLists && !fc->cantCoalesce()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2178 guarantee(_sp->verifyChunkInFreeLists(fc),
a61af66fc99e Initial load
duke
parents:
diff changeset
2179 "Chunk should be on a free list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2180 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2181 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2182 guarantee(res != 0, "Livelock: no rank reduction!");
a61af66fc99e Initial load
duke
parents:
diff changeset
2183 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2184 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2185 };
a61af66fc99e Initial load
duke
parents:
diff changeset
2186
a61af66fc99e Initial load
duke
parents:
diff changeset
2187 class VerifyAllOopsClosure: public OopClosure {
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2188 private:
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2189 const CMSCollector* _collector;
a61af66fc99e Initial load
duke
parents:
diff changeset
2190 const CompactibleFreeListSpace* _sp;
a61af66fc99e Initial load
duke
parents:
diff changeset
2191 const MemRegion _span;
a61af66fc99e Initial load
duke
parents:
diff changeset
2192 const bool _past_remark;
a61af66fc99e Initial load
duke
parents:
diff changeset
2193 const CMSBitMap* _bit_map;
a61af66fc99e Initial load
duke
parents:
diff changeset
2194
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2195 protected:
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2196 void do_oop(void* p, oop obj) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2197 if (_span.contains(obj)) { // the interior oop points into CMS heap
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2198 if (!_span.contains(p)) { // reference from outside CMS heap
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2199 // Should be a valid object; the first disjunct below allows
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2200 // us to sidestep an assertion in block_is_obj() that insists
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2201 // that p be in _sp. Note that several generations (and spaces)
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2202 // are spanned by _span (CMS heap) above.
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2203 guarantee(!_sp->is_in_reserved(obj) ||
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2204 _sp->block_is_obj((HeapWord*)obj),
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2205 "Should be an object");
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2206 guarantee(obj->is_oop(), "Should be an oop");
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2207 obj->verify();
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2208 if (_past_remark) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2209 // Remark has been completed, the object should be marked
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2210 _bit_map->isMarked((HeapWord*)obj);
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2211 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2212 } else { // reference within CMS heap
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2213 if (_past_remark) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2214 // Remark has been completed -- so the referent should have
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2215 // been marked, if referring object is.
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2216 if (_bit_map->isMarked(_collector->block_start(p))) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2217 guarantee(_bit_map->isMarked((HeapWord*)obj), "Marking error?");
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2218 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2219 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2220 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2221 } else if (_sp->is_in_reserved(p)) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2222 // the reference is from FLS, and points out of FLS
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2223 guarantee(obj->is_oop(), "Should be an oop");
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2224 obj->verify();
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2225 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2226 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2227
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2228 template <class T> void do_oop_work(T* p) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2229 T heap_oop = oopDesc::load_heap_oop(p);
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2230 if (!oopDesc::is_null(heap_oop)) {
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2231 oop obj = oopDesc::decode_heap_oop_not_null(heap_oop);
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2232 do_oop(p, obj);
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2233 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2234 }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2235
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2236 public:
a61af66fc99e Initial load
duke
parents:
diff changeset
2237 VerifyAllOopsClosure(const CMSCollector* collector,
a61af66fc99e Initial load
duke
parents:
diff changeset
2238 const CompactibleFreeListSpace* sp, MemRegion span,
a61af66fc99e Initial load
duke
parents:
diff changeset
2239 bool past_remark, CMSBitMap* bit_map) :
a61af66fc99e Initial load
duke
parents:
diff changeset
2240 OopClosure(), _collector(collector), _sp(sp), _span(span),
a61af66fc99e Initial load
duke
parents:
diff changeset
2241 _past_remark(past_remark), _bit_map(bit_map) { }
a61af66fc99e Initial load
duke
parents:
diff changeset
2242
113
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2243 virtual void do_oop(oop* p) { VerifyAllOopsClosure::do_oop_work(p); }
ba764ed4b6f2 6420645: Create a vm that uses compressed oops for up to 32gb heapsizes
coleenp
parents: 12
diff changeset
2244 virtual void do_oop(narrowOop* p) { VerifyAllOopsClosure::do_oop_work(p); }
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2245 };
a61af66fc99e Initial load
duke
parents:
diff changeset
2246
a61af66fc99e Initial load
duke
parents:
diff changeset
2247 void CompactibleFreeListSpace::verify(bool ignored) const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2248 assert_lock_strong(&_freelistLock);
a61af66fc99e Initial load
duke
parents:
diff changeset
2249 verify_objects_initialized();
a61af66fc99e Initial load
duke
parents:
diff changeset
2250 MemRegion span = _collector->_span;
a61af66fc99e Initial load
duke
parents:
diff changeset
2251 bool past_remark = (_collector->abstract_state() ==
a61af66fc99e Initial load
duke
parents:
diff changeset
2252 CMSCollector::Sweeping);
a61af66fc99e Initial load
duke
parents:
diff changeset
2253
a61af66fc99e Initial load
duke
parents:
diff changeset
2254 ResourceMark rm;
a61af66fc99e Initial load
duke
parents:
diff changeset
2255 HandleMark hm;
a61af66fc99e Initial load
duke
parents:
diff changeset
2256
a61af66fc99e Initial load
duke
parents:
diff changeset
2257 // Check integrity of CFL data structures
a61af66fc99e Initial load
duke
parents:
diff changeset
2258 _promoInfo.verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2259 _dictionary->verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2260 if (FLSVerifyIndexTable) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2261 verifyIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
2262 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2263 // Check integrity of all objects and free blocks in space
a61af66fc99e Initial load
duke
parents:
diff changeset
2264 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2265 VerifyAllBlksClosure cl(this, span);
a61af66fc99e Initial load
duke
parents:
diff changeset
2266 ((CompactibleFreeListSpace*)this)->blk_iterate(&cl); // cast off const
a61af66fc99e Initial load
duke
parents:
diff changeset
2267 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2268 // Check that all references in the heap to FLS
a61af66fc99e Initial load
duke
parents:
diff changeset
2269 // are to valid objects in FLS or that references in
a61af66fc99e Initial load
duke
parents:
diff changeset
2270 // FLS are to valid objects elsewhere in the heap
a61af66fc99e Initial load
duke
parents:
diff changeset
2271 if (FLSVerifyAllHeapReferences)
a61af66fc99e Initial load
duke
parents:
diff changeset
2272 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2273 VerifyAllOopsClosure cl(_collector, this, span, past_remark,
a61af66fc99e Initial load
duke
parents:
diff changeset
2274 _collector->markBitMap());
a61af66fc99e Initial load
duke
parents:
diff changeset
2275 CollectedHeap* ch = Universe::heap();
a61af66fc99e Initial load
duke
parents:
diff changeset
2276 ch->oop_iterate(&cl); // all oops in generations
a61af66fc99e Initial load
duke
parents:
diff changeset
2277 ch->permanent_oop_iterate(&cl); // all oops in perm gen
a61af66fc99e Initial load
duke
parents:
diff changeset
2278 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2279
a61af66fc99e Initial load
duke
parents:
diff changeset
2280 if (VerifyObjectStartArray) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2281 // Verify the block offset table
a61af66fc99e Initial load
duke
parents:
diff changeset
2282 _bt.verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2283 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2284 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2285
a61af66fc99e Initial load
duke
parents:
diff changeset
2286 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
2287 void CompactibleFreeListSpace::verifyFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2288 if (FLSVerifyLists) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2289 _dictionary->verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2290 verifyIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
2291 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2292 if (FLSVerifyDictionary) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2293 _dictionary->verify();
a61af66fc99e Initial load
duke
parents:
diff changeset
2294 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2295 if (FLSVerifyIndexTable) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2296 verifyIndexedFreeLists();
a61af66fc99e Initial load
duke
parents:
diff changeset
2297 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2298 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2299 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2300 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
2301
a61af66fc99e Initial load
duke
parents:
diff changeset
2302 void CompactibleFreeListSpace::verifyIndexedFreeLists() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2303 size_t i = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
2304 for (; i < MinChunkSize; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2305 guarantee(_indexedFreeList[i].head() == NULL, "should be NULL");
a61af66fc99e Initial load
duke
parents:
diff changeset
2306 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2307 for (; i < IndexSetSize; i++) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2308 verifyIndexedFreeList(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
2309 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2310 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2311
a61af66fc99e Initial load
duke
parents:
diff changeset
2312 void CompactibleFreeListSpace::verifyIndexedFreeList(size_t size) const {
342
37f87013dfd8 6711316: Open source the Garbage-First garbage collector
ysr
parents: 113
diff changeset
2313 FreeChunk* fc = _indexedFreeList[size].head();
37f87013dfd8 6711316: Open source the Garbage-First garbage collector
ysr
parents: 113
diff changeset
2314 guarantee((size % 2 == 0) || fc == NULL, "Odd slots should be empty");
37f87013dfd8 6711316: Open source the Garbage-First garbage collector
ysr
parents: 113
diff changeset
2315 for (; fc != NULL; fc = fc->next()) {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2316 guarantee(fc->size() == size, "Size inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2317 guarantee(fc->isFree(), "!free?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2318 guarantee(fc->next() == NULL || fc->next()->prev() == fc, "Broken list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2319 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2320 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2321
a61af66fc99e Initial load
duke
parents:
diff changeset
2322 #ifndef PRODUCT
a61af66fc99e Initial load
duke
parents:
diff changeset
2323 void CompactibleFreeListSpace::checkFreeListConsistency() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2324 assert(_dictionary->minSize() <= IndexSetSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
2325 "Some sizes can't be allocated without recourse to"
a61af66fc99e Initial load
duke
parents:
diff changeset
2326 " linear allocation buffers");
a61af66fc99e Initial load
duke
parents:
diff changeset
2327 assert(MIN_TREE_CHUNK_SIZE*HeapWordSize == sizeof(TreeChunk),
a61af66fc99e Initial load
duke
parents:
diff changeset
2328 "else MIN_TREE_CHUNK_SIZE is wrong");
a61af66fc99e Initial load
duke
parents:
diff changeset
2329 assert((IndexSetStride == 2 && IndexSetStart == 2) ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2330 (IndexSetStride == 1 && IndexSetStart == 1), "just checking");
a61af66fc99e Initial load
duke
parents:
diff changeset
2331 assert((IndexSetStride != 2) || (MinChunkSize % 2 == 0),
a61af66fc99e Initial load
duke
parents:
diff changeset
2332 "Some for-loops may be incorrectly initialized");
a61af66fc99e Initial load
duke
parents:
diff changeset
2333 assert((IndexSetStride != 2) || (IndexSetSize % 2 == 1),
a61af66fc99e Initial load
duke
parents:
diff changeset
2334 "For-loops that iterate over IndexSet with stride 2 may be wrong");
a61af66fc99e Initial load
duke
parents:
diff changeset
2335 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2336 #endif
a61af66fc99e Initial load
duke
parents:
diff changeset
2337
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2338 void CompactibleFreeListSpace::printFLCensus(size_t sweep_count) const {
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2339 assert_lock_strong(&_freelistLock);
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2340 FreeList total;
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2341 gclog_or_tty->print("end sweep# " SIZE_FORMAT "\n", sweep_count);
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2342 FreeList::print_labels_on(gclog_or_tty, "size");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2343 size_t totalFree = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
2344 for (size_t i = IndexSetStart; i < IndexSetSize; i += IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2345 const FreeList *fl = &_indexedFreeList[i];
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2346 totalFree += fl->count() * fl->size();
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2347 if (i % (40*IndexSetStride) == 0) {
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2348 FreeList::print_labels_on(gclog_or_tty, "size");
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2349 }
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2350 fl->print_on(gclog_or_tty);
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2351 total.set_bfrSurp( total.bfrSurp() + fl->bfrSurp() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2352 total.set_surplus( total.surplus() + fl->surplus() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2353 total.set_desired( total.desired() + fl->desired() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2354 total.set_prevSweep( total.prevSweep() + fl->prevSweep() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2355 total.set_beforeSweep(total.beforeSweep() + fl->beforeSweep());
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2356 total.set_count( total.count() + fl->count() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2357 total.set_coalBirths( total.coalBirths() + fl->coalBirths() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2358 total.set_coalDeaths( total.coalDeaths() + fl->coalDeaths() );
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2359 total.set_splitBirths(total.splitBirths() + fl->splitBirths());
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2360 total.set_splitDeaths(total.splitDeaths() + fl->splitDeaths());
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2361 }
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2362 total.print_on(gclog_or_tty, "TOTAL");
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2363 gclog_or_tty->print_cr("Total free in indexed lists "
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2364 SIZE_FORMAT " words", totalFree);
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2365 gclog_or_tty->print("growth: %8.5f deficit: %8.5f\n",
12
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2366 (double)(total.splitBirths()+total.coalBirths()-total.splitDeaths()-total.coalDeaths())/
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2367 (total.prevSweep() != 0 ? (double)total.prevSweep() : 1.0),
6432c3bb6240 6668743: CMS: Consolidate block statistics reporting code
ysr
parents: 0
diff changeset
2368 (double)(total.desired() - total.count())/(total.desired() != 0 ? (double)total.desired() : 1.0));
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2369 _dictionary->printDictCensus();
a61af66fc99e Initial load
duke
parents:
diff changeset
2370 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2371
a61af66fc99e Initial load
duke
parents:
diff changeset
2372 // Return the next displaced header, incrementing the pointer and
a61af66fc99e Initial load
duke
parents:
diff changeset
2373 // recycling spool area as necessary.
a61af66fc99e Initial load
duke
parents:
diff changeset
2374 markOop PromotionInfo::nextDisplacedHeader() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2375 assert(_spoolHead != NULL, "promotionInfo inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2376 assert(_spoolHead != _spoolTail || _firstIndex < _nextIndex,
a61af66fc99e Initial load
duke
parents:
diff changeset
2377 "Empty spool space: no displaced header can be fetched");
a61af66fc99e Initial load
duke
parents:
diff changeset
2378 assert(_spoolHead->bufferSize > _firstIndex, "Off by one error at head?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2379 markOop hdr = _spoolHead->displacedHdr[_firstIndex];
a61af66fc99e Initial load
duke
parents:
diff changeset
2380 // Spool forward
a61af66fc99e Initial load
duke
parents:
diff changeset
2381 if (++_firstIndex == _spoolHead->bufferSize) { // last location in this block
a61af66fc99e Initial load
duke
parents:
diff changeset
2382 // forward to next block, recycling this block into spare spool buffer
a61af66fc99e Initial load
duke
parents:
diff changeset
2383 SpoolBlock* tmp = _spoolHead->nextSpoolBlock;
a61af66fc99e Initial load
duke
parents:
diff changeset
2384 assert(_spoolHead != _spoolTail, "Spooling storage mix-up");
a61af66fc99e Initial load
duke
parents:
diff changeset
2385 _spoolHead->nextSpoolBlock = _spareSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2386 _spareSpool = _spoolHead;
a61af66fc99e Initial load
duke
parents:
diff changeset
2387 _spoolHead = tmp;
a61af66fc99e Initial load
duke
parents:
diff changeset
2388 _firstIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2389 NOT_PRODUCT(
a61af66fc99e Initial load
duke
parents:
diff changeset
2390 if (_spoolHead == NULL) { // all buffers fully consumed
a61af66fc99e Initial load
duke
parents:
diff changeset
2391 assert(_spoolTail == NULL && _nextIndex == 1,
a61af66fc99e Initial load
duke
parents:
diff changeset
2392 "spool buffers processing inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2393 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2394 )
a61af66fc99e Initial load
duke
parents:
diff changeset
2395 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2396 return hdr;
a61af66fc99e Initial load
duke
parents:
diff changeset
2397 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2398
a61af66fc99e Initial load
duke
parents:
diff changeset
2399 void PromotionInfo::track(PromotedObject* trackOop) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2400 track(trackOop, oop(trackOop)->klass());
a61af66fc99e Initial load
duke
parents:
diff changeset
2401 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2402
a61af66fc99e Initial load
duke
parents:
diff changeset
2403 void PromotionInfo::track(PromotedObject* trackOop, klassOop klassOfOop) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2404 // make a copy of header as it may need to be spooled
a61af66fc99e Initial load
duke
parents:
diff changeset
2405 markOop mark = oop(trackOop)->mark();
a61af66fc99e Initial load
duke
parents:
diff changeset
2406 trackOop->clearNext();
a61af66fc99e Initial load
duke
parents:
diff changeset
2407 if (mark->must_be_preserved_for_cms_scavenge(klassOfOop)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2408 // save non-prototypical header, and mark oop
a61af66fc99e Initial load
duke
parents:
diff changeset
2409 saveDisplacedHeader(mark);
a61af66fc99e Initial load
duke
parents:
diff changeset
2410 trackOop->setDisplacedMark();
a61af66fc99e Initial load
duke
parents:
diff changeset
2411 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2412 // we'd like to assert something like the following:
a61af66fc99e Initial load
duke
parents:
diff changeset
2413 // assert(mark == markOopDesc::prototype(), "consistency check");
a61af66fc99e Initial load
duke
parents:
diff changeset
2414 // ... but the above won't work because the age bits have not (yet) been
a61af66fc99e Initial load
duke
parents:
diff changeset
2415 // cleared. The remainder of the check would be identical to the
a61af66fc99e Initial load
duke
parents:
diff changeset
2416 // condition checked in must_be_preserved() above, so we don't really
a61af66fc99e Initial load
duke
parents:
diff changeset
2417 // have anything useful to check here!
a61af66fc99e Initial load
duke
parents:
diff changeset
2418 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2419 if (_promoTail != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2420 assert(_promoHead != NULL, "List consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2421 _promoTail->setNext(trackOop);
a61af66fc99e Initial load
duke
parents:
diff changeset
2422 _promoTail = trackOop;
a61af66fc99e Initial load
duke
parents:
diff changeset
2423 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2424 assert(_promoHead == NULL, "List consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2425 _promoHead = _promoTail = trackOop;
a61af66fc99e Initial load
duke
parents:
diff changeset
2426 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2427 // Mask as newly promoted, so we can skip over such objects
a61af66fc99e Initial load
duke
parents:
diff changeset
2428 // when scanning dirty cards
a61af66fc99e Initial load
duke
parents:
diff changeset
2429 assert(!trackOop->hasPromotedMark(), "Should not have been marked");
a61af66fc99e Initial load
duke
parents:
diff changeset
2430 trackOop->setPromotedMark();
a61af66fc99e Initial load
duke
parents:
diff changeset
2431 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2432
a61af66fc99e Initial load
duke
parents:
diff changeset
2433 // Save the given displaced header, incrementing the pointer and
a61af66fc99e Initial load
duke
parents:
diff changeset
2434 // obtaining more spool area as necessary.
a61af66fc99e Initial load
duke
parents:
diff changeset
2435 void PromotionInfo::saveDisplacedHeader(markOop hdr) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2436 assert(_spoolHead != NULL && _spoolTail != NULL,
a61af66fc99e Initial load
duke
parents:
diff changeset
2437 "promotionInfo inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2438 assert(_spoolTail->bufferSize > _nextIndex, "Off by one error at tail?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2439 _spoolTail->displacedHdr[_nextIndex] = hdr;
a61af66fc99e Initial load
duke
parents:
diff changeset
2440 // Spool forward
a61af66fc99e Initial load
duke
parents:
diff changeset
2441 if (++_nextIndex == _spoolTail->bufferSize) { // last location in this block
a61af66fc99e Initial load
duke
parents:
diff changeset
2442 // get a new spooling block
a61af66fc99e Initial load
duke
parents:
diff changeset
2443 assert(_spoolTail->nextSpoolBlock == NULL, "tail should terminate spool list");
a61af66fc99e Initial load
duke
parents:
diff changeset
2444 _splice_point = _spoolTail; // save for splicing
a61af66fc99e Initial load
duke
parents:
diff changeset
2445 _spoolTail->nextSpoolBlock = getSpoolBlock(); // might fail
a61af66fc99e Initial load
duke
parents:
diff changeset
2446 _spoolTail = _spoolTail->nextSpoolBlock; // might become NULL ...
a61af66fc99e Initial load
duke
parents:
diff changeset
2447 // ... but will attempt filling before next promotion attempt
a61af66fc99e Initial load
duke
parents:
diff changeset
2448 _nextIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2449 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2450 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2451
a61af66fc99e Initial load
duke
parents:
diff changeset
2452 // Ensure that spooling space exists. Return false if spooling space
a61af66fc99e Initial load
duke
parents:
diff changeset
2453 // could not be obtained.
a61af66fc99e Initial load
duke
parents:
diff changeset
2454 bool PromotionInfo::ensure_spooling_space_work() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2455 assert(!has_spooling_space(), "Only call when there is no spooling space");
a61af66fc99e Initial load
duke
parents:
diff changeset
2456 // Try and obtain more spooling space
a61af66fc99e Initial load
duke
parents:
diff changeset
2457 SpoolBlock* newSpool = getSpoolBlock();
a61af66fc99e Initial load
duke
parents:
diff changeset
2458 assert(newSpool == NULL ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2459 (newSpool->bufferSize != 0 && newSpool->nextSpoolBlock == NULL),
a61af66fc99e Initial load
duke
parents:
diff changeset
2460 "getSpoolBlock() sanity check");
a61af66fc99e Initial load
duke
parents:
diff changeset
2461 if (newSpool == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2462 return false;
a61af66fc99e Initial load
duke
parents:
diff changeset
2463 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2464 _nextIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2465 if (_spoolTail == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2466 _spoolTail = newSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2467 if (_spoolHead == NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2468 _spoolHead = newSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2469 _firstIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2470 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2471 assert(_splice_point != NULL && _splice_point->nextSpoolBlock == NULL,
a61af66fc99e Initial load
duke
parents:
diff changeset
2472 "Splice point invariant");
a61af66fc99e Initial load
duke
parents:
diff changeset
2473 // Extra check that _splice_point is connected to list
a61af66fc99e Initial load
duke
parents:
diff changeset
2474 #ifdef ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
2475 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2476 SpoolBlock* blk = _spoolHead;
a61af66fc99e Initial load
duke
parents:
diff changeset
2477 for (; blk->nextSpoolBlock != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2478 blk = blk->nextSpoolBlock);
a61af66fc99e Initial load
duke
parents:
diff changeset
2479 assert(blk != NULL && blk == _splice_point,
a61af66fc99e Initial load
duke
parents:
diff changeset
2480 "Splice point incorrect");
a61af66fc99e Initial load
duke
parents:
diff changeset
2481 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2482 #endif // ASSERT
a61af66fc99e Initial load
duke
parents:
diff changeset
2483 _splice_point->nextSpoolBlock = newSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2484 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2485 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2486 assert(_spoolHead != NULL, "spool list consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2487 _spoolTail->nextSpoolBlock = newSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2488 _spoolTail = newSpool;
a61af66fc99e Initial load
duke
parents:
diff changeset
2489 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2490 return true;
a61af66fc99e Initial load
duke
parents:
diff changeset
2491 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2492
a61af66fc99e Initial load
duke
parents:
diff changeset
2493 // Get a free spool buffer from the free pool, getting a new block
a61af66fc99e Initial load
duke
parents:
diff changeset
2494 // from the heap if necessary.
a61af66fc99e Initial load
duke
parents:
diff changeset
2495 SpoolBlock* PromotionInfo::getSpoolBlock() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2496 SpoolBlock* res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2497 if ((res = _spareSpool) != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2498 _spareSpool = _spareSpool->nextSpoolBlock;
a61af66fc99e Initial load
duke
parents:
diff changeset
2499 res->nextSpoolBlock = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2500 } else { // spare spool exhausted, get some from heap
a61af66fc99e Initial load
duke
parents:
diff changeset
2501 res = (SpoolBlock*)(space()->allocateScratch(refillSize()));
a61af66fc99e Initial load
duke
parents:
diff changeset
2502 if (res != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2503 res->init();
a61af66fc99e Initial load
duke
parents:
diff changeset
2504 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2505 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2506 assert(res == NULL || res->nextSpoolBlock == NULL, "postcondition");
a61af66fc99e Initial load
duke
parents:
diff changeset
2507 return res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2508 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2509
a61af66fc99e Initial load
duke
parents:
diff changeset
2510 void PromotionInfo::startTrackingPromotions() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2511 assert(_spoolHead == _spoolTail && _firstIndex == _nextIndex,
a61af66fc99e Initial load
duke
parents:
diff changeset
2512 "spooling inconsistency?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2513 _firstIndex = _nextIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2514 _tracking = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
2515 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2516
a61af66fc99e Initial load
duke
parents:
diff changeset
2517 void PromotionInfo::stopTrackingPromotions() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2518 assert(_spoolHead == _spoolTail && _firstIndex == _nextIndex,
a61af66fc99e Initial load
duke
parents:
diff changeset
2519 "spooling inconsistency?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2520 _firstIndex = _nextIndex = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2521 _tracking = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
2522 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2523
a61af66fc99e Initial load
duke
parents:
diff changeset
2524 // When _spoolTail is not NULL, then the slot <_spoolTail, _nextIndex>
a61af66fc99e Initial load
duke
parents:
diff changeset
2525 // points to the next slot available for filling.
a61af66fc99e Initial load
duke
parents:
diff changeset
2526 // The set of slots holding displaced headers are then all those in the
a61af66fc99e Initial load
duke
parents:
diff changeset
2527 // right-open interval denoted by:
a61af66fc99e Initial load
duke
parents:
diff changeset
2528 //
a61af66fc99e Initial load
duke
parents:
diff changeset
2529 // [ <_spoolHead, _firstIndex>, <_spoolTail, _nextIndex> )
a61af66fc99e Initial load
duke
parents:
diff changeset
2530 //
a61af66fc99e Initial load
duke
parents:
diff changeset
2531 // When _spoolTail is NULL, then the set of slots with displaced headers
a61af66fc99e Initial load
duke
parents:
diff changeset
2532 // is all those starting at the slot <_spoolHead, _firstIndex> and
a61af66fc99e Initial load
duke
parents:
diff changeset
2533 // going up to the last slot of last block in the linked list.
a61af66fc99e Initial load
duke
parents:
diff changeset
2534 // In this lartter case, _splice_point points to the tail block of
a61af66fc99e Initial load
duke
parents:
diff changeset
2535 // this linked list of blocks holding displaced headers.
a61af66fc99e Initial load
duke
parents:
diff changeset
2536 void PromotionInfo::verify() const {
a61af66fc99e Initial load
duke
parents:
diff changeset
2537 // Verify the following:
a61af66fc99e Initial load
duke
parents:
diff changeset
2538 // 1. the number of displaced headers matches the number of promoted
a61af66fc99e Initial load
duke
parents:
diff changeset
2539 // objects that have displaced headers
a61af66fc99e Initial load
duke
parents:
diff changeset
2540 // 2. each promoted object lies in this space
a61af66fc99e Initial load
duke
parents:
diff changeset
2541 debug_only(
a61af66fc99e Initial load
duke
parents:
diff changeset
2542 PromotedObject* junk = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2543 assert(junk->next_addr() == (void*)(oop(junk)->mark_addr()),
a61af66fc99e Initial load
duke
parents:
diff changeset
2544 "Offset of PromotedObject::_next is expected to align with "
a61af66fc99e Initial load
duke
parents:
diff changeset
2545 " the OopDesc::_mark within OopDesc");
a61af66fc99e Initial load
duke
parents:
diff changeset
2546 )
a61af66fc99e Initial load
duke
parents:
diff changeset
2547 // FIXME: guarantee????
a61af66fc99e Initial load
duke
parents:
diff changeset
2548 guarantee(_spoolHead == NULL || _spoolTail != NULL ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2549 _splice_point != NULL, "list consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2550 guarantee(_promoHead == NULL || _promoTail != NULL, "list consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2551 // count the number of objects with displaced headers
a61af66fc99e Initial load
duke
parents:
diff changeset
2552 size_t numObjsWithDisplacedHdrs = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
2553 for (PromotedObject* curObj = _promoHead; curObj != NULL; curObj = curObj->next()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2554 guarantee(space()->is_in_reserved((HeapWord*)curObj), "Containment");
a61af66fc99e Initial load
duke
parents:
diff changeset
2555 // the last promoted object may fail the mark() != NULL test of is_oop().
a61af66fc99e Initial load
duke
parents:
diff changeset
2556 guarantee(curObj->next() == NULL || oop(curObj)->is_oop(), "must be an oop");
a61af66fc99e Initial load
duke
parents:
diff changeset
2557 if (curObj->hasDisplacedMark()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2558 numObjsWithDisplacedHdrs++;
a61af66fc99e Initial load
duke
parents:
diff changeset
2559 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2560 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2561 // Count the number of displaced headers
a61af66fc99e Initial load
duke
parents:
diff changeset
2562 size_t numDisplacedHdrs = 0;
a61af66fc99e Initial load
duke
parents:
diff changeset
2563 for (SpoolBlock* curSpool = _spoolHead;
a61af66fc99e Initial load
duke
parents:
diff changeset
2564 curSpool != _spoolTail && curSpool != NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2565 curSpool = curSpool->nextSpoolBlock) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2566 // the first entry is just a self-pointer; indices 1 through
a61af66fc99e Initial load
duke
parents:
diff changeset
2567 // bufferSize - 1 are occupied (thus, bufferSize - 1 slots).
a61af66fc99e Initial load
duke
parents:
diff changeset
2568 guarantee((void*)curSpool->displacedHdr == (void*)&curSpool->displacedHdr,
a61af66fc99e Initial load
duke
parents:
diff changeset
2569 "first entry of displacedHdr should be self-referential");
a61af66fc99e Initial load
duke
parents:
diff changeset
2570 numDisplacedHdrs += curSpool->bufferSize - 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2571 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2572 guarantee((_spoolHead == _spoolTail) == (numDisplacedHdrs == 0),
a61af66fc99e Initial load
duke
parents:
diff changeset
2573 "internal consistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2574 guarantee(_spoolTail != NULL || _nextIndex == 1,
a61af66fc99e Initial load
duke
parents:
diff changeset
2575 "Inconsistency between _spoolTail and _nextIndex");
a61af66fc99e Initial load
duke
parents:
diff changeset
2576 // We overcounted (_firstIndex-1) worth of slots in block
a61af66fc99e Initial load
duke
parents:
diff changeset
2577 // _spoolHead and we undercounted (_nextIndex-1) worth of
a61af66fc99e Initial load
duke
parents:
diff changeset
2578 // slots in block _spoolTail. We make an appropriate
a61af66fc99e Initial load
duke
parents:
diff changeset
2579 // adjustment by subtracting the first and adding the
a61af66fc99e Initial load
duke
parents:
diff changeset
2580 // second: - (_firstIndex - 1) + (_nextIndex - 1)
a61af66fc99e Initial load
duke
parents:
diff changeset
2581 numDisplacedHdrs += (_nextIndex - _firstIndex);
a61af66fc99e Initial load
duke
parents:
diff changeset
2582 guarantee(numDisplacedHdrs == numObjsWithDisplacedHdrs, "Displaced hdr count");
a61af66fc99e Initial load
duke
parents:
diff changeset
2583 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2584
a61af66fc99e Initial load
duke
parents:
diff changeset
2585
a61af66fc99e Initial load
duke
parents:
diff changeset
2586 CFLS_LAB::CFLS_LAB(CompactibleFreeListSpace* cfls) :
a61af66fc99e Initial load
duke
parents:
diff changeset
2587 _cfls(cfls)
a61af66fc99e Initial load
duke
parents:
diff changeset
2588 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2589 _blocks_to_claim = CMSParPromoteBlocksToClaim;
a61af66fc99e Initial load
duke
parents:
diff changeset
2590 for (size_t i = CompactibleFreeListSpace::IndexSetStart;
a61af66fc99e Initial load
duke
parents:
diff changeset
2591 i < CompactibleFreeListSpace::IndexSetSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
2592 i += CompactibleFreeListSpace::IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2593 _indexedFreeList[i].set_size(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
2594 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2595 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2596
a61af66fc99e Initial load
duke
parents:
diff changeset
2597 HeapWord* CFLS_LAB::alloc(size_t word_sz) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2598 FreeChunk* res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2599 word_sz = _cfls->adjustObjectSize(word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2600 if (word_sz >= CompactibleFreeListSpace::IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2601 // This locking manages sync with other large object allocations.
a61af66fc99e Initial load
duke
parents:
diff changeset
2602 MutexLockerEx x(_cfls->parDictionaryAllocLock(),
a61af66fc99e Initial load
duke
parents:
diff changeset
2603 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2604 res = _cfls->getChunkFromDictionaryExact(word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2605 if (res == NULL) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2606 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2607 FreeList* fl = &_indexedFreeList[word_sz];
a61af66fc99e Initial load
duke
parents:
diff changeset
2608 bool filled = false; //TRAP
a61af66fc99e Initial load
duke
parents:
diff changeset
2609 if (fl->count() == 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2610 bool filled = true; //TRAP
a61af66fc99e Initial load
duke
parents:
diff changeset
2611 // Attempt to refill this local free list.
a61af66fc99e Initial load
duke
parents:
diff changeset
2612 _cfls->par_get_chunk_of_blocks(word_sz, _blocks_to_claim, fl);
a61af66fc99e Initial load
duke
parents:
diff changeset
2613 // If it didn't work, give up.
a61af66fc99e Initial load
duke
parents:
diff changeset
2614 if (fl->count() == 0) return NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2615 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2616 res = fl->getChunkAtHead();
a61af66fc99e Initial load
duke
parents:
diff changeset
2617 assert(res != NULL, "Why was count non-zero?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2618 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2619 res->markNotFree();
a61af66fc99e Initial load
duke
parents:
diff changeset
2620 assert(!res->isFree(), "shouldn't be marked free");
187
790e66e5fbac 6687581: Make CMS work with compressed oops
coleenp
parents: 113
diff changeset
2621 assert(oop(res)->klass_or_null() == NULL, "should look uninitialized");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2622 // mangle a just allocated object with a distinct pattern.
a61af66fc99e Initial load
duke
parents:
diff changeset
2623 debug_only(res->mangleAllocated(word_sz));
a61af66fc99e Initial load
duke
parents:
diff changeset
2624 return (HeapWord*)res;
a61af66fc99e Initial load
duke
parents:
diff changeset
2625 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2626
a61af66fc99e Initial load
duke
parents:
diff changeset
2627 void CFLS_LAB::retire() {
a61af66fc99e Initial load
duke
parents:
diff changeset
2628 for (size_t i = CompactibleFreeListSpace::IndexSetStart;
a61af66fc99e Initial load
duke
parents:
diff changeset
2629 i < CompactibleFreeListSpace::IndexSetSize;
a61af66fc99e Initial load
duke
parents:
diff changeset
2630 i += CompactibleFreeListSpace::IndexSetStride) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2631 if (_indexedFreeList[i].count() > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2632 MutexLockerEx x(_cfls->_indexedFreeListParLocks[i],
a61af66fc99e Initial load
duke
parents:
diff changeset
2633 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2634 _cfls->_indexedFreeList[i].prepend(&_indexedFreeList[i]);
a61af66fc99e Initial load
duke
parents:
diff changeset
2635 // Reset this list.
a61af66fc99e Initial load
duke
parents:
diff changeset
2636 _indexedFreeList[i] = FreeList();
a61af66fc99e Initial load
duke
parents:
diff changeset
2637 _indexedFreeList[i].set_size(i);
a61af66fc99e Initial load
duke
parents:
diff changeset
2638 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2639 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2640 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2641
a61af66fc99e Initial load
duke
parents:
diff changeset
2642 void
a61af66fc99e Initial load
duke
parents:
diff changeset
2643 CompactibleFreeListSpace::
a61af66fc99e Initial load
duke
parents:
diff changeset
2644 par_get_chunk_of_blocks(size_t word_sz, size_t n, FreeList* fl) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2645 assert(fl->count() == 0, "Precondition.");
a61af66fc99e Initial load
duke
parents:
diff changeset
2646 assert(word_sz < CompactibleFreeListSpace::IndexSetSize,
a61af66fc99e Initial load
duke
parents:
diff changeset
2647 "Precondition");
a61af66fc99e Initial load
duke
parents:
diff changeset
2648
a61af66fc99e Initial load
duke
parents:
diff changeset
2649 // We'll try all multiples of word_sz in the indexed set (starting with
a61af66fc99e Initial load
duke
parents:
diff changeset
2650 // word_sz itself), then try getting a big chunk and splitting it.
a61af66fc99e Initial load
duke
parents:
diff changeset
2651 int k = 1;
a61af66fc99e Initial load
duke
parents:
diff changeset
2652 size_t cur_sz = k * word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2653 bool found = false;
a61af66fc99e Initial load
duke
parents:
diff changeset
2654 while (cur_sz < CompactibleFreeListSpace::IndexSetSize && k == 1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2655 FreeList* gfl = &_indexedFreeList[cur_sz];
a61af66fc99e Initial load
duke
parents:
diff changeset
2656 FreeList fl_for_cur_sz; // Empty.
a61af66fc99e Initial load
duke
parents:
diff changeset
2657 fl_for_cur_sz.set_size(cur_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2658 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2659 MutexLockerEx x(_indexedFreeListParLocks[cur_sz],
a61af66fc99e Initial load
duke
parents:
diff changeset
2660 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2661 if (gfl->count() != 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2662 size_t nn = MAX2(n/k, (size_t)1);
a61af66fc99e Initial load
duke
parents:
diff changeset
2663 gfl->getFirstNChunksFromList(nn, &fl_for_cur_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2664 found = true;
a61af66fc99e Initial load
duke
parents:
diff changeset
2665 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2666 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2667 // Now transfer fl_for_cur_sz to fl. Common case, we hope, is k = 1.
a61af66fc99e Initial load
duke
parents:
diff changeset
2668 if (found) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2669 if (k == 1) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2670 fl->prepend(&fl_for_cur_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2671 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2672 // Divide each block on fl_for_cur_sz up k ways.
a61af66fc99e Initial load
duke
parents:
diff changeset
2673 FreeChunk* fc;
a61af66fc99e Initial load
duke
parents:
diff changeset
2674 while ((fc = fl_for_cur_sz.getChunkAtHead()) != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2675 // Must do this in reverse order, so that anybody attempting to
a61af66fc99e Initial load
duke
parents:
diff changeset
2676 // access the main chunk sees it as a single free block until we
a61af66fc99e Initial load
duke
parents:
diff changeset
2677 // change it.
a61af66fc99e Initial load
duke
parents:
diff changeset
2678 size_t fc_size = fc->size();
a61af66fc99e Initial load
duke
parents:
diff changeset
2679 for (int i = k-1; i >= 0; i--) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2680 FreeChunk* ffc = (FreeChunk*)((HeapWord*)fc + i * word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2681 ffc->setSize(word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2682 ffc->linkNext(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
2683 ffc->linkPrev(NULL); // Mark as a free block for other (parallel) GC threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
2684 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
2685 // splitting from the right, fc_size == (k - i + 1) * wordsize
a61af66fc99e Initial load
duke
parents:
diff changeset
2686 _bt.mark_block((HeapWord*)ffc, word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2687 fc_size -= word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2688 _bt.verify_not_unallocated((HeapWord*)ffc, ffc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2689 _bt.verify_single_block((HeapWord*)fc, fc_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2690 _bt.verify_single_block((HeapWord*)ffc, ffc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2691 // Push this on "fl".
a61af66fc99e Initial load
duke
parents:
diff changeset
2692 fl->returnChunkAtHead(ffc);
a61af66fc99e Initial load
duke
parents:
diff changeset
2693 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2694 // TRAP
a61af66fc99e Initial load
duke
parents:
diff changeset
2695 assert(fl->tail()->next() == NULL, "List invariant.");
a61af66fc99e Initial load
duke
parents:
diff changeset
2696 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2697 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2698 return;
a61af66fc99e Initial load
duke
parents:
diff changeset
2699 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2700 k++; cur_sz = k * word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2701 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2702 // Otherwise, we'll split a block from the dictionary.
a61af66fc99e Initial load
duke
parents:
diff changeset
2703 FreeChunk* fc = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2704 FreeChunk* rem_fc = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2705 size_t rem;
a61af66fc99e Initial load
duke
parents:
diff changeset
2706 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2707 MutexLockerEx x(parDictionaryAllocLock(),
a61af66fc99e Initial load
duke
parents:
diff changeset
2708 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2709 while (n > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2710 fc = dictionary()->getChunk(MAX2(n * word_sz,
a61af66fc99e Initial load
duke
parents:
diff changeset
2711 _dictionary->minSize()),
a61af66fc99e Initial load
duke
parents:
diff changeset
2712 FreeBlockDictionary::atLeast);
a61af66fc99e Initial load
duke
parents:
diff changeset
2713 if (fc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2714 _bt.allocated((HeapWord*)fc, fc->size()); // update _unallocated_blk
a61af66fc99e Initial load
duke
parents:
diff changeset
2715 dictionary()->dictCensusUpdate(fc->size(),
a61af66fc99e Initial load
duke
parents:
diff changeset
2716 true /*split*/,
a61af66fc99e Initial load
duke
parents:
diff changeset
2717 false /*birth*/);
a61af66fc99e Initial load
duke
parents:
diff changeset
2718 break;
a61af66fc99e Initial load
duke
parents:
diff changeset
2719 } else {
a61af66fc99e Initial load
duke
parents:
diff changeset
2720 n--;
a61af66fc99e Initial load
duke
parents:
diff changeset
2721 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2722 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2723 if (fc == NULL) return;
a61af66fc99e Initial load
duke
parents:
diff changeset
2724 // Otherwise, split up that block.
a61af66fc99e Initial load
duke
parents:
diff changeset
2725 size_t nn = fc->size() / word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2726 n = MIN2(nn, n);
a61af66fc99e Initial load
duke
parents:
diff changeset
2727 rem = fc->size() - n * word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2728 // If there is a remainder, and it's too small, allocate one fewer.
a61af66fc99e Initial load
duke
parents:
diff changeset
2729 if (rem > 0 && rem < MinChunkSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2730 n--; rem += word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2731 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2732 // First return the remainder, if any.
a61af66fc99e Initial load
duke
parents:
diff changeset
2733 // Note that we hold the lock until we decide if we're going to give
a61af66fc99e Initial load
duke
parents:
diff changeset
2734 // back the remainder to the dictionary, since a contending allocator
a61af66fc99e Initial load
duke
parents:
diff changeset
2735 // may otherwise see the heap as empty. (We're willing to take that
a61af66fc99e Initial load
duke
parents:
diff changeset
2736 // hit if the block is a small block.)
a61af66fc99e Initial load
duke
parents:
diff changeset
2737 if (rem > 0) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2738 size_t prefix_size = n * word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2739 rem_fc = (FreeChunk*)((HeapWord*)fc + prefix_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2740 rem_fc->setSize(rem);
a61af66fc99e Initial load
duke
parents:
diff changeset
2741 rem_fc->linkNext(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
2742 rem_fc->linkPrev(NULL); // Mark as a free block for other (parallel) GC threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
2743 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
2744 _bt.split_block((HeapWord*)fc, fc->size(), prefix_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2745 if (rem >= IndexSetSize) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2746 returnChunkToDictionary(rem_fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
2747 dictionary()->dictCensusUpdate(fc->size(),
a61af66fc99e Initial load
duke
parents:
diff changeset
2748 true /*split*/,
a61af66fc99e Initial load
duke
parents:
diff changeset
2749 true /*birth*/);
a61af66fc99e Initial load
duke
parents:
diff changeset
2750 rem_fc = NULL;
a61af66fc99e Initial load
duke
parents:
diff changeset
2751 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2752 // Otherwise, return it to the small list below.
a61af66fc99e Initial load
duke
parents:
diff changeset
2753 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2754 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2755 //
a61af66fc99e Initial load
duke
parents:
diff changeset
2756 if (rem_fc != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2757 MutexLockerEx x(_indexedFreeListParLocks[rem],
a61af66fc99e Initial load
duke
parents:
diff changeset
2758 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2759 _bt.verify_not_unallocated((HeapWord*)rem_fc, rem_fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2760 _indexedFreeList[rem].returnChunkAtHead(rem_fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
2761 smallSplitBirth(rem);
a61af66fc99e Initial load
duke
parents:
diff changeset
2762 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2763
a61af66fc99e Initial load
duke
parents:
diff changeset
2764 // Now do the splitting up.
a61af66fc99e Initial load
duke
parents:
diff changeset
2765 // Must do this in reverse order, so that anybody attempting to
a61af66fc99e Initial load
duke
parents:
diff changeset
2766 // access the main chunk sees it as a single free block until we
a61af66fc99e Initial load
duke
parents:
diff changeset
2767 // change it.
a61af66fc99e Initial load
duke
parents:
diff changeset
2768 size_t fc_size = n * word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2769 // All but first chunk in this loop
a61af66fc99e Initial load
duke
parents:
diff changeset
2770 for (ssize_t i = n-1; i > 0; i--) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2771 FreeChunk* ffc = (FreeChunk*)((HeapWord*)fc + i * word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2772 ffc->setSize(word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2773 ffc->linkNext(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
2774 ffc->linkPrev(NULL); // Mark as a free block for other (parallel) GC threads.
a61af66fc99e Initial load
duke
parents:
diff changeset
2775 // Above must occur before BOT is updated below.
a61af66fc99e Initial load
duke
parents:
diff changeset
2776 // splitting from the right, fc_size == (n - i + 1) * wordsize
a61af66fc99e Initial load
duke
parents:
diff changeset
2777 _bt.mark_block((HeapWord*)ffc, word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2778 fc_size -= word_sz;
a61af66fc99e Initial load
duke
parents:
diff changeset
2779 _bt.verify_not_unallocated((HeapWord*)ffc, ffc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2780 _bt.verify_single_block((HeapWord*)ffc, ffc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2781 _bt.verify_single_block((HeapWord*)fc, fc_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2782 // Push this on "fl".
a61af66fc99e Initial load
duke
parents:
diff changeset
2783 fl->returnChunkAtHead(ffc);
a61af66fc99e Initial load
duke
parents:
diff changeset
2784 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2785 // First chunk
a61af66fc99e Initial load
duke
parents:
diff changeset
2786 fc->setSize(word_sz);
a61af66fc99e Initial load
duke
parents:
diff changeset
2787 fc->linkNext(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
2788 fc->linkPrev(NULL);
a61af66fc99e Initial load
duke
parents:
diff changeset
2789 _bt.verify_not_unallocated((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2790 _bt.verify_single_block((HeapWord*)fc, fc->size());
a61af66fc99e Initial load
duke
parents:
diff changeset
2791 fl->returnChunkAtHead(fc);
a61af66fc99e Initial load
duke
parents:
diff changeset
2792
a61af66fc99e Initial load
duke
parents:
diff changeset
2793 {
a61af66fc99e Initial load
duke
parents:
diff changeset
2794 MutexLockerEx x(_indexedFreeListParLocks[word_sz],
a61af66fc99e Initial load
duke
parents:
diff changeset
2795 Mutex::_no_safepoint_check_flag);
a61af66fc99e Initial load
duke
parents:
diff changeset
2796 ssize_t new_births = _indexedFreeList[word_sz].splitBirths() + n;
a61af66fc99e Initial load
duke
parents:
diff changeset
2797 _indexedFreeList[word_sz].set_splitBirths(new_births);
a61af66fc99e Initial load
duke
parents:
diff changeset
2798 ssize_t new_surplus = _indexedFreeList[word_sz].surplus() + n;
a61af66fc99e Initial load
duke
parents:
diff changeset
2799 _indexedFreeList[word_sz].set_surplus(new_surplus);
a61af66fc99e Initial load
duke
parents:
diff changeset
2800 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2801
a61af66fc99e Initial load
duke
parents:
diff changeset
2802 // TRAP
a61af66fc99e Initial load
duke
parents:
diff changeset
2803 assert(fl->tail()->next() == NULL, "List invariant.");
a61af66fc99e Initial load
duke
parents:
diff changeset
2804 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2805
a61af66fc99e Initial load
duke
parents:
diff changeset
2806 // Set up the space's par_seq_tasks structure for work claiming
a61af66fc99e Initial load
duke
parents:
diff changeset
2807 // for parallel rescan. See CMSParRemarkTask where this is currently used.
a61af66fc99e Initial load
duke
parents:
diff changeset
2808 // XXX Need to suitably abstract and generalize this and the next
a61af66fc99e Initial load
duke
parents:
diff changeset
2809 // method into one.
a61af66fc99e Initial load
duke
parents:
diff changeset
2810 void
a61af66fc99e Initial load
duke
parents:
diff changeset
2811 CompactibleFreeListSpace::
a61af66fc99e Initial load
duke
parents:
diff changeset
2812 initialize_sequential_subtasks_for_rescan(int n_threads) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2813 // The "size" of each task is fixed according to rescan_task_size.
a61af66fc99e Initial load
duke
parents:
diff changeset
2814 assert(n_threads > 0, "Unexpected n_threads argument");
a61af66fc99e Initial load
duke
parents:
diff changeset
2815 const size_t task_size = rescan_task_size();
a61af66fc99e Initial load
duke
parents:
diff changeset
2816 size_t n_tasks = (used_region().word_size() + task_size - 1)/task_size;
340
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2817 assert((n_tasks == 0) == used_region().is_empty(), "n_tasks incorrect");
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2818 assert(n_tasks == 0 ||
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2819 ((used_region().start() + (n_tasks - 1)*task_size < used_region().end()) &&
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2820 (used_region().start() + n_tasks*task_size >= used_region().end())),
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2821 "n_tasks calculation incorrect");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2822 SequentialSubTasksDone* pst = conc_par_seq_tasks();
a61af66fc99e Initial load
duke
parents:
diff changeset
2823 assert(!pst->valid(), "Clobbering existing data?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2824 pst->set_par_threads(n_threads);
a61af66fc99e Initial load
duke
parents:
diff changeset
2825 pst->set_n_tasks((int)n_tasks);
a61af66fc99e Initial load
duke
parents:
diff changeset
2826 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2827
a61af66fc99e Initial load
duke
parents:
diff changeset
2828 // Set up the space's par_seq_tasks structure for work claiming
a61af66fc99e Initial load
duke
parents:
diff changeset
2829 // for parallel concurrent marking. See CMSConcMarkTask where this is currently used.
a61af66fc99e Initial load
duke
parents:
diff changeset
2830 void
a61af66fc99e Initial load
duke
parents:
diff changeset
2831 CompactibleFreeListSpace::
a61af66fc99e Initial load
duke
parents:
diff changeset
2832 initialize_sequential_subtasks_for_marking(int n_threads,
a61af66fc99e Initial load
duke
parents:
diff changeset
2833 HeapWord* low) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2834 // The "size" of each task is fixed according to rescan_task_size.
a61af66fc99e Initial load
duke
parents:
diff changeset
2835 assert(n_threads > 0, "Unexpected n_threads argument");
a61af66fc99e Initial load
duke
parents:
diff changeset
2836 const size_t task_size = marking_task_size();
a61af66fc99e Initial load
duke
parents:
diff changeset
2837 assert(task_size > CardTableModRefBS::card_size_in_words &&
a61af66fc99e Initial load
duke
parents:
diff changeset
2838 (task_size % CardTableModRefBS::card_size_in_words == 0),
a61af66fc99e Initial load
duke
parents:
diff changeset
2839 "Otherwise arithmetic below would be incorrect");
a61af66fc99e Initial load
duke
parents:
diff changeset
2840 MemRegion span = _gen->reserved();
a61af66fc99e Initial load
duke
parents:
diff changeset
2841 if (low != NULL) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2842 if (span.contains(low)) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2843 // Align low down to a card boundary so that
a61af66fc99e Initial load
duke
parents:
diff changeset
2844 // we can use block_offset_careful() on span boundaries.
a61af66fc99e Initial load
duke
parents:
diff changeset
2845 HeapWord* aligned_low = (HeapWord*)align_size_down((uintptr_t)low,
a61af66fc99e Initial load
duke
parents:
diff changeset
2846 CardTableModRefBS::card_size);
a61af66fc99e Initial load
duke
parents:
diff changeset
2847 // Clip span prefix at aligned_low
a61af66fc99e Initial load
duke
parents:
diff changeset
2848 span = span.intersection(MemRegion(aligned_low, span.end()));
a61af66fc99e Initial load
duke
parents:
diff changeset
2849 } else if (low > span.end()) {
a61af66fc99e Initial load
duke
parents:
diff changeset
2850 span = MemRegion(low, low); // Null region
a61af66fc99e Initial load
duke
parents:
diff changeset
2851 } // else use entire span
a61af66fc99e Initial load
duke
parents:
diff changeset
2852 }
a61af66fc99e Initial load
duke
parents:
diff changeset
2853 assert(span.is_empty() ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2854 ((uintptr_t)span.start() % CardTableModRefBS::card_size == 0),
a61af66fc99e Initial load
duke
parents:
diff changeset
2855 "span should start at a card boundary");
a61af66fc99e Initial load
duke
parents:
diff changeset
2856 size_t n_tasks = (span.word_size() + task_size - 1)/task_size;
a61af66fc99e Initial load
duke
parents:
diff changeset
2857 assert((n_tasks == 0) == span.is_empty(), "Inconsistency");
a61af66fc99e Initial load
duke
parents:
diff changeset
2858 assert(n_tasks == 0 ||
a61af66fc99e Initial load
duke
parents:
diff changeset
2859 ((span.start() + (n_tasks - 1)*task_size < span.end()) &&
a61af66fc99e Initial load
duke
parents:
diff changeset
2860 (span.start() + n_tasks*task_size >= span.end())),
340
ebeb6490b814 6722116: CMS: Incorrect overflow handling when using parallel concurrent marking
ysr
parents: 269
diff changeset
2861 "n_tasks calculation incorrect");
0
a61af66fc99e Initial load
duke
parents:
diff changeset
2862 SequentialSubTasksDone* pst = conc_par_seq_tasks();
a61af66fc99e Initial load
duke
parents:
diff changeset
2863 assert(!pst->valid(), "Clobbering existing data?");
a61af66fc99e Initial load
duke
parents:
diff changeset
2864 pst->set_par_threads(n_threads);
a61af66fc99e Initial load
duke
parents:
diff changeset
2865 pst->set_n_tasks((int)n_tasks);
a61af66fc99e Initial load
duke
parents:
diff changeset
2866 }