2019-07-09 17:27:38 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019, Google Inc. All rights reserved.
|
2020-02-24 13:25:35 +00:00
|
|
|
* Copyright (c) 2019, 2020, Oracle and/or its affiliates. All rights reserved.
|
2019-07-09 17:27:38 +00:00
|
|
|
* 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 <jni.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2020-02-24 13:25:35 +00:00
|
|
|
#include <gnu/libc-version.h>
|
2019-07-09 17:27:38 +00:00
|
|
|
|
|
|
|
// Declare the thread local variable(s) in the main executable. This can be
|
|
|
|
// used to demonstrate the issues associated with the on-stack static TLS blocks
|
|
|
|
// that may cause insufficient stack space. The dynamic TLS blocks for shared
|
|
|
|
// objects (such as a JNI library) loaded via dlopen are not allocated on stack.
|
|
|
|
__thread int tls[128 * 1024];
|
|
|
|
|
|
|
|
JNIEnv* create_vm(JavaVM **jvm, char* argTLS) {
|
|
|
|
JNIEnv* env;
|
|
|
|
JavaVMInitArgs args;
|
|
|
|
JavaVMOption options[3];
|
|
|
|
args.version = JNI_VERSION_1_8;
|
|
|
|
args.nOptions = 3;
|
|
|
|
char classpath[4096];
|
|
|
|
snprintf(classpath, sizeof classpath,
|
|
|
|
"-Djava.class.path=%s", getenv("CLASSPATH"));
|
|
|
|
options[0].optionString = classpath;
|
|
|
|
options[1].optionString = "-Xlog:os+thread=info";
|
|
|
|
options[2].optionString = argTLS;
|
|
|
|
args.options = &options[0];
|
|
|
|
args.ignoreUnrecognized = 0;
|
|
|
|
int rv;
|
|
|
|
rv = JNI_CreateJavaVM(jvm, (void**)&env, &args);
|
|
|
|
if (rv < 0) return NULL;
|
|
|
|
return env;
|
|
|
|
}
|
|
|
|
|
2020-02-24 13:25:35 +00:00
|
|
|
// glibc 2.15 introduced __pthread_get_minstack
|
|
|
|
int glibc_has_pthread_get_minstack() {
|
|
|
|
const char* glibc_vers = gnu_get_libc_version();
|
|
|
|
const int glibc_vers_major = atoi(glibc_vers);
|
|
|
|
const int glibc_vers_minor = atoi(strchr(glibc_vers, '.') + 1);;
|
|
|
|
printf("GNU libc version: %s\n", glibc_vers);
|
|
|
|
if ((glibc_vers_major > 2) || ((glibc_vers_major == 2) && (glibc_vers_minor >= 15))) {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
printf("This version does not provide __pthread_get_minstack\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-07-09 17:27:38 +00:00
|
|
|
int run(jboolean addTLS) {
|
|
|
|
JavaVM *jvm;
|
|
|
|
jclass testClass;
|
|
|
|
jmethodID runMethod;
|
|
|
|
char* argTLS;
|
|
|
|
int res = -1;
|
|
|
|
|
|
|
|
if (addTLS) {
|
2020-02-24 13:25:35 +00:00
|
|
|
if (!glibc_has_pthread_get_minstack()) {
|
|
|
|
printf("Skipping the test.\n");
|
|
|
|
return 0;
|
|
|
|
}
|
2019-07-09 17:27:38 +00:00
|
|
|
argTLS = "-XX:+AdjustStackSizeForTLS";
|
|
|
|
} else {
|
|
|
|
argTLS = "-XX:-AdjustStackSizeForTLS"; // default
|
|
|
|
}
|
|
|
|
printf("Running test with %s ...\n", argTLS);
|
|
|
|
JNIEnv *env = create_vm(&jvm, argTLS);
|
|
|
|
|
|
|
|
// Run T.run() and check result:
|
|
|
|
// - Expect T.run() to return 'true' when stack size is adjusted for TLS,
|
|
|
|
// return 0 if so
|
|
|
|
// - Expect T.run() to return 'false' if stack size is not adjusted for
|
|
|
|
// TLS, return 0 if so
|
|
|
|
// Return -1 (fail) for other cases
|
|
|
|
testClass = (*env)->FindClass(env, "T");
|
|
|
|
runMethod = (*env)->GetStaticMethodID(env, testClass, "run", "()Z");
|
|
|
|
if ((*env)->CallStaticBooleanMethod(env, testClass, runMethod, NULL)) {
|
|
|
|
if (addTLS) {
|
|
|
|
// expect T.run() to return 'true'
|
|
|
|
res = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!addTLS) {
|
|
|
|
// expect T.run() to return 'false'
|
|
|
|
res = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (res == 0) {
|
|
|
|
printf("Test passed with %s\n", argTLS);
|
|
|
|
} else {
|
|
|
|
printf("Test failed with %s\n", argTLS);
|
|
|
|
}
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
if (argc == 2 && strcmp(argv[1], "-add_tls") == 0) {
|
|
|
|
return run(JNI_TRUE);
|
|
|
|
} else {
|
|
|
|
return run(JNI_FALSE);
|
|
|
|
}
|
|
|
|
}
|