365 lines
9.6 KiB
C
Raw Normal View History

/*
* Copyright (c) 2007, 2018, Oracle and/or its affiliates. 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 Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
#include <stdio.h>
#include <string.h>
#include <jvmti.h>
#include "agent_common.h"
#include "nsk_tools.h"
#include "JVMTITools.h"
#include "jvmti_tools.h"
#include "jni_tools.h"
#define FOO 1
#define WRAPPED_FOO 2
/* ============================================================================= */
JNIEXPORT jint JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_AutomaticResolution1_foo (
JNIEnv *jni
, jclass klass
)
{
NSK_DISPLAY0(" >>> AutomaticResolution1.foo()\n");
return FOO;
}
/* ============================================================================= */
JNIEXPORT jint JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_AutomaticResolution1_wrapped_1foo (
JNIEnv *jni
, jclass klass
)
{
NSK_DISPLAY0(" >>> AutomaticResolution1.wrapped_foo()\n");
return WRAPPED_FOO;
}
/* ============================================================================= */
JNIEXPORT jint JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_AutomaticResolution2_foo (
JNIEnv *jni
, jclass klass
)
{
NSK_DISPLAY0(" >>> AutomaticResolution2.foo()\n");
return FOO;
}
/* ============================================================================= */
JNIEXPORT jint JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_AutomaticResolution3_foo (
JNIEnv *jni
, jclass klass
)
{
NSK_DISPLAY0(" >>> AutomaticResolution3.foo()\n");
return FOO;
}
/* ============================================================================= */
JNIEXPORT jint JNICALL foo (JNIEnv *jni, jclass klass)
{
NSK_DISPLAY0(" >>> ::foo()\n");
return FOO;
}
/* ============================================================================= */
JNIEXPORT jint JNICALL wrapped_foo (JNIEnv *jni, jclass klass)
{
NSK_DISPLAY0(" >>> ::wrapped_foo()\n");
return WRAPPED_FOO;
}
/* ============================================================================= */
#define METHODS_COUNT 2
static const void *METHODS [METHODS_COUNT] = {
(void *) &foo,
(void *) &wrapped_foo,
};
/* ============================================================================= */
static jvmtiEnv *jvmti = NULL;
/* ============================================================================= */
JNIEXPORT jboolean JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_Binder_setMethodPrefix (
JNIEnv *jni
, jclass klass
, jstring prefix
)
{
jboolean result = JNI_TRUE;
char *str = NULL;
if (prefix != NULL) {
if (!NSK_VERIFY(
(str = (char *) NSK_CPP_STUB3(
GetStringUTFChars
, jni
, prefix
, 0
)
) != NULL
)
)
{ result = JNI_FALSE; goto finally; }
}
if (!NSK_JVMTI_VERIFY(
NSK_CPP_STUB2(
SetNativeMethodPrefix
, jvmti
, str
)
)
)
{ result = JNI_FALSE; goto finally; }
if (str != NULL) {
NSK_DISPLAY1("New PREFIX is set: %s\n"
, str
);
} else {
NSK_DISPLAY0("Old PREFIX is reset\n");
}
finally:
if (str != NULL) {
(*jni)->ReleaseStringUTFChars(jni, prefix, str);
}
return JNI_TRUE;
}
/* ============================================================================= */
JNIEXPORT jboolean JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_Binder_setMultiplePrefixes (
JNIEnv *jni
, jclass klass
, jstring prefix
)
{
jboolean result = JNI_TRUE;
char *str = NULL;
if (prefix != NULL) {
if (!NSK_VERIFY(
(str = (char *) NSK_CPP_STUB3(
GetStringUTFChars
, jni
, prefix
, 0
)
) != NULL
)
)
{ result = JNI_FALSE; goto finally; }
if (!NSK_JVMTI_VERIFY(
NSK_CPP_STUB3(
SetNativeMethodPrefixes
, jvmti
, 1
, (char **) &str
)
)
)
{ result = JNI_FALSE; goto finally; }
NSK_DISPLAY1("MultiplePrefixes: New PREFIX is set: %s\n"
, str
);
} else {
char* prefixes[1];
prefixes[0] = NULL;
if (!NSK_JVMTI_VERIFY(
NSK_CPP_STUB3(
SetNativeMethodPrefixes
, jvmti
, 0
, (char **)&prefixes
)
)
)
{ result = JNI_FALSE; goto finally; }
NSK_DISPLAY0("Old PREFIX is reset\n");
}
finally:
if (str != NULL) {
(*jni)->ReleaseStringUTFChars(jni, prefix, str);
}
return JNI_TRUE;
}
/* ============================================================================= */
JNIEXPORT jboolean JNICALL
Java_nsk_jvmti_SetNativeMethodPrefix_Binder_registerMethod (
JNIEnv *jni
, jclass klass
, jclass bound_klass
, jstring method_name_obj
, jstring method_sig_obj
, jint native_method_number
)
{
JNINativeMethod method;
jboolean result = JNI_FALSE;
if (native_method_number < 0 || native_method_number >= METHODS_COUNT) {
NSK_DISPLAY2("Method index is out of the bound: %d of %d"
, native_method_number
, METHODS_COUNT
);
return JNI_FALSE;
}
if (!NSK_VERIFY(
(method.name =
(char *) NSK_CPP_STUB3(
GetStringUTFChars
, jni
, method_name_obj
, 0
)
) != NULL
)
)
{
goto finally;
}
if (!NSK_VERIFY(
(method.signature =
(char *) NSK_CPP_STUB3(
GetStringUTFChars
, jni
, method_sig_obj
, 0
)
) != NULL
)
)
{
goto finally;
}
method.fnPtr = (void *) METHODS[native_method_number];
NSK_DISPLAY2(">>>> Register native method: %s %s\n"
, method.name
, method.signature
);
if ((*jni)->RegisterNatives(jni, bound_klass, (const JNINativeMethod*) &method, 1) != 0)
{
if ((*jni)->ExceptionOccurred(jni)) {
(*jni)->ExceptionClear(jni);
}
goto finally;
}
NSK_DISPLAY0("<<<< Finished native method registration\n");
result = JNI_TRUE;
finally:
if (method.name != NULL) {
(*jni)->ReleaseStringUTFChars(jni, method_name_obj, method.name);
}
if (method.signature != NULL) {
(*jni)->ReleaseStringUTFChars(jni, method_sig_obj, method.signature);
}
return result;
}
/* ============================================================================= */
/* Agent initialization procedure */
#ifdef STATIC_BUILD
JNIEXPORT jint JNICALL Agent_OnLoad_SetNativeMethodPrefix001(JavaVM *jvm, char *options, void *reserved) {
return Agent_Initialize(jvm, options, reserved);
}
JNIEXPORT jint JNICALL Agent_OnAttach_SetNativeMethodPrefix001(JavaVM *jvm, char *options, void *reserved) {
return Agent_Initialize(jvm, options, reserved);
}
JNIEXPORT jint JNI_OnLoad_SetNativeMethodPrefix001(JavaVM *jvm, char *options, void *reserved) {
return JNI_VERSION_1_8;
}
#endif
jint Agent_Initialize(JavaVM *vm, char *options, void *reserved)
{
jvmtiCapabilities caps;
if (!NSK_VERIFY(
nsk_jvmti_parseOptions(options)
)
)
return JNI_ERR;
if (!NSK_VERIFY(
(jvmti = nsk_jvmti_createJVMTIEnv(vm, reserved)) != NULL
)
)
return JNI_ERR;
if (!NSK_JVMTI_VERIFY(
NSK_CPP_STUB2(
GetCapabilities
, jvmti
, &caps)
)
)
return JNI_ERR;
// Register all necessary JVM capabilities
caps.can_set_native_method_prefix = 1;
if (!NSK_JVMTI_VERIFY(
NSK_CPP_STUB2(
AddCapabilities
, jvmti
, &caps)
)
)
return JNI_ERR;
return JNI_OK;
}