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