365 lines
9.6 KiB
C
365 lines
9.6 KiB
C
|
/*
|
||
|
* 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;
|
||
|
}
|