src/hotspot/share/jfr/writers/jfrJavaEventWriter.cpp
author mgronlun
Sat, 24 Aug 2019 14:30:27 +0200
branchJEP-349-branch
changeset 57870 00860d9caf4d
parent 54847 59ea39bb2809
permissions -rw-r--r--
New metadata system for oldobjects built on top of simplified tagging model. Caching and serialization improvements. Flushpoint checkpoint with chunkheader contents.
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     1
/*
53582
881c5fbeb849 8218041: Assorted wrong/missing includes
rehn
parents: 51467
diff changeset
     2
 * Copyright (c) 2016, 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 "jni.h"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    27
#include "classfile/symbolTable.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    28
#include "classfile/systemDictionary.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    29
#include "classfile/vmSymbols.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    30
#include "jfr/jni/jfrJavaSupport.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    31
#include "jfr/recorder/storage/jfrStorage.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    32
#include "jfr/support/jfrThreadId.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    33
#include "jfr/utilities/jfrTypes.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    34
#include "jfr/writers/jfrJavaEventWriter.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    35
#include "oops/instanceKlass.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    36
#include "oops/oop.inline.hpp"
51467
12997ebbc0d8 8209647: constantPoolHandle::constantPoolHandle(ConstantPool*) when precompiled header is disabled
iklam
parents: 50113
diff changeset
    37
#include "runtime/fieldDescriptor.inline.hpp"
53582
881c5fbeb849 8218041: Assorted wrong/missing includes
rehn
parents: 51467
diff changeset
    38
#include "runtime/handles.inline.hpp"
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    39
#include "runtime/jniHandles.inline.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    40
#include "runtime/thread.inline.hpp"
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    41
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    42
static int start_pos_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    43
static int start_pos_address_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    44
static int current_pos_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    45
static int max_pos_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    46
static int max_event_size_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    47
static int notified_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    48
static int thread_id_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    49
static int valid_offset = invalid_offset;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    50
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    51
static bool find_field(InstanceKlass* ik,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    52
                       Symbol* name_symbol,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    53
                       Symbol* signature_symbol,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    54
                       fieldDescriptor* fd,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    55
                       bool is_static = false,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    56
                       bool allow_super = false) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    57
  if (allow_super || is_static) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    58
    return ik->find_field(name_symbol, signature_symbol, is_static, fd) != NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    59
  } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    60
    return ik->find_local_field(name_symbol, signature_symbol, fd);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    61
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    62
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    63
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    64
static void compute_offset(int &dest_offset,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    65
                           Klass* klass,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    66
                           Symbol* name_symbol,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    67
                           Symbol* signature_symbol,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    68
                           bool is_static = false, bool allow_super = false) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    69
  fieldDescriptor fd;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    70
  InstanceKlass* ik = InstanceKlass::cast(klass);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    71
  if (!find_field(ik, name_symbol, signature_symbol, &fd, is_static, allow_super)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    72
    assert(false, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    73
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    74
  dest_offset = fd.offset();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    75
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    76
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    77
static bool setup_event_writer_offsets(TRAPS) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    78
  const char class_name[] = "jdk/jfr/internal/EventWriter";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
    79
  Symbol* const k_sym = SymbolTable::new_symbol(class_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    80
  assert(k_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    81
  Klass* klass = SystemDictionary::resolve_or_fail(k_sym, true, CHECK_false);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    82
  assert(klass != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    83
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    84
  const char start_pos_name[] = "startPosition";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
    85
  Symbol* const start_pos_sym = SymbolTable::new_symbol(start_pos_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    86
  assert(start_pos_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    87
  assert(invalid_offset == start_pos_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    88
  compute_offset(start_pos_offset, klass, start_pos_sym, vmSymbols::long_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    89
  assert(start_pos_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    90
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    91
  const char start_pos_address_name[] = "startPositionAddress";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
    92
  Symbol* const start_pos_address_sym = SymbolTable::new_symbol(start_pos_address_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    93
  assert(start_pos_address_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    94
  assert(invalid_offset == start_pos_address_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    95
  compute_offset(start_pos_address_offset, klass, start_pos_address_sym, vmSymbols::long_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    96
  assert(start_pos_address_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    97
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    98
  const char event_pos_name[] = "currentPosition";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
    99
  Symbol* const event_pos_sym = SymbolTable::new_symbol(event_pos_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   100
  assert(event_pos_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   101
  assert(invalid_offset == current_pos_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   102
  compute_offset(current_pos_offset, klass, event_pos_sym,vmSymbols::long_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   103
  assert(current_pos_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   104
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   105
  const char max_pos_name[] = "maxPosition";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
   106
  Symbol* const max_pos_sym = SymbolTable::new_symbol(max_pos_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   107
  assert(max_pos_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   108
  assert(invalid_offset == max_pos_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   109
  compute_offset(max_pos_offset, klass, max_pos_sym, vmSymbols::long_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   110
  assert(max_pos_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   111
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   112
  const char max_event_size_name[] = "maxEventSize";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
   113
  Symbol* const max_event_size_sym = SymbolTable::new_symbol(max_event_size_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   114
  assert (max_event_size_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   115
  assert(invalid_offset == max_event_size_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   116
  compute_offset(max_event_size_offset, klass, max_event_size_sym, vmSymbols::int_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   117
  assert(max_event_size_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   118
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   119
  const char notified_name[] = "notified";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
   120
  Symbol* const notified_sym = SymbolTable::new_symbol(notified_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   121
  assert (notified_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   122
  assert(invalid_offset == notified_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   123
  compute_offset(notified_offset, klass, notified_sym, vmSymbols::bool_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   124
  assert(notified_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   125
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   126
  const char valid_name[] = "valid";
54847
59ea39bb2809 8223657: Remove unused THREAD argument from SymbolTable functions
coleenp
parents: 53582
diff changeset
   127
  Symbol* const valid_sym = SymbolTable::new_symbol(valid_name);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   128
  assert (valid_sym != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   129
  assert(invalid_offset == valid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   130
  compute_offset(valid_offset, klass, valid_sym, vmSymbols::bool_signature());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   131
  assert(valid_offset != invalid_offset, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   132
  return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   133
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   134
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   135
bool JfrJavaEventWriter::initialize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   136
  static bool initialized = false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   137
  if (!initialized) {
57870
00860d9caf4d New metadata system for oldobjects built on top of simplified tagging model. Caching and serialization improvements. Flushpoint checkpoint with chunkheader contents.
mgronlun
parents: 54847
diff changeset
   138
    initialized = setup_event_writer_offsets(Thread::current());
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   139
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   140
  return initialized;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   141
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   142
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   143
jboolean JfrJavaEventWriter::flush(jobject writer, jint used, jint requested, JavaThread* jt) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   144
  DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(jt));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   145
  assert(writer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   146
  oop const w = JNIHandles::resolve_non_null(writer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   147
  assert(w != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   148
  JfrBuffer* const current = jt->jfr_thread_local()->java_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   149
  assert(current != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   150
  JfrBuffer* const buffer = JfrStorage::flush(current, used, requested, false, jt);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   151
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   152
  // "validity" is contextually defined here to mean
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   153
  // that some memory location was provided that is
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   154
  // large enough to accommodate the "requested size".
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   155
  const bool is_valid = buffer->free_size() >= (size_t)(used + requested);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   156
  u1* const new_current_position = is_valid ? buffer->pos() + used : buffer->pos();
57870
00860d9caf4d New metadata system for oldobjects built on top of simplified tagging model. Caching and serialization improvements. Flushpoint checkpoint with chunkheader contents.
mgronlun
parents: 54847
diff changeset
   157
  assert(start_pos_offset != invalid_offset, "invariant");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   158
  w->long_field_put(start_pos_offset, (jlong)buffer->pos());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   159
  w->long_field_put(current_pos_offset, (jlong)new_current_position);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   160
  // only update java writer if underlying memory changed
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   161
  if (buffer != current) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   162
    w->long_field_put(start_pos_address_offset, (jlong)buffer->pos_address());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   163
    w->long_field_put(max_pos_offset, (jlong)buffer->end());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   164
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   165
  if (!is_valid) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   166
    // mark writer as invalid for this write attempt
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   167
    w->release_bool_field_put(valid_offset, JNI_FALSE);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   168
    return JNI_FALSE;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   169
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   170
  // An exclusive use of a leased buffer is treated equivalent to
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   171
  // holding a system resource. As such, it should be released as soon as possible.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   172
  // Returning true here signals that the thread will need to call flush again
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   173
  // on EventWriter.endEvent() and that flush will return the lease.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   174
  return buffer->lease() ? JNI_TRUE : JNI_FALSE;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   175
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   176
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   177
class JfrJavaEventWriterNotificationClosure : public ThreadClosure {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   178
 public:
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   179
   void do_thread(Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   180
     if (t->is_Java_thread()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   181
       JfrJavaEventWriter::notify((JavaThread*)t);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   182
     }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   183
   }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   184
};
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   185
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   186
void JfrJavaEventWriter::notify() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   187
  assert(SafepointSynchronize::is_at_safepoint(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   188
  JfrJavaEventWriterNotificationClosure closure;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   189
  Threads::threads_do(&closure);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   190
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   191
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   192
void JfrJavaEventWriter::notify(JavaThread* jt) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   193
  assert(jt != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   194
  assert(SafepointSynchronize::is_at_safepoint(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   195
  if (jt->jfr_thread_local()->has_java_event_writer()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   196
    oop buffer_writer = JNIHandles::resolve_non_null(jt->jfr_thread_local()->java_event_writer());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   197
    assert(buffer_writer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   198
    buffer_writer->release_bool_field_put(notified_offset, JNI_TRUE);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   199
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   200
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   201
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   202
static jobject create_new_event_writer(JfrBuffer* buffer, TRAPS) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   203
  assert(buffer != NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   204
  DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(THREAD));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   205
  HandleMark hm(THREAD);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   206
  static const char klass[] = "jdk/jfr/internal/EventWriter";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   207
  static const char method[] = "<init>";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   208
  static const char signature[] = "(JJJJZ)V";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   209
  JavaValue result(T_OBJECT);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   210
  JfrJavaArguments args(&result, klass, method, signature, CHECK_NULL);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   211
  // parameters
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   212
  args.push_long((jlong)buffer->pos());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   213
  args.push_long((jlong)buffer->end());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   214
  args.push_long((jlong)buffer->pos_address());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   215
  args.push_long((jlong)JFR_THREAD_ID(THREAD));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   216
  args.push_int((int)JNI_TRUE);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   217
  JfrJavaSupport::new_object_global_ref(&args, CHECK_NULL);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   218
  return result.get_jobject();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   219
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   220
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   221
jobject JfrJavaEventWriter::event_writer(Thread* t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   222
  DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(t));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   223
  JfrThreadLocal* const tl = t->jfr_thread_local();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   224
  assert(tl->shelved_buffer() == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   225
  return tl->java_event_writer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   226
}
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   227
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   228
jobject JfrJavaEventWriter::new_event_writer(TRAPS) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   229
  DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(THREAD));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   230
  assert(event_writer(THREAD) == NULL, "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   231
  JfrThreadLocal* const tl = THREAD->jfr_thread_local();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   232
  assert(!tl->has_java_buffer(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   233
  JfrBuffer* const buffer = tl->java_buffer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   234
  if (buffer == NULL) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   235
    JfrJavaSupport::throw_out_of_memory_error("OOME for thread local buffer", THREAD);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   236
    return NULL;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   237
  }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   238
  jobject java_event_writer = create_new_event_writer(buffer, CHECK_NULL);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   239
  tl->set_java_event_writer(java_event_writer);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   240
  assert(tl->has_java_event_writer(), "invariant");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   241
  return java_event_writer;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   242
}