src/hotspot/share/jfr/recorder/storage/jfrStorage.cpp
author mgronlun
Tue, 21 May 2019 20:53:27 +0200
changeset 54964 ec7d6d8effc7
parent 54623 1126f0607c70
child 55053 d58e1a447d2b
permissions -rw-r--r--
8220293: Deadlock in JFR string pool Reviewed-by: rehn, egahlin
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     1
/*
54623
1126f0607c70 8222811: Consolidate MutexLockerEx and MutexLocker
coleenp
parents: 50429
diff changeset
     2
 * Copyright (c) 2012, 2019, Oracle and/or its affiliates. All rights reserved.
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     4
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     7
 * published by the Free Software Foundation.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     8
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     9
 * This code is distributed in the hope that it will be useful, but WITHOUT
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    10
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    11
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    12
 * version 2 for more details (a copy is included in the LICENSE file that
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    13
 * accompanied this code).
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    14
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    15
 * You should have received a copy of the GNU General Public License version
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    16
 * 2 along with this work; if not, write to the Free Software Foundation,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    17
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    18
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    19
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    20
 * or visit www.oracle.com if you need additional information or have any
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    21
 * questions.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    22
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    23
 */
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    24
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    25
#include "precompiled.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    26
#include "jfr/jfrEvents.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    27
#include "jfr/jni/jfrJavaSupport.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    28
#include "jfr/recorder/jfrRecorder.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    29
#include "jfr/recorder/repository/jfrChunkWriter.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    30
#include "jfr/recorder/service/jfrOptionSet.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    31
#include "jfr/recorder/service/jfrPostBox.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    32
#include "jfr/recorder/storage/jfrMemorySpace.inline.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    33
#include "jfr/recorder/storage/jfrStorage.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    34
#include "jfr/recorder/storage/jfrStorageControl.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    35
#include "jfr/recorder/storage/jfrStorageUtils.inline.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    36
#include "jfr/utilities/jfrIterator.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    37
#include "jfr/utilities/jfrTime.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    38
#include "jfr/writers/jfrNativeEventWriter.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    39
#include "logging/log.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    40
#include "runtime/mutexLocker.hpp"
50429
83aec1d357d4 8204301: Make OrderAccess functions available to hpp rather than inline.hpp files
coleenp
parents: 50234
diff changeset
    41
#include "runtime/orderAccess.hpp"
50117
fb66b2959eaf 8203251: Non-PCH build failed after JDK-8199712 (Flight Recorder)
shade
parents: 50113
diff changeset
    42
#include "runtime/os.inline.hpp"
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    43
#include "runtime/safepoint.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    44
#include "runtime/thread.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    45
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    46
typedef JfrStorage::Buffer* BufferPtr;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    47
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    48
static JfrStorage* _instance = NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    49
static JfrStorageControl* _control;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    50
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    51
JfrStorage& JfrStorage::instance() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    52
  return *_instance;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    53
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    54
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    55
JfrStorage* JfrStorage::create(JfrChunkWriter& chunkwriter, JfrPostBox& post_box) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    56
  assert(_instance == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    57
  _instance = new JfrStorage(chunkwriter, post_box);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    58
  return _instance;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    59
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    60
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    61
void JfrStorage::destroy() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    62
  if (_instance != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    63
    delete _instance;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    64
    _instance = NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    65
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    66
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    67
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    68
JfrStorage::JfrStorage(JfrChunkWriter& chunkwriter, JfrPostBox& post_box) :
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    69
  _control(NULL),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    70
  _global_mspace(NULL),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    71
  _thread_local_mspace(NULL),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    72
  _transient_mspace(NULL),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    73
  _age_mspace(NULL),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    74
  _chunkwriter(chunkwriter),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    75
  _post_box(post_box) {}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    76
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    77
JfrStorage::~JfrStorage() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    78
  if (_control != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    79
    delete _control;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    80
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    81
  if (_global_mspace != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    82
    delete _global_mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    83
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    84
  if (_thread_local_mspace != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    85
    delete _thread_local_mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    86
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    87
  if (_transient_mspace != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    88
    delete _transient_mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    89
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    90
  if (_age_mspace != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    91
    delete _age_mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    92
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    93
  _instance = NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    94
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    95
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    96
static const size_t in_memory_discard_threshold_delta = 2; // start to discard data when the only this number of free buffers are left
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    97
static const size_t unlimited_mspace_size = 0;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    98
static const size_t thread_local_cache_count = 8;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    99
static const size_t thread_local_scavenge_threshold = thread_local_cache_count / 2;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   100
static const size_t transient_buffer_size_multiplier = 8; // against thread local buffer size
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   101
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   102
template <typename Mspace>
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   103
static Mspace* create_mspace(size_t buffer_size, size_t limit, size_t cache_count, JfrStorage* storage_instance) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   104
  Mspace* mspace = new Mspace(buffer_size, limit, cache_count, storage_instance);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   105
  if (mspace != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   106
    mspace->initialize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   107
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   108
  return mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   109
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   110
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   111
bool JfrStorage::initialize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   112
  assert(_control == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   113
  assert(_global_mspace == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   114
  assert(_thread_local_mspace == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   115
  assert(_transient_mspace == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   116
  assert(_age_mspace == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   117
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   118
  const size_t num_global_buffers = (size_t)JfrOptionSet::num_global_buffers();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   119
  assert(num_global_buffers >= in_memory_discard_threshold_delta, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   120
  const size_t memory_size = (size_t)JfrOptionSet::memory_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   121
  const size_t global_buffer_size = (size_t)JfrOptionSet::global_buffer_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   122
  const size_t thread_buffer_size = (size_t)JfrOptionSet::thread_buffer_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   123
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   124
  _control = new JfrStorageControl(num_global_buffers, num_global_buffers - in_memory_discard_threshold_delta);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   125
  if (_control == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   126
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   127
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   128
  _global_mspace = create_mspace<JfrStorageMspace>(global_buffer_size, memory_size, num_global_buffers, this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   129
  if (_global_mspace == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   130
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   131
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   132
  _thread_local_mspace = create_mspace<JfrThreadLocalMspace>(thread_buffer_size, unlimited_mspace_size, thread_local_cache_count, this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   133
  if (_thread_local_mspace == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   134
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   135
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   136
  _transient_mspace = create_mspace<JfrStorageMspace>(thread_buffer_size * transient_buffer_size_multiplier, unlimited_mspace_size, 0, this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   137
  if (_transient_mspace == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   138
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   139
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   140
  _age_mspace = create_mspace<JfrStorageAgeMspace>(0 /* no extra size except header */, unlimited_mspace_size, num_global_buffers, this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   141
  if (_age_mspace == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   142
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   143
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   144
  control().set_scavenge_threshold(thread_local_scavenge_threshold);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   145
  return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   146
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   147
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   148
JfrStorageControl& JfrStorage::control() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   149
  return *instance()._control;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   150
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   151
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   152
static void log_allocation_failure(const char* msg, size_t size) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   153
  log_warning(jfr)("Unable to allocate " SIZE_FORMAT " bytes of %s.", size, msg);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   154
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   155
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   156
BufferPtr JfrStorage::acquire_thread_local(Thread* thread, size_t size /* 0 */) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   157
  BufferPtr buffer = mspace_get_to_full(size, instance()._thread_local_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   158
  if (buffer == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   159
    log_allocation_failure("thread local_memory", size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   160
    return NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   161
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   162
  assert(buffer->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   163
  return buffer;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   164
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   165
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   166
BufferPtr JfrStorage::acquire_transient(size_t size, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   167
  BufferPtr buffer = mspace_allocate_transient_lease_to_full(size, instance()._transient_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   168
  if (buffer == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   169
    log_allocation_failure("transient memory", size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   170
    return NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   171
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   172
  assert(buffer->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   173
  assert(buffer->transient(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   174
  assert(buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   175
  return buffer;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   176
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   177
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   178
static BufferPtr get_lease(size_t size, JfrStorageMspace* mspace, JfrStorage& storage_instance, size_t retry_count, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   179
  assert(size <= mspace->min_elem_size(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   180
  while (true) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   181
    BufferPtr t = mspace_get_free_lease_with_retry(size, mspace, retry_count, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   182
    if (t == NULL && storage_instance.control().should_discard()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   183
      storage_instance.discard_oldest(thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   184
      continue;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   185
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   186
    return t;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   187
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   188
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   189
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   190
static BufferPtr get_promotion_buffer(size_t size, JfrStorageMspace* mspace, JfrStorage& storage_instance, size_t retry_count, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   191
  assert(size <= mspace->min_elem_size(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   192
  while (true) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   193
    BufferPtr t = mspace_get_free_with_retry(size, mspace, retry_count, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   194
    if (t == NULL && storage_instance.control().should_discard()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   195
      storage_instance.discard_oldest(thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   196
      continue;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   197
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   198
    return t;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   199
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   200
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   201
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   202
static const size_t lease_retry = 10;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   203
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   204
BufferPtr JfrStorage::acquire_large(size_t size, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   205
  JfrStorage& storage_instance = instance();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   206
  const size_t max_elem_size = storage_instance._global_mspace->min_elem_size(); // min is also max
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   207
  // if not too large and capacity is still available, ask for a lease from the global system
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   208
  if (size < max_elem_size && storage_instance.control().is_global_lease_allowed()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   209
    BufferPtr const buffer = get_lease(size, storage_instance._global_mspace, storage_instance, lease_retry, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   210
    if (buffer != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   211
      assert(buffer->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   212
      assert(!buffer->transient(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   213
      assert(buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   214
      storage_instance.control().increment_leased();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   215
      return buffer;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   216
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   217
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   218
  return acquire_transient(size, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   219
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   220
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   221
static void write_data_loss_event(JfrBuffer* buffer, u8 unflushed_size, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   222
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   223
  assert(buffer->empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   224
  const u8 total_data_loss = thread->jfr_thread_local()->add_data_lost(unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   225
  if (EventDataLoss::is_enabled()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   226
    JfrNativeEventWriter writer(buffer, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   227
    writer.write<u8>(EventDataLoss::eventId);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   228
    writer.write(JfrTicks::now());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   229
    writer.write(unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   230
    writer.write(total_data_loss);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   231
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   232
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   233
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   234
static void write_data_loss(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   235
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   236
  const size_t unflushed_size = buffer->unflushed_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   237
  buffer->concurrent_reinitialization();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   238
  if (unflushed_size == 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   239
    return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   240
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   241
  write_data_loss_event(buffer, unflushed_size, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   242
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   243
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   244
static const size_t promotion_retry = 100;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   245
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   246
bool JfrStorage::flush_regular_buffer(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   247
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   248
  assert(!buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   249
  assert(!buffer->transient(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   250
  const size_t unflushed_size = buffer->unflushed_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   251
  if (unflushed_size == 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   252
    buffer->concurrent_reinitialization();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   253
    assert(buffer->empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   254
    return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   255
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   256
  BufferPtr const promotion_buffer = get_promotion_buffer(unflushed_size, _global_mspace, *this, promotion_retry, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   257
  if (promotion_buffer == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   258
    write_data_loss(buffer, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   259
    return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   260
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   261
  assert(promotion_buffer->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   262
  assert(promotion_buffer->free_size() >= unflushed_size, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   263
  buffer->concurrent_move_and_reinitialize(promotion_buffer, unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   264
  assert(buffer->empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   265
  return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   266
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   267
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   268
/*
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   269
* 1. If the buffer was a "lease" from the global system, release back.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   270
* 2. If the buffer is transient (temporal dynamically allocated), retire and register full.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   271
*
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   272
* The buffer is effectively invalidated for the thread post-return,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   273
* and the caller should take means to ensure that it is not referenced any longer.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   274
*/
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   275
void JfrStorage::release_large(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   276
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   277
  assert(buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   278
  assert(buffer->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   279
  buffer->clear_lease();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   280
  if (buffer->transient()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   281
    buffer->set_retired();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   282
    register_full(buffer, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   283
  } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   284
    buffer->release();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   285
    control().decrement_leased();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   286
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   287
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   288
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   289
static JfrAgeNode* new_age_node(BufferPtr buffer, JfrStorageAgeMspace* age_mspace, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   290
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   291
  assert(age_mspace != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   292
  return mspace_allocate_transient(0, age_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   293
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   294
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   295
static void log_registration_failure(size_t unflushed_size) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   296
  log_warning(jfr)("Unable to register a full buffer of " SIZE_FORMAT " bytes.", unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   297
  log_debug(jfr, system)("Cleared 1 full buffer of " SIZE_FORMAT " bytes.", unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   298
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   299
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   300
static void handle_registration_failure(BufferPtr buffer) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   301
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   302
  assert(buffer->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   303
  const size_t unflushed_size = buffer->unflushed_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   304
  buffer->reinitialize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   305
  log_registration_failure(unflushed_size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   306
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   307
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   308
static JfrAgeNode* get_free_age_node(JfrStorageAgeMspace* age_mspace, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   309
  assert(JfrBuffer_lock->owned_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   310
  return mspace_get_free_with_detach(0, age_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   311
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   312
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   313
static bool insert_full_age_node(JfrAgeNode* age_node, JfrStorageAgeMspace* age_mspace, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   314
  assert(JfrBuffer_lock->owned_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   315
  assert(age_node->retired_buffer()->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   316
  age_mspace->insert_full_head(age_node);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   317
  return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   318
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   319
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   320
static bool full_buffer_registration(BufferPtr buffer, JfrStorageAgeMspace* age_mspace, JfrStorageControl& control, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   321
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   322
  assert(buffer->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   323
  assert(age_mspace != NULL, "invariant");
54623
1126f0607c70 8222811: Consolidate MutexLockerEx and MutexLocker
coleenp
parents: 50429
diff changeset
   324
  MutexLocker lock(JfrBuffer_lock, Mutex::_no_safepoint_check_flag);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   325
  JfrAgeNode* age_node = get_free_age_node(age_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   326
  if (age_node == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   327
    age_node = new_age_node(buffer, age_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   328
    if (age_node == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   329
      return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   330
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   331
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   332
  assert(age_node->acquired_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   333
  assert(age_node != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   334
  age_node->set_retired_buffer(buffer);
50234
6ba3e32a9882 8203457: Add back missing full buffer notification
mgronlun
parents: 50117
diff changeset
   335
  control.increment_full();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   336
  return insert_full_age_node(age_node, age_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   337
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   338
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   339
void JfrStorage::register_full(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   340
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   341
  assert(buffer->retired(), "invariant");
54964
ec7d6d8effc7 8220293: Deadlock in JFR string pool
mgronlun
parents: 54623
diff changeset
   342
  assert(buffer->acquired_by(thread), "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   343
  if (!full_buffer_registration(buffer, _age_mspace, control(), thread)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   344
    handle_registration_failure(buffer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   345
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   346
  if (control().should_post_buffer_full_message()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   347
    _post_box.post(MSG_FULLBUFFER);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   348
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   349
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   350
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   351
void JfrStorage::lock() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   352
  assert(!JfrBuffer_lock->owned_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   353
  JfrBuffer_lock->lock_without_safepoint_check();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   354
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   355
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   356
void JfrStorage::unlock() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   357
  assert(JfrBuffer_lock->owned_by_self(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   358
  JfrBuffer_lock->unlock();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   359
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   360
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   361
#ifdef ASSERT
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   362
bool JfrStorage::is_locked() const {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   363
  return JfrBuffer_lock->owned_by_self();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   364
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   365
#endif
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   366
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   367
// don't use buffer on return, it is gone
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   368
void JfrStorage::release(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   369
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   370
  assert(!buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   371
  assert(!buffer->transient(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   372
  assert(!buffer->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   373
  if (!buffer->empty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   374
    if (!flush_regular_buffer(buffer, thread)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   375
      buffer->concurrent_reinitialization();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   376
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   377
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   378
  assert(buffer->empty(), "invariant");
54964
ec7d6d8effc7 8220293: Deadlock in JFR string pool
mgronlun
parents: 54623
diff changeset
   379
  assert(buffer->identity() != NULL, "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   380
  control().increment_dead();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   381
  buffer->set_retired();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   382
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   383
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   384
void JfrStorage::release_thread_local(BufferPtr buffer, Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   385
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   386
  JfrStorage& storage_instance = instance();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   387
  storage_instance.release(buffer, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   388
  if (storage_instance.control().should_scavenge()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   389
    storage_instance._post_box.post(MSG_DEADBUFFER);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   390
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   391
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   392
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   393
static void log_discard(size_t count, size_t amount, size_t current) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   394
  if (log_is_enabled(Debug, jfr, system)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   395
    assert(count > 0, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   396
    log_debug(jfr, system)("Cleared " SIZE_FORMAT " full buffer(s) of " SIZE_FORMAT" bytes.", count, amount);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   397
    log_debug(jfr, system)("Current number of full buffers " SIZE_FORMAT "", current);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   398
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   399
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   400
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   401
void JfrStorage::discard_oldest(Thread* thread) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   402
  if (JfrBuffer_lock->try_lock()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   403
    if (!control().should_discard()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   404
      // another thread handled it
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   405
      return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   406
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   407
    const size_t num_full_pre_discard = control().full_count();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   408
    size_t num_full_post_discard = 0;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   409
    size_t discarded_size = 0;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   410
    while (true) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   411
      JfrAgeNode* const oldest_age_node = _age_mspace->full_tail();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   412
      if (oldest_age_node == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   413
        break;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   414
      }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   415
      BufferPtr const buffer = oldest_age_node->retired_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   416
      assert(buffer->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   417
      discarded_size += buffer->unflushed_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   418
      num_full_post_discard = control().decrement_full();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   419
      if (buffer->transient()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   420
        mspace_release_full(buffer, _transient_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   421
        mspace_release_full(oldest_age_node, _age_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   422
        continue;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   423
      } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   424
        mspace_release_full(oldest_age_node, _age_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   425
        buffer->reinitialize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   426
        buffer->release(); // pusb
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   427
        break;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   428
      }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   429
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   430
    JfrBuffer_lock->unlock();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   431
    const size_t number_of_discards = num_full_pre_discard - num_full_post_discard;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   432
    if (number_of_discards > 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   433
      log_discard(number_of_discards, discarded_size, num_full_post_discard);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   434
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   435
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   436
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   437
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   438
#ifdef ASSERT
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   439
typedef const BufferPtr ConstBufferPtr;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   440
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   441
static void assert_flush_precondition(ConstBufferPtr cur, size_t used, bool native, const Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   442
  assert(t != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   443
  assert(cur != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   444
  assert(cur->pos() + used <= cur->end(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   445
  assert(native ? t->jfr_thread_local()->native_buffer() == cur : t->jfr_thread_local()->java_buffer() == cur, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   446
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   447
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   448
static void assert_flush_regular_precondition(ConstBufferPtr cur, const u1* const cur_pos, size_t used, size_t req, const Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   449
  assert(t != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   450
  assert(t->jfr_thread_local()->shelved_buffer() == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   451
  assert(cur != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   452
  assert(!cur->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   453
  assert(cur_pos != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   454
  assert(req >= used, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   455
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   456
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   457
static void assert_provision_large_precondition(ConstBufferPtr cur, size_t used, size_t req, const Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   458
  assert(cur != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   459
  assert(t != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   460
  assert(t->jfr_thread_local()->shelved_buffer() != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   461
  assert(req >= used, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   462
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   463
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   464
static void assert_flush_large_precondition(ConstBufferPtr cur, const u1* const cur_pos, size_t used, size_t req, bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   465
  assert(t != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   466
  assert(cur != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   467
  assert(cur->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   468
  assert(cur_pos != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   469
  assert(native ? t->jfr_thread_local()->native_buffer() == cur : t->jfr_thread_local()->java_buffer() == cur, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   470
  assert(t->jfr_thread_local()->shelved_buffer() != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   471
  assert(req >= used, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   472
  assert(cur != t->jfr_thread_local()->shelved_buffer(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   473
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   474
#endif // ASSERT
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   475
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   476
BufferPtr JfrStorage::flush(BufferPtr cur, size_t used, size_t req, bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   477
  debug_only(assert_flush_precondition(cur, used, native, t);)
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   478
  const u1* const cur_pos = cur->pos();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   479
  req += used;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   480
  // requested size now encompass the outstanding used size
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   481
  return cur->lease() ? instance().flush_large(cur, cur_pos, used, req, native, t) :
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   482
                          instance().flush_regular(cur, cur_pos, used, req, native, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   483
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   484
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   485
BufferPtr JfrStorage::flush_regular(BufferPtr cur, const u1* const cur_pos, size_t used, size_t req, bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   486
  debug_only(assert_flush_regular_precondition(cur, cur_pos, used, req, t);)
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   487
  // A flush is needed before memcpy since a non-large buffer is thread stable
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   488
  // (thread local). The flush will not modify memory in addresses above pos()
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   489
  // which is where the "used / uncommitted" data resides. It is therefore both
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   490
  // possible and valid to migrate data after the flush. This is however only
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   491
  // the case for stable thread local buffers; it is not the case for large buffers.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   492
  if (!cur->empty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   493
    flush_regular_buffer(cur, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   494
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   495
  assert(t->jfr_thread_local()->shelved_buffer() == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   496
  if (cur->free_size() >= req) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   497
    // simplest case, no switching of buffers
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   498
    if (used > 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   499
      memcpy(cur->pos(), (void*)cur_pos, used);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   500
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   501
    assert(native ? t->jfr_thread_local()->native_buffer() == cur : t->jfr_thread_local()->java_buffer() == cur, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   502
    return cur;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   503
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   504
  // Going for a "larger-than-regular" buffer.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   505
  // Shelve the current buffer to make room for a temporary lease.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   506
  t->jfr_thread_local()->shelve_buffer(cur);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   507
  return provision_large(cur, cur_pos, used, req, native, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   508
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   509
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   510
static BufferPtr store_buffer_to_thread_local(BufferPtr buffer, JfrThreadLocal* jfr_thread_local, bool native) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   511
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   512
  if (native) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   513
    jfr_thread_local->set_native_buffer(buffer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   514
  } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   515
    jfr_thread_local->set_java_buffer(buffer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   516
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   517
  return buffer;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   518
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   519
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   520
static BufferPtr restore_shelved_buffer(bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   521
  JfrThreadLocal* const tl = t->jfr_thread_local();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   522
  BufferPtr shelved = tl->shelved_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   523
  assert(shelved != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   524
  tl->shelve_buffer(NULL);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   525
  // restore shelved buffer back as primary
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   526
  return store_buffer_to_thread_local(shelved, tl, native);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   527
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   528
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   529
BufferPtr JfrStorage::flush_large(BufferPtr cur, const u1* const cur_pos, size_t used, size_t req, bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   530
  debug_only(assert_flush_large_precondition(cur, cur_pos, used, req, native, t);)
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   531
  // Can the "regular" buffer (now shelved) accommodate the requested size?
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   532
  BufferPtr shelved = t->jfr_thread_local()->shelved_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   533
  assert(shelved != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   534
  if (shelved->free_size() >= req) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   535
    if (req > 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   536
      memcpy(shelved->pos(), (void*)cur_pos, (size_t)used);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   537
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   538
    // release and invalidate
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   539
    release_large(cur, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   540
    return restore_shelved_buffer(native, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   541
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   542
  // regular too small
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   543
  return provision_large(cur, cur_pos,  used, req, native, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   544
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   545
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   546
static BufferPtr large_fail(BufferPtr cur, bool native, JfrStorage& storage_instance, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   547
  assert(cur != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   548
  assert(t != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   549
  if (cur->lease()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   550
    storage_instance.release_large(cur, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   551
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   552
  return restore_shelved_buffer(native, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   553
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   554
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   555
// Always returns a non-null buffer.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   556
// If accommodating the large request fails, the shelved buffer is returned
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   557
// even though it might be smaller than the requested size.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   558
// Caller needs to ensure if the size was successfully accommodated.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   559
BufferPtr JfrStorage::provision_large(BufferPtr cur, const u1* const cur_pos, size_t used, size_t req, bool native, Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   560
  debug_only(assert_provision_large_precondition(cur, used, req, t);)
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   561
  assert(t->jfr_thread_local()->shelved_buffer() != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   562
  BufferPtr const buffer = acquire_large(req, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   563
  if (buffer == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   564
    // unable to allocate and serve the request
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   565
    return large_fail(cur, native, *this, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   566
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   567
  // ok managed to acquire a "large" buffer for the requested size
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   568
  assert(buffer->free_size() >= req, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   569
  assert(buffer->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   570
  // transfer outstanding data
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   571
  memcpy(buffer->pos(), (void*)cur_pos, used);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   572
  if (cur->lease()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   573
    release_large(cur, t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   574
    // don't use current anymore, it is gone
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   575
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   576
  return store_buffer_to_thread_local(buffer, t->jfr_thread_local(), native);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   577
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   578
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   579
typedef UnBufferedWriteToChunk<JfrBuffer> WriteOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   580
typedef MutexedWriteOp<WriteOperation> MutexedWriteOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   581
typedef ConcurrentWriteOp<WriteOperation> ConcurrentWriteOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   582
typedef ConcurrentWriteOpExcludeRetired<WriteOperation> ThreadLocalConcurrentWriteOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   583
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   584
size_t JfrStorage::write() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   585
  const size_t full_size_processed = write_full();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   586
  WriteOperation wo(_chunkwriter);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   587
  ThreadLocalConcurrentWriteOperation tlwo(wo);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   588
  process_full_list(tlwo, _thread_local_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   589
  ConcurrentWriteOperation cwo(wo);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   590
  process_free_list(cwo, _global_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   591
  return full_size_processed + wo.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   592
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   593
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   594
size_t JfrStorage::write_at_safepoint() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   595
  assert(SafepointSynchronize::is_at_safepoint(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   596
  WriteOperation wo(_chunkwriter);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   597
  MutexedWriteOperation writer(wo); // mutexed write mode
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   598
  process_full_list(writer, _thread_local_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   599
  assert(_transient_mspace->is_free_empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   600
  process_full_list(writer, _transient_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   601
  assert(_global_mspace->is_full_empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   602
  process_free_list(writer, _global_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   603
  return wo.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   604
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   605
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   606
typedef DiscardOp<DefaultDiscarder<JfrStorage::Buffer> > DiscardOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   607
typedef ReleaseOp<JfrStorageMspace> ReleaseOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   608
typedef CompositeOperation<MutexedWriteOperation, ReleaseOperation> FullOperation;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   609
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   610
size_t JfrStorage::clear() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   611
  const size_t full_size_processed = clear_full();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   612
  DiscardOperation discarder(concurrent); // concurrent discard mode
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   613
  process_full_list(discarder, _thread_local_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   614
  assert(_transient_mspace->is_free_empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   615
  process_full_list(discarder, _transient_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   616
  assert(_global_mspace->is_full_empty(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   617
  process_free_list(discarder, _global_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   618
  return full_size_processed + discarder.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   619
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   620
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   621
static void insert_free_age_nodes(JfrStorageAgeMspace* age_mspace, JfrAgeNode* head, JfrAgeNode* tail, size_t count) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   622
  if (tail != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   623
    assert(tail->next() == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   624
    assert(head != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   625
    assert(head->prev() == NULL, "invariant");
54623
1126f0607c70 8222811: Consolidate MutexLockerEx and MutexLocker
coleenp
parents: 50429
diff changeset
   626
    MutexLocker buffer_lock(JfrBuffer_lock, Mutex::_no_safepoint_check_flag);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   627
    age_mspace->insert_free_tail(head, tail, count);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   628
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   629
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   630
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   631
template <typename Processor>
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   632
static void process_age_list(Processor& processor, JfrStorageAgeMspace* age_mspace, JfrAgeNode* head, size_t count) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   633
  assert(age_mspace != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   634
  assert(head != NULL, "invariant");
50234
6ba3e32a9882 8203457: Add back missing full buffer notification
mgronlun
parents: 50117
diff changeset
   635
  assert(count > 0, "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   636
  JfrAgeNode* node = head;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   637
  JfrAgeNode* last = NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   638
  while (node != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   639
    last = node;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   640
    BufferPtr const buffer = node->retired_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   641
    assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   642
    assert(buffer->retired(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   643
    processor.process(buffer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   644
    // at this point, buffer is already live or destroyed
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   645
    node->clear_identity();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   646
    JfrAgeNode* const next = (JfrAgeNode*)node->next();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   647
    if (node->transient()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   648
      // detach
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   649
      last = (JfrAgeNode*)last->prev();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   650
      if (last != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   651
        last->set_next(next);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   652
      } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   653
        head = next;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   654
      }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   655
      if (next != NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   656
        next->set_prev(last);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   657
      }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   658
      --count;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   659
      age_mspace->deallocate(node);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   660
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   661
    node = next;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   662
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   663
  insert_free_age_nodes(age_mspace, head, last, count);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   664
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   665
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   666
template <typename Processor>
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   667
static size_t process_full(Processor& processor, JfrStorageControl& control, JfrStorageAgeMspace* age_mspace) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   668
  assert(age_mspace != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   669
  if (age_mspace->is_full_empty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   670
    // nothing to do
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   671
    return 0;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   672
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   673
  size_t count;
50234
6ba3e32a9882 8203457: Add back missing full buffer notification
mgronlun
parents: 50117
diff changeset
   674
  JfrAgeNode* head;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   675
  {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   676
    // fetch age list
54623
1126f0607c70 8222811: Consolidate MutexLockerEx and MutexLocker
coleenp
parents: 50429
diff changeset
   677
    MutexLocker buffer_lock(JfrBuffer_lock, Mutex::_no_safepoint_check_flag);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   678
    count = age_mspace->full_count();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   679
    head = age_mspace->clear_full();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   680
    control.reset_full();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   681
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   682
  assert(head != NULL, "invariant");
50234
6ba3e32a9882 8203457: Add back missing full buffer notification
mgronlun
parents: 50117
diff changeset
   683
  assert(count > 0, "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   684
  process_age_list(processor, age_mspace, head, count);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   685
  return count;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   686
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   687
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   688
static void log(size_t count, size_t amount, bool clear = false) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   689
  if (log_is_enabled(Debug, jfr, system)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   690
    if (count > 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   691
      log_debug(jfr, system)("%s " SIZE_FORMAT " full buffer(s) of " SIZE_FORMAT" B of data%s",
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   692
        clear ? "Discarded" : "Wrote", count, amount, clear ? "." : " to chunk.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   693
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   694
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   695
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   696
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   697
// full writer
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   698
// Assumption is retired only; exclusive access
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   699
// MutexedWriter -> ReleaseOp
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   700
//
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   701
size_t JfrStorage::write_full() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   702
  assert(_chunkwriter.is_valid(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   703
  Thread* const thread = Thread::current();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   704
  WriteOperation wo(_chunkwriter);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   705
  MutexedWriteOperation writer(wo); // a retired buffer implies mutexed access
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   706
  ReleaseOperation ro(_transient_mspace, thread);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   707
  FullOperation cmd(&writer, &ro);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   708
  const size_t count = process_full(cmd, control(), _age_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   709
  log(count, writer.processed());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   710
  return writer.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   711
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   712
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   713
size_t JfrStorage::clear_full() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   714
  DiscardOperation discarder(mutexed); // a retired buffer implies mutexed access
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   715
  const size_t count = process_full(discarder, control(), _age_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   716
  log(count, discarder.processed(), true);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   717
  return discarder.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   718
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   719
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   720
static void scavenge_log(size_t count, size_t amount, size_t current) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   721
  if (count > 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   722
    if (log_is_enabled(Debug, jfr, system)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   723
      log_debug(jfr, system)("Released " SIZE_FORMAT " dead buffer(s) of " SIZE_FORMAT" B of data.", count, amount);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   724
      log_debug(jfr, system)("Current number of dead buffers " SIZE_FORMAT "", current);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   725
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   726
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   727
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   728
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   729
template <typename Mspace>
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   730
class Scavenger {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   731
private:
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   732
  JfrStorageControl& _control;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   733
  Mspace* _mspace;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   734
  size_t _count;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   735
  size_t _amount;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   736
public:
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   737
  typedef typename Mspace::Type Type;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   738
  Scavenger(JfrStorageControl& control, Mspace* mspace) : _control(control), _mspace(mspace), _count(0), _amount(0) {}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   739
  bool process(Type* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   740
    if (t->retired()) {
54964
ec7d6d8effc7 8220293: Deadlock in JFR string pool
mgronlun
parents: 54623
diff changeset
   741
      assert(t->identity() != NULL, "invariant");
ec7d6d8effc7 8220293: Deadlock in JFR string pool
mgronlun
parents: 54623
diff changeset
   742
      assert(t->empty(), "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   743
      assert(!t->transient(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   744
      assert(!t->lease(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   745
      ++_count;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   746
      _amount += t->total_size();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   747
      t->clear_retired();
54964
ec7d6d8effc7 8220293: Deadlock in JFR string pool
mgronlun
parents: 54623
diff changeset
   748
      t->release();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   749
      _control.decrement_dead();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   750
      mspace_release_full_critical(t, _mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   751
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   752
    return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   753
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   754
  size_t processed() const { return _count; }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   755
  size_t amount() const { return _amount; }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   756
};
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   757
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   758
size_t JfrStorage::scavenge() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   759
  JfrStorageControl& ctrl = control();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   760
  if (ctrl.dead_count() == 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   761
    return 0;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   762
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   763
  Scavenger<JfrThreadLocalMspace> scavenger(ctrl, _thread_local_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   764
  process_full_list(scavenger, _thread_local_mspace);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   765
  scavenge_log(scavenger.processed(), scavenger.amount(), ctrl.dead_count());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   766
  return scavenger.processed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   767
}