998d0baab0
Reviewed-by: jwaters, gli, coleenp, lmesnik
285 lines
8.2 KiB
C++
285 lines
8.2 KiB
C++
/*
|
|
* Copyright (c) 2008, 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 <string.h>
|
|
#include <stdlib.h>
|
|
#include <jni.h>
|
|
#include <jni_tools.hpp>
|
|
#include <nsk_tools.hpp>
|
|
#include <aod.hpp>
|
|
|
|
extern "C" {
|
|
|
|
static volatile int internalError = 0;
|
|
|
|
/*
|
|
* This function can be used to inform AOD framework that some non critical for test logic
|
|
* error happened inside shared function (e.g. JVMTI Deallocate failed).
|
|
*
|
|
* If this function was called status of all finishing AOD agents is changed to failed.
|
|
*/
|
|
|
|
void nsk_aod_internal_error() {
|
|
NSK_COMPLAIN0("WARNING: some error happened inside common function, see log for details\n");
|
|
internalError = 1;
|
|
}
|
|
|
|
void nsk_free_options(Options* options) {
|
|
if (options != nullptr) {
|
|
int i;
|
|
for (i = 0; i < NSK_AOD_MAX_OPTIONS; i++) {
|
|
if (options->names[i] != nullptr) {
|
|
free(options->names[i]);
|
|
}
|
|
if (options->values[i] != nullptr) {
|
|
free(options->values[i]);
|
|
}
|
|
}
|
|
free(options);
|
|
}
|
|
}
|
|
/*
|
|
* Work with agent options
|
|
*/
|
|
|
|
/*
|
|
* Parse options and create structure Options
|
|
*/
|
|
static Options* nsk_aod_createOptionsObject(char* optionsString) {
|
|
int i = 0;
|
|
Options* options;
|
|
char* name;
|
|
char* value;
|
|
char* sep;
|
|
|
|
if (optionsString == nullptr) {
|
|
NSK_COMPLAIN0("options were not passed to the native agent\n");
|
|
return nullptr;
|
|
}
|
|
options = (Options*) malloc(sizeof(Options));
|
|
memset(options, 0, sizeof(Options));
|
|
options->size = 0;
|
|
name = optionsString;
|
|
while (name != nullptr && i < NSK_AOD_MAX_OPTIONS) {
|
|
sep = strchr(name, '=');
|
|
if (sep == nullptr) { // name not found
|
|
NSK_COMPLAIN1("Invalid options format: '%s'\n", optionsString);
|
|
nsk_free_options(options);
|
|
return nullptr;
|
|
}
|
|
*sep = '\0';
|
|
options->names[i] = strdup(name);
|
|
value = sep + 1;
|
|
if (*value == '\0') { // value not found
|
|
NSK_COMPLAIN1("Option '%s' is empty\n", options->names[i]);
|
|
nsk_free_options(options);
|
|
return nullptr;
|
|
}
|
|
sep = strchr(value, ' ');
|
|
if (sep != nullptr) {
|
|
*sep = '\0';
|
|
name = sep + 1;
|
|
} else {
|
|
name = strchr(value, '\0');
|
|
}
|
|
options->values[i] = strdup(value);
|
|
i++;
|
|
|
|
if (*name == '\0') {
|
|
name = nullptr;
|
|
}
|
|
}
|
|
if (name != nullptr) {
|
|
NSK_COMPLAIN1("WARNING: not all options were parsed, only %d options can be specified\n",
|
|
NSK_AOD_MAX_OPTIONS);
|
|
}
|
|
options->size = i;
|
|
return options;
|
|
}
|
|
|
|
Options* nsk_aod_createOptions(char* optionsString) {
|
|
Options* options;
|
|
|
|
if (!NSK_VERIFY((options = (Options*) nsk_aod_createOptionsObject(optionsString)) != nullptr))
|
|
return nullptr;
|
|
|
|
if (!NSK_VERIFY(nsk_aod_optionSpecified(options, NSK_AOD_AGENT_NAME_OPTION))) {
|
|
NSK_COMPLAIN0("Agent name wasn't specified\n");
|
|
return nullptr;
|
|
}
|
|
|
|
/*
|
|
* verbose mode is true by default
|
|
*/
|
|
nsk_setVerboseMode(NSK_TRUE);
|
|
|
|
if (nsk_aod_optionSpecified(options, NSK_AOD_VERBOSE_OPTION)) {
|
|
if (strcmp(nsk_aod_getOptionValue(options, NSK_AOD_VERBOSE_OPTION), "false") == 0)
|
|
nsk_setVerboseMode(NSK_FALSE);
|
|
}
|
|
|
|
return options;
|
|
}
|
|
|
|
const char* nsk_aod_getOptionValue(Options* options, const char* option) {
|
|
int i;
|
|
|
|
if (!NSK_VERIFY(options != nullptr)) {
|
|
NSK_COMPLAIN0("Options null\n");
|
|
return nullptr;
|
|
}
|
|
|
|
for (i = 0; i < options->size; i++) {
|
|
if (strcmp(option, options->names[i]) == 0) {
|
|
return options->values[i];
|
|
}
|
|
}
|
|
|
|
NSK_COMPLAIN1("Option '%s' isn't defined\n", option);
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
int nsk_aod_optionSpecified(Options* options, const char* option) {
|
|
int i;
|
|
|
|
if (!NSK_VERIFY(options != nullptr)) {
|
|
NSK_COMPLAIN0("Options null\n");
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
for (i = 0; i < options->size; i++) {
|
|
if (strcmp(option, options->names[i]) == 0) {
|
|
return NSK_TRUE;
|
|
}
|
|
}
|
|
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
/*
|
|
* Agent synchronization with target application
|
|
*/
|
|
|
|
static const char* TARGET_APP_CLASS_NAME = "nsk/share/aod/TargetApplicationWaitingAgents";
|
|
|
|
static const char* AGENT_LOADED_METHOD_NAME = "agentLoaded";
|
|
static const char* AGENT_LOADED_METHOD_SIGNATURE = "(Ljava/lang/String;)V";
|
|
|
|
static const char* AGENT_FINISHED_METHOD_NAME = "agentFinished";
|
|
static const char* AGENT_FINISHED_METHOD_SIGNATURE = "(Ljava/lang/String;Z)V";
|
|
|
|
static jclass targetAppClass = nullptr;
|
|
static jmethodID agentLoadedMethod = nullptr;
|
|
static jmethodID agentFinishedMethod = nullptr;
|
|
|
|
// this function is used to notify target application that native agent has been loaded
|
|
int nsk_aod_agentLoaded(JNIEnv* jni, const char* agentName) {
|
|
jstring agentNameString;
|
|
|
|
NSK_DISPLAY1("Agent %s is loaded\n", agentName);
|
|
|
|
if (targetAppClass == nullptr) {
|
|
/*
|
|
* FindClass returns local reference, to cache reference to target application class
|
|
* global reference should be created
|
|
*/
|
|
jclass localTargetAppClass;
|
|
if (!NSK_JNI_VERIFY(jni, (localTargetAppClass =
|
|
jni->FindClass(TARGET_APP_CLASS_NAME)) != nullptr)) {
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
if (!NSK_JNI_VERIFY(jni, (targetAppClass = (jclass)
|
|
jni->NewGlobalRef(localTargetAppClass)) != nullptr)) {
|
|
return NSK_FALSE;
|
|
}
|
|
}
|
|
|
|
if (agentLoadedMethod == nullptr) {
|
|
if (!NSK_JNI_VERIFY(jni, (agentLoadedMethod =
|
|
jni->GetStaticMethodID(targetAppClass, AGENT_LOADED_METHOD_NAME, AGENT_LOADED_METHOD_SIGNATURE)) != nullptr))
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
if (!NSK_JNI_VERIFY(jni, (agentNameString =
|
|
jni->NewStringUTF(agentName)) != nullptr))
|
|
return NSK_FALSE;
|
|
|
|
jni->CallStaticVoidMethod(targetAppClass, agentLoadedMethod, agentNameString);
|
|
|
|
return NSK_TRUE;
|
|
}
|
|
|
|
// this function is used to notify target application that native agent has been finished execution
|
|
int nsk_aod_agentFinished(JNIEnv* jni, const char* agentName, int success) {
|
|
jstring agentNameString;
|
|
|
|
if (!targetAppClass) {
|
|
NSK_COMPLAIN1(
|
|
"%s: TEST LOGIC ERROR: method 'agentFinished' was called before "
|
|
"targetAppClass was initialized\n",
|
|
agentName);
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
if (internalError && success) {
|
|
success = 0;
|
|
NSK_COMPLAIN1(
|
|
"Status of agent '%s' is 'passed', but some error happened during test execution "
|
|
"(see log for details), change agent status to 'failed'\n",
|
|
agentName);
|
|
}
|
|
|
|
NSK_DISPLAY2("Agent %s finished (success: %d)\n", agentName, success);
|
|
|
|
if (agentFinishedMethod == nullptr) {
|
|
if (!NSK_JNI_VERIFY(jni, (agentFinishedMethod =
|
|
jni->GetStaticMethodID(targetAppClass, AGENT_FINISHED_METHOD_NAME, AGENT_FINISHED_METHOD_SIGNATURE)) != nullptr))
|
|
return NSK_FALSE;
|
|
}
|
|
|
|
if (!NSK_JNI_VERIFY(jni, (agentNameString = jni->NewStringUTF(agentName)) != nullptr))
|
|
return NSK_FALSE;
|
|
|
|
jni->CallStaticVoidMethod(targetAppClass, agentFinishedMethod, agentNameString, success ? JNI_TRUE : JNI_FALSE);
|
|
|
|
return NSK_TRUE;
|
|
}
|
|
|
|
/*
|
|
* Auxiliary functions
|
|
*/
|
|
|
|
// JNI env creation
|
|
|
|
JNIEnv* nsk_aod_createJNIEnv(JavaVM* vm) {
|
|
JNIEnv* jni;
|
|
vm->GetEnv((void**)&jni, JNI_VERSION_1_2);
|
|
|
|
NSK_VERIFY(jni != nullptr);
|
|
|
|
return jni;
|
|
}
|
|
|
|
}
|