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