Mercurial > hg > truffle
diff src/share/vm/prims/perf.cpp @ 0:a61af66fc99e jdk7-b24
Initial load
author | duke |
---|---|
date | Sat, 01 Dec 2007 00:00:00 +0000 |
parents | |
children | c18cbe5936b8 |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/share/vm/prims/perf.cpp Sat Dec 01 00:00:00 2007 +0000 @@ -0,0 +1,313 @@ +/* + * Copyright 2001-2005 Sun Microsystems, Inc. All Rights Reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, + * CA 95054 USA or visit www.sun.com if you need additional information or + * have any questions. + * + */ + +/* + * Implementation of class sun.misc.Perf + */ + +#include "incls/_precompiled.incl" +#include "incls/_perf.cpp.incl" + + +#define PERF_ENTRY(result_type, header) \ + JVM_ENTRY(result_type, header) + +#define PERF_END JVM_END + +#define PerfWrapper(arg) /* Unimplemented at this time */ + +static char* jstr_to_utf(JNIEnv *env, jstring str, TRAPS) { + + char* utfstr = NULL; + + if (str == NULL) { + THROW_0(vmSymbols::java_lang_NullPointerException()); + //throw_new(env,"NullPointerException"); + } + + int len = env->GetStringUTFLength(str); + int unicode_len = env->GetStringLength(str); + + utfstr = NEW_RESOURCE_ARRAY(char, len + 1); + + env->GetStringUTFRegion(str, 0, unicode_len, utfstr); + + return utfstr; +} + +PERF_ENTRY(jobject, Perf_Attach(JNIEnv *env, jobject unused, jstring user, int vmid, int mode)) + + PerfWrapper("Perf_Attach"); + + char* address = 0; + size_t capacity = 0; + const char* user_utf = NULL; + + ResourceMark rm; + + { + ThreadToNativeFromVM ttnfv(thread); + + user_utf = user == NULL ? NULL : jstr_to_utf(env, user, CHECK_NULL); + } + + if (mode != PerfMemory::PERF_MODE_RO && + mode != PerfMemory::PERF_MODE_RW) { + THROW_0(vmSymbols::java_lang_IllegalArgumentException()); + } + + // attach to the PerfData memory region for the specified VM + PerfMemory::attach(user_utf, vmid, (PerfMemory::PerfMemoryMode) mode, + &address, &capacity, CHECK_NULL); + + { + ThreadToNativeFromVM ttnfv(thread); + return env->NewDirectByteBuffer(address, (jlong)capacity); + } + +PERF_END + +PERF_ENTRY(void, Perf_Detach(JNIEnv *env, jobject unused, jobject buffer)) + + PerfWrapper("Perf_Detach"); + + void* address = 0; + jlong capacity = 0; + + // get buffer address and capacity + { + ThreadToNativeFromVM ttnfv(thread); + address = env->GetDirectBufferAddress(buffer); + capacity = env->GetDirectBufferCapacity(buffer); + } + + PerfMemory::detach((char*)address, capacity, CHECK); + +PERF_END + +PERF_ENTRY(jobject, Perf_CreateLong(JNIEnv *env, jobject perf, jstring name, + int variability, int units, jlong value)) + + PerfWrapper("Perf_CreateLong"); + + char* name_utf = NULL; + + if (units <= 0 || units > PerfData::U_Last) { + debug_only(warning("unexpected units argument, units = %d", units)); + THROW_0(vmSymbols::java_lang_IllegalArgumentException()); + } + + ResourceMark rm; + + { + ThreadToNativeFromVM ttnfv(thread); + + name_utf = jstr_to_utf(env, name, CHECK_NULL); + } + + PerfLong* pl = NULL; + + // check that the PerfData name doesn't already exist + if (PerfDataManager::exists(name_utf)) { + THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "PerfLong name already exists"); + } + + switch(variability) { + case 1: /* V_Constant */ + pl = PerfDataManager::create_long_constant(NULL_NS, (char *)name_utf, + (PerfData::Units)units, value, + CHECK_NULL); + break; + + case 2: /* V_Variable */ + pl = PerfDataManager::create_long_variable(NULL_NS, (char *)name_utf, + (PerfData::Units)units, value, + CHECK_NULL); + break; + + case 3: /* V_Monotonic Counter */ + pl = PerfDataManager::create_long_counter(NULL_NS, (char *)name_utf, + (PerfData::Units)units, value, + CHECK_NULL); + break; + + default: /* Illegal Argument */ + debug_only(warning("unexpected variability value: %d", variability)); + THROW_0(vmSymbols::java_lang_IllegalArgumentException()); + break; + } + + long* lp = (long*)pl->get_address(); + + { + ThreadToNativeFromVM ttnfv(thread); + return env->NewDirectByteBuffer(lp, sizeof(jlong)); + } + +PERF_END + +PERF_ENTRY(jobject, Perf_CreateByteArray(JNIEnv *env, jobject perf, + jstring name, jint variability, + jint units, jbyteArray value, + jint maxlength)) + + PerfWrapper("Perf_CreateByteArray"); + + // check for valid byte array objects + if (name == NULL || value == NULL) { + THROW_0(vmSymbols::java_lang_NullPointerException()); + } + + // check for valid variability classification + if (variability != PerfData::V_Constant && + variability != PerfData::V_Variable) { + debug_only(warning("unexpected variability value: %d", variability)); + THROW_0(vmSymbols::java_lang_IllegalArgumentException()); + } + + // check for valid units + if (units != PerfData::U_String) { + // only String based ByteArray objects are currently supported + debug_only(warning("unexpected units value: %d", variability)); + THROW_0(vmSymbols::java_lang_IllegalArgumentException()); + } + + int value_length; + char* name_utf = NULL; + jbyte* value_local = NULL; + + ResourceMark rm; + + { + ThreadToNativeFromVM ttnfv(thread); + + name_utf = jstr_to_utf(env, name, CHECK_NULL); + + value_length = env->GetArrayLength(value); + + value_local = NEW_RESOURCE_ARRAY(jbyte, value_length + 1); + + env->GetByteArrayRegion(value, 0, value_length, value_local); + } + + // check that the counter name doesn't already exist + if (PerfDataManager::exists((char*)name_utf)) { + THROW_MSG_0(vmSymbols::java_lang_IllegalArgumentException(), "PerfByteArray name already exists"); + } + + PerfByteArray* pbv = NULL; + + if (units == PerfData::U_String) { + + if (variability == PerfData::V_Constant) { + // create the string constant + pbv = PerfDataManager::create_string_constant(NULL_NS, (char*)name_utf, + (char*)value_local, + CHECK_NULL); + + assert(maxlength == value_length, "string constant length should be == maxlength"); + maxlength = value_length; + } + else { + + // create the string variable + pbv = PerfDataManager::create_string_variable(NULL_NS, (char*)name_utf, + maxlength, + (char*)value_local, + CHECK_NULL); + + assert(maxlength >= value_length,"string variable length should be <= maxlength"); + } + } + + char* cp = (char*)pbv->get_address(); + + { + ThreadToNativeFromVM ttnfv(thread); + return env->NewDirectByteBuffer(cp, maxlength+1); + } + +PERF_END + +PERF_ENTRY(jlong, Perf_HighResCounter(JNIEnv *env, jobject perf)) + + PerfWrapper("Perf_HighResCounter"); + + // this should be a method in java.lang.System. This value could + // be acquired through access to a PerfData performance counter, but + // doing so would require that the PerfData monitoring overhead be + // incurred by all Java applications, which is unacceptable. + + return os::elapsed_counter(); + +PERF_END + +PERF_ENTRY(jlong, Perf_HighResFrequency(JNIEnv *env, jobject perf)) + + PerfWrapper("Perf_HighResFrequency"); + + // this should be a method in java.lang.System. This value could + // be acquired through access to a PerfData performance counter, but + // doing so would require that the PerfData monitoring overhead be + // incurred by all Java applications, which is unacceptable. + + return os::elapsed_frequency(); + +PERF_END + +/// JVM_RegisterPerfMethods + +#define CC (char*) /*cast a literal from (const char*)*/ +#define FN_PTR(f) CAST_FROM_FN_PTR(void*, &f) +#define BB "Ljava/nio/ByteBuffer;" +#define JLS "Ljava/lang/String;" +#define CL_ARGS CC"("JLS"IIJ)"BB +#define CBA_ARGS CC"("JLS"II[BI)"BB + +static JNINativeMethod perfmethods[] = { + + {CC"attach", CC"("JLS"II)"BB, FN_PTR(Perf_Attach)}, + {CC"detach", CC"("BB")V", FN_PTR(Perf_Detach)}, + {CC"createLong", CL_ARGS, FN_PTR(Perf_CreateLong)}, + {CC"createByteArray", CBA_ARGS, FN_PTR(Perf_CreateByteArray)}, + {CC"highResCounter", CC"()J", FN_PTR(Perf_HighResCounter)}, + {CC"highResFrequency", CC"()J", FN_PTR(Perf_HighResFrequency)} +}; + +#undef CBA_ARGS +#undef CL_ARGS +#undef JLS +#undef BB +#undef FN_PTR +#undef CC + +// This one function is exported, used by NativeLookup. +JVM_ENTRY(void, JVM_RegisterPerfMethods(JNIEnv *env, jclass perfclass)) + PerfWrapper("JVM_RegisterPerfMethods"); + { + ThreadToNativeFromVM ttnfv(thread); + int ok = env->RegisterNatives(perfclass, perfmethods, sizeof(perfmethods)/sizeof(JNINativeMethod)); + guarantee(ok == 0, "register perf natives"); + } +JVM_END