691 lines
21 KiB
C++

/*
* Copyright (c) 2003, 2024, 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 <stdlib.h>
#include <stdio.h>
#include <stdarg.h>
#include <string.h>
#include <ctype.h>
/*************************************************************/
#include "jvmti.h"
/*************************************************************/
#include "nsk_tools.hpp"
#include "jni_tools.hpp"
#include "jvmti_tools.hpp"
#include "JVMTITools.hpp"
/*************************************************************/
extern "C" {
/*************************************************************/
#define NSK_JVMTI_WAITTIME 2
#define NSK_JVMTI_MAX_OPTIONS 10
#define NSK_JVMTI_OPTION_START '-'
#define NSK_JVMTI_OPT_PATH_TO_NEW_BYTE_CODE "pathToNewByteCode"
#define PATH_FORMAT "%s%02d/%s"
#define DIR_NAME "newclass"
static volatile int redefineAttempted = NSK_FALSE;
static volatile int redefineSucceed = NSK_FALSE;
static volatile int agentFailed = NSK_FALSE;
static struct {
struct {
int count;
char* names[NSK_JVMTI_MAX_OPTIONS];
char* values[NSK_JVMTI_MAX_OPTIONS];
char* string;
} options;
int waittime;
} context;
/*************************************************************/
static int check_option(int dashed, const char name[], const char value[]) {
if (strcmp("verbose", name) == 0) {
if (strlen(value) > 0) {
nsk_complain("nsk_jvmti_parseOptions(): unexpected value in option: %s=%s\n", name, value);
return NSK_FALSE;
}
nsk_setVerboseMode(NSK_TRUE);
} else if (strcmp("trace", name) == 0) {
if (strlen(value) <= 0) {
nsk_complain("nsk_jvmti_parseOptions(): no value for option: %s\n", name);
return NSK_FALSE;
}
if (strcmp("none", value) == 0) {
nsk_setTraceMode(NSK_TRACE_NONE);
} else if (strcmp("before", value) == 0) {
nsk_setTraceMode(NSK_TRACE_BEFORE);
} else if (strcmp("after", value) == 0) {
nsk_setTraceMode(NSK_TRACE_AFTER);
} else if (strcmp("all", value) == 0) {
nsk_setTraceMode(NSK_TRACE_ALL);
} else {
nsk_complain("nsk_jvmti_parseOptions(): uexpected value in option: %s=%s\n", name, value);
return NSK_FALSE;
}
nsk_setVerboseMode(NSK_TRUE);
} else if (strcmp("waittime", name) == 0) {
if (strlen(value) <= 0) {
nsk_complain("nsk_jvmti_parseOptions(): no value for option: %s\n", name);
return NSK_FALSE;
}
{
char* end = nullptr;
long n = strtol(value, &end, 10);
if (end == nullptr || end == value || *end != '\0') {
nsk_complain("nsk_jvmti_parseOptions(): not integer value in option: %s=%s\n", name, value);
return NSK_FALSE;
}
if (n < 0) {
nsk_complain("nsk_jvmti_parseOptions(): negative value in option: %s=%s\n", name, value);
return NSK_FALSE;
}
context.waittime = (int)n;
}
} else if (dashed) {
nsk_complain("nsk_jvmti_parseOptions(): unknown option: %c%s\n",
NSK_JVMTI_OPTION_START, name);
return NSK_FALSE;
}
return NSK_TRUE;
}
static int add_option(const char opt[], int opt_len, const char val[], int val_len) {
char* name;
char* value;
int success = NSK_TRUE;
int dashed_opt = NSK_FALSE;
if (opt[0] == NSK_JVMTI_OPTION_START) {
dashed_opt = NSK_TRUE;
opt++;
opt_len--;
}
if (opt_len <= 0) {
nsk_complain("nsk_jvmti_parseOptions(): found empty option\n");
return NSK_FALSE;
}
name = (char*)malloc(opt_len + 1);
value = (char*)malloc(val_len + 1);
if (name == nullptr || value == nullptr) {
nsk_complain("nsk_jvmti_parseOptions(): out of memory\n");
success = NSK_FALSE;
} else {
strncpy(name, opt, opt_len);
name[opt_len] = '\0';
if (val != nullptr) {
strncpy(value, val, val_len);
}
value[val_len] = '\0';
if (!check_option(dashed_opt, name, value)) {
success = NSK_FALSE;
}
}
if (success) {
if (context.options.count >= NSK_JVMTI_MAX_OPTIONS) {
nsk_complain("nsk_jvmti_parseOptions(): too many options for parsing\n");
success = NSK_FALSE;
} else {
context.options.names[context.options.count] = name;
context.options.values[context.options.count] = value;
context.options.count++;
}
}
if (!success) {
if (name != nullptr)
free(name);
if (value != nullptr)
free(value);
}
return success;
}
static void nsk_jvmti_free() {
if (context.options.count > 0) {
int i;
for (i = 0; i < context.options.count; i++) {
free(context.options.names[i]);
free(context.options.values[i]);
}
context.options.count = 0;
}
if (context.options.string != nullptr) {
free(context.options.string);
context.options.string = nullptr;
}
}
/*
* Tokenize a string based on a list of delimiters.
*/
static char* token(char **s, const char *delim) {
char *p;
char *start = *s;
if (s == nullptr || *s == nullptr) {
return nullptr;
}
p = strpbrk(*s, delim);
if (p != nullptr) {
/* Advance to next token. */
*p = '\0';
*s = p + 1;
} else {
/* End of tokens. */
*s = nullptr;
}
return start;
}
int nsk_jvmti_parseOptions(const char options[]) {
int success = NSK_TRUE;
char *str = nullptr;
char *name = nullptr;
char *value = nullptr;
const char *delimiters = " ,~";
if (options == nullptr)
return success;
/*
* Save a copy of the full options string for
* ArgumentHandler.getAgentOptionsString().
*/
context.options.string = strdup(options);
/* Create a temporary copy of the options string to be tokenized. */
str = strdup(options);
while ((name = token(&str, delimiters)) != nullptr) {
value = strchr(name, '=');
if (value != nullptr) {
*value++ = '\0';
}
if (!add_option(name, (int)strlen(name), value,
value ? (int)strlen(value) : 0)) {
success = NSK_FALSE;
break;
}
}
if (!success) {
nsk_jvmti_free();
}
if (str != nullptr) {
free(str);
}
return success;
}
/*************************************************************/
/**
* Returns value of given option name; or nullptr if no such option found.
* If search name is nullptr then complains an error and returns nullptr.
*/
const char* nsk_jvmti_findOptionValue(const char name[]) {
int i;
if (name == nullptr) {
nsk_complain("nsk_jvmti_findOptionValue(): option name is null\n");
return nullptr;
}
for (i = 0; i < context.options.count; i++) {
if (strcmp(name, context.options.names[i]) == 0)
return context.options.values[i];
}
return nullptr;
}
/**
* Returns string value of given option; or defaultValue if no such option found.
* If options is specified but has empty value then complains an error and returns nullptr.
*/
const char* nsk_jvmti_findOptionStringValue(const char name[], const char* defaultValue) {
const char* value;
if (name == nullptr) {
nsk_complain("nsk_jvmti_findOptionStringValue(): option name is null\n");
return nullptr;
}
value = nsk_jvmti_findOptionValue(name);
if (value == nullptr) {
return defaultValue;
}
if (strlen(value) <= 0) {
nsk_complain("nsk_jvmti_findOptionStringValue(): empty value of option: %s=%s\n",
name, value);
return nullptr;
}
return value;
}
/**
* Returns integer value of given option; or defaultValue if no such option found.
* If options is specified but has no integer value then complains an error and returns -1.
*/
int nsk_jvmti_findOptionIntValue(const char name[], int defaultValue) {
const char* value;
if (name == nullptr) {
nsk_complain("nsk_jvmti_findOptionIntValue(): option name is null\n");
return -1;
}
value = nsk_jvmti_findOptionValue(name);
if (value == nullptr) {
return defaultValue;
}
if (strlen(value) <= 0) {
nsk_complain("nsk_jvmti_findOptionIntValue(): empty value of option: %s=%s\n",
name, value);
return -1;
}
{
char* endptr = nullptr;
int n = strtol(value, &endptr, 10);
if (endptr == nullptr || *endptr != '\0') {
nsk_complain("nsk_jvmti_findOptionIntValue(): not integer value of option: %s=%s\n",
name, value);
return -1;
}
return n;
}
}
/**
* Returns number of parsed options.
*/
int nsk_jvmti_getOptionsCount() {
return context.options.count;
}
/**
* Returns name of i-th parsed option.
* If no such option then complains an error and returns nullptr.
*/
const char* nsk_jvmti_getOptionName(int i) {
if (i < 0 || i >= context.options.count) {
nsk_complain("nsk_jvmti_getOptionName(): option index out of bounds: %d\n", i);
return nullptr;
}
return context.options.names[i];
}
/**
* Returns value of i-th parsed option.
* If no such option then complains an error and returns nullptr.
*/
const char* nsk_jvmti_getOptionValue(int i) {
if (i < 0 || i >= context.options.count) {
nsk_complain("nsk_jvmti_getOptionValue(): option index out of bounds: %d\n", i);
return nullptr;
}
return context.options.values[i];
}
/*************************************************************/
/**
* Returns value of -waittime option or default value if not specified.
*/
int nsk_jvmti_getWaitTime() {
return context.waittime;
}
/**
* Sets specified waittime value.
*/
void nsk_jvmti_setWaitTime(int waittime) {
context.waittime = waittime;
}
/*************************************************************/
int nsk_jvmti_lverify(int positive, jvmtiError error, jvmtiError expected,
const char file[], int line, const char format[], ...)
{
int failure=0;
int negative = !positive;
int errorCode = (int)error;
const char* errorName = TranslateError(error);
va_list ap;
va_start(ap,format);
nsk_lvtrace(NSK_TRACE_AFTER,file,line,format,ap);
if (negative || expected != JVMTI_ERROR_NONE)
nsk_ltrace(NSK_TRACE_AFTER,file,line,
" jvmti error: code=%d, name=%s\n",errorCode,errorName);
if ((error == expected) == negative) {
nsk_lvcomplain(file,line,format,ap);
nsk_printf("# jvmti error: code=%d, name=%s\n",errorCode,errorName);
if (expected != JVMTI_ERROR_NONE)
nsk_printf("# error expected: code=%d, name=%s\n",
expected, TranslateError(expected));
failure=1;
};
va_end(ap);
return !failure;
}
/*************************************************************/
JNIEXPORT jstring JNICALL
Java_nsk_share_jvmti_ArgumentHandler_getAgentOptionsString(JNIEnv *jni, jobject obj) {
jstring str_obj = nullptr;
if (!NSK_JNI_VERIFY(jni, (str_obj = jni->NewStringUTF(context.options.string)) != nullptr)) {
return nullptr;
}
return str_obj;
}
/*************************************************************/
/**
* This method will try to redefine the class (classToRedefine) by loading
* physical file. <b>pathToNewByteCode</b> option which is passed
* on OnLoad Phase also used.
*
* So This method will do a file read pathToByteCode+fileName+.class (total path).
* Constrcuts a class objects and does a redefine of the class.
* On successfull redefine this method will return eaither JNI_TRUE or JNI_FALSE
*
* Hint::
* 1)
* If there are many redefine on same testcase, then please try to use
* integer value (newclass00, newclass01, newclass02 , ....) way.
*
* 2) When you compile these please do keep, a metatag on testcase as
* # build : native classes classes.redef
*
* 3) When you do build these classes are psysically located in build as.
*
* TESTBASE/bin/newclass0* directory.
* eg: for nks/jvmti/scenarios/hotswap/HS204/hs204t001 you should see
* TESTBASE/bin/newclass0* /nsk/hotswap/HS204/hs204t001/MyClass.class
*
*/
int nsk_jvmti_redefineClass(jvmtiEnv * jvmti,
jclass classToRedefine,
const char * fileName) {
redefineAttempted = NSK_TRUE;
if (nsk_jvmti_findOptionValue(NSK_JVMTI_OPT_PATH_TO_NEW_BYTE_CODE) == nullptr) {
nsk_printf("# error expected: %s \n", NSK_JVMTI_OPT_PATH_TO_NEW_BYTE_CODE);
nsk_printf("Hint :: missing java -agentlib:agentlib=%s=DirName, ($TESTBASE/bin) \n",
NSK_JVMTI_OPT_PATH_TO_NEW_BYTE_CODE);
return NSK_FALSE;
}
if (fileName == nullptr) {
nsk_printf("# error file name expected did not found \n");
return NSK_FALSE;
}
{
char file [1024];
//= "DEFAULT";
snprintf(file, sizeof(file), "%s/%s.class",
nsk_jvmti_findOptionValue(NSK_JVMTI_OPT_PATH_TO_NEW_BYTE_CODE),
fileName);
nsk_printf("# info :: File = %s \n",file);
{
FILE *bytecode;
unsigned char * classBytes;
jvmtiError error;
jint size;
bytecode = fopen(file, "rb");
error= JVMTI_ERROR_NONE;
if (bytecode == nullptr) {
nsk_printf("# error **Agent::error opening file %s \n",file);
return NSK_FALSE;
}
nsk_printf("# info **Agent:: opening file %s \n",file);
fseek(bytecode, 0, SEEK_END);
size = ftell(bytecode);
nsk_printf("# info file size= %ld\n",ftell(bytecode));
rewind(bytecode);
error = jvmti->Allocate(size,&classBytes);
if (error != JVMTI_ERROR_NONE) {
nsk_printf(" Failed to create memory %s \n",TranslateError(error));
return NSK_FALSE;
}
if (((jint) fread(classBytes, 1,size,bytecode)) != size) {
nsk_printf(" # error failed to read all the bytes , could be less or more \n");
return NSK_FALSE;
} else {
nsk_printf(" File red completely \n");
}
fclose(bytecode);
{
jvmtiClassDefinition classDef;
classDef.klass = classToRedefine;
classDef.class_byte_count= size;
classDef.class_bytes = classBytes;
error = jvmti->RedefineClasses(1,&classDef);
if (error != JVMTI_ERROR_NONE) {
nsk_printf("# error occured while redefining %s ",
TranslateError(error));
return NSK_FALSE;
}
}
}
}
redefineSucceed= NSK_TRUE;
return NSK_TRUE;
}
JNIEXPORT jboolean JNICALL
Java_nsk_share_jvmti_RedefineAgent_redefineAttempted(JNIEnv *jni, jobject obj) {
if (redefineAttempted) {
return JNI_TRUE;
}else {
return JNI_FALSE;
}
}
JNIEXPORT jboolean JNICALL
Java_nsk_share_jvmti_RedefineAgent_isRedefined(JNIEnv * jni, jobject obj) {
if (redefineSucceed) {
return JNI_TRUE;
}else {
return JNI_FALSE;
}
}
/**
* This jni method is a Java wrapper for agent status.
*/
JNIEXPORT jboolean JNICALL
Java_nsk_share_jvmti_RedefineAgent_agentStatus(JNIEnv * jni, jobject obj) {
if (agentFailed) {
return JNI_FALSE;
} else {
return JNI_TRUE;
}
}
void nsk_jvmti_getFileName(int redefineCnt, const char * dir, char * buf, size_t bufsize) {
snprintf(buf, bufsize, PATH_FORMAT, DIR_NAME, redefineCnt, dir);
buf[bufsize-1] = '\0';
}
int nsk_jvmti_enableNotification(jvmtiEnv *jvmti,jvmtiEvent event, jthread thread) {
jvmtiError rc=JVMTI_ERROR_NONE;
rc = jvmti->SetEventNotificationMode(JVMTI_ENABLE, event, thread);
if (rc != JVMTI_ERROR_NONE) {
nsk_printf("# error Failed to set Notification for Event \n ");
return NSK_FALSE;
}
return NSK_TRUE;
}
int nsk_jvmti_disableNotification(jvmtiEnv *jvmti,jvmtiEvent event, jthread thread) {
jvmtiError rc=JVMTI_ERROR_NONE;
rc = jvmti->SetEventNotificationMode(JVMTI_DISABLE, event, thread);
if (rc != JVMTI_ERROR_NONE) {
nsk_printf(" Failed to disaable Notification for Event ");
return NSK_FALSE;
}
return NSK_TRUE;
}
void nsk_jvmti_agentFailed() {
agentFailed = NSK_TRUE;
}
#define SLEEP_DELAY 10L
int suspendThreadAtMethod(jvmtiEnv *jvmti, jclass cls, jobject thread, jmethodID testMethod) {
printf(">>>>>>>> Invoke SuspendThread()\n");
jvmtiError err = jvmti->SuspendThread(thread);
if (err != JVMTI_ERROR_NONE) {
printf("%s: Failed to call SuspendThread(): error=%d: %s\n",
__FILE__, err, TranslateError(err));
return NSK_FALSE;
}
int result = NSK_TRUE;
jmethodID method = nullptr;
jlocation loc;
// We need to ensure that the thread is suspended at the right place when the top
// frame belongs to the test rather than to incidental Java code (classloading,
// JVMCI, etc). Below we do resume/suspend in the loop until the target method
// is executed in the top frame or the loop counter exceeds the limit.
for (int i = 0; i < 10; i++) {
err = jvmti->GetFrameLocation(thread, 0, &method, &loc);
if (err != JVMTI_ERROR_NONE) {
printf("(GetFrameLocation) unexpected error: %s (%d)\n",
TranslateError(err), err);
result = NSK_FALSE;
break;
}
char *name, *sig, *generic;
jvmti->GetMethodName(method, &name, &sig, &generic);
printf(">>> Attempt %d to suspend the thread. Top frame: \"%s%s\"\n",
i, name, sig);
if (method == testMethod) break;
err = jvmti->ResumeThread(thread);
if (err != JVMTI_ERROR_NONE) {
printf("(ResumeThread) unexpected error: %s (%d)\n",
TranslateError(err), err);
result = NSK_FALSE;
}
mssleep(SLEEP_DELAY);
err = jvmti->SuspendThread(thread);
if (err != JVMTI_ERROR_NONE) {
printf("(SuspendThread) unexpected error: %s (%d)\n",
TranslateError(err), err);
result = NSK_FALSE;
}
}
if(method == testMethod) {
printf("<<<<<<<< SuspendThread() is successfully done\n");
} else {
char *name, *sig, *generic;
jvmti->GetMethodName(testMethod, &name, &sig, &generic);
printf("Failed in the suspendThread: was not able to suspend thread "
"with required method \"%s%s\" on the top\n", name, sig);
result = NSK_FALSE;
}
return result;
}
jint createRawMonitor(jvmtiEnv *env, const char *name, jrawMonitorID *monitor) {
jvmtiError error = env->CreateRawMonitor(name, monitor);
if (!NSK_JVMTI_VERIFY(error)) {
return JNI_ERR;
}
return JNI_OK;
}
void exitOnError(jvmtiError error) {
if (!NSK_JVMTI_VERIFY(error)) {
exit(error);
}
}
void rawMonitorEnter(jvmtiEnv *env, jrawMonitorID monitor) {
jvmtiError error = env->RawMonitorEnter(monitor);
exitOnError(error);
}
void rawMonitorExit(jvmtiEnv *env, jrawMonitorID monitor) {
jvmtiError error = env->RawMonitorExit(monitor);
exitOnError(error);
}
void rawMonitorNotify(jvmtiEnv *env, jrawMonitorID monitor) {
jvmtiError error = env->RawMonitorNotify(monitor);
exitOnError(error);
}
void rawMonitorWait(jvmtiEnv *env, jrawMonitorID monitor, jlong millis) {
jvmtiError error = env->RawMonitorWait(monitor, millis);
exitOnError(error);
}
void getPhase(jvmtiEnv *env, jvmtiPhase *phase) {
jvmtiError error = env->GetPhase(phase);
exitOnError(error);
}
/*************************************************************/
}