view src/share/vm/gc/shenandoah/shenandoahJNICritical.cpp @ 9765:edd77b2146e0

Fix copyright header for all new Shenandoah files.
author rkennke
date Tue, 06 Oct 2015 16:37:53 +0200
parents 04773a6b9229
children
line wrap: on
line source

/*
 * Copyright (c) 2015, Red Hat, Inc. and/or its affiliates.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 *
 */

#include "gc/shenandoah/shenandoahJNICritical.hpp"
#include "gc/shenandoah/shenandoahHeap.hpp"

#include "gc/shared/gcLocker.hpp"
#include "runtime/mutexLocker.hpp"
#include "runtime/thread.hpp"
#include "runtime/vmThread.hpp"

class VM_ShenandoahJNICriticalOperation : public VM_Operation {
private:
  VM_Operation* _target;
public:
  VM_ShenandoahJNICriticalOperation(VM_Operation* target);
  VMOp_Type type() const;
  bool doit_prologue();
  void doit_epilogue();
  void doit();
  const char* name() const;
};

ShenandoahJNICritical::ShenandoahJNICritical() : _op_waiting_for_jni_critical(NULL) {
}

/*
 * This is called by the Java thread who leaves the last JNI critical block.
 */
void ShenandoahJNICritical::notify_jni_critical() {
  assert(Thread::current()->is_Java_thread(), "call only from Java thread");
  assert(_op_waiting_for_jni_critical != NULL, "must be waiting for jni critical notification");  

  MonitorLockerEx ml(ShenandoahJNICritical_lock, true);

  VMThread::execute(_op_waiting_for_jni_critical);
  _op_waiting_for_jni_critical = NULL;

  ml.notify_all();

}

/*
 * This is called by the VM thread, if it determines that the task must wait
 * for JNI critical regions to be left.
 */
void ShenandoahJNICritical::set_waiting_for_jni_before_gc(VM_Operation* op) {
  assert(Thread::current()->is_VM_thread(), "call only from VM thread");
  _op_waiting_for_jni_critical = op;
}

/**
 * This is called by the Shenandoah concurrent thread in order
 * to execute a VM_Operation on the VM thread, that needs to perform
 * a JNI critical region check.
 */
void ShenandoahJNICritical::execute_in_vm_thread(VM_Operation* op) {
  MonitorLockerEx ml(ShenandoahJNICritical_lock, true);
  VM_ShenandoahJNICriticalOperation jni_op(op);
  VMThread::execute(&jni_op);
  while (_op_waiting_for_jni_critical != NULL) {
    ml.wait(true);
  }
}


VM_ShenandoahJNICriticalOperation::VM_ShenandoahJNICriticalOperation(VM_Operation* target)
  : _target(target) {
}

VM_Operation::VMOp_Type VM_ShenandoahJNICriticalOperation::type() const {
  return _target->type();
}

const char* VM_ShenandoahJNICriticalOperation::name() const {
  return _target->name();
}

bool VM_ShenandoahJNICriticalOperation::doit_prologue() {
  return _target->doit_prologue();
}

void VM_ShenandoahJNICriticalOperation::doit_epilogue() {
  _target->doit_epilogue();
}

void VM_ShenandoahJNICriticalOperation::doit() {
  if (! GC_locker::check_active_before_gc()) {
    _target->doit();
  } else {

    if (ShenandoahTraceJNICritical) {
      gclog_or_tty->print_cr("Deferring JNI critical op because of active JNI critical regions");
    }

    // This makes the GC background thread wait, and kick off evacuation as
    // soon as JNI notifies us that critical regions have all been left.
    ShenandoahHeap *sh = ShenandoahHeap::heap();
    sh->jni_critical()->set_waiting_for_jni_before_gc(this);
  }
}