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