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