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