From b577a431bd0e7eaf05c917a5a897556feeb32b5c Mon Sep 17 00:00:00 2001 From: Jaroslav Bachorik Date: Fri, 6 Nov 2015 14:34:06 +0100 Subject: [PATCH 1/7] 8141526: Allow to collect stdout/stderr from the FinalizationRunner even before the process returns Reviewed-by: dsamersoff --- .../dcmd/gc/RunFinalizationTest.java | 25 +++++++++++++++---- .../testlibrary/jdk/test/lib/Asserts.java | 3 +++ .../jdk/test/lib/JDKToolFinder.java | 5 ++++ .../jdk/test/lib/JDKToolLauncher.java | 3 +++ .../jdk/test/lib/OutputAnalyzer.java | 4 +++ .../jdk/test/lib/OutputBuffer.java | 5 ++++ .../testlibrary/jdk/test/lib/Platform.java | 5 ++++ .../jdk/test/lib/ProcessTools.java | 5 ++++ .../jdk/test/lib/StreamPumper.java | 5 ++++ .../test/testlibrary/jdk/test/lib/Utils.java | 4 +++ 10 files changed, 59 insertions(+), 5 deletions(-) diff --git a/hotspot/test/serviceability/dcmd/gc/RunFinalizationTest.java b/hotspot/test/serviceability/dcmd/gc/RunFinalizationTest.java index dcc2450757c..485ecc65ba8 100644 --- a/hotspot/test/serviceability/dcmd/gc/RunFinalizationTest.java +++ b/hotspot/test/serviceability/dcmd/gc/RunFinalizationTest.java @@ -23,20 +23,22 @@ import java.util.ArrayList; import java.util.List; +import java.util.concurrent.atomic.AtomicBoolean; -import jdk.test.lib.OutputAnalyzer; -import jdk.test.lib.ProcessTools; +import jdk.test.lib.process.ProcessTools; /* * @test * @summary Test of diagnostic command GC.run_finalization * @library /testlibrary + * @library /test/lib/share/classes * @modules java.base/sun.misc * java.compiler * java.management * jdk.jvmstat/sun.jvmstat.monitor * @build jdk.test.lib.* * @build jdk.test.lib.dcmd.* + * @build jdk.test.lib.process.* * @build RunFinalizationTest FinalizationRunner * @run main RunFinalizationTest */ @@ -50,8 +52,21 @@ public class RunFinalizationTest { javaArgs.add(TEST_APP_NAME); ProcessBuilder testAppPb = ProcessTools.createJavaProcessBuilder(javaArgs.toArray(new String[javaArgs.size()])); - OutputAnalyzer out = ProcessTools.executeProcess(testAppPb); - out.stderrShouldNotMatch("^" + FinalizationRunner.FAILED + ".*") - .stdoutShouldMatch("^" + FinalizationRunner.PASSED + ".*"); + final AtomicBoolean failed = new AtomicBoolean(); + final AtomicBoolean passed = new AtomicBoolean(); + + Process runner = ProcessTools.startProcess( + "FinalizationRunner", + testAppPb, + l -> { + failed.compareAndSet(false, l.contains(FinalizationRunner.FAILED)); + passed.compareAndSet(false, l.contains(FinalizationRunner.PASSED)); + } + ); + runner.waitFor(); + + if (failed.get() || !passed.get()) { + throw new Error("RunFinalizationTest failed"); + } } } diff --git a/hotspot/test/testlibrary/jdk/test/lib/Asserts.java b/hotspot/test/testlibrary/jdk/test/lib/Asserts.java index 13dbddb03be..c07a3f38fe5 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/Asserts.java +++ b/hotspot/test/testlibrary/jdk/test/lib/Asserts.java @@ -41,7 +41,10 @@ package jdk.test.lib; * multiple times, then the line number won't provide enough context to * understand the failure. * + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib} */ +@Deprecated public class Asserts { /** diff --git a/hotspot/test/testlibrary/jdk/test/lib/JDKToolFinder.java b/hotspot/test/testlibrary/jdk/test/lib/JDKToolFinder.java index 3ad008e0005..dc924bbfc39 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/JDKToolFinder.java +++ b/hotspot/test/testlibrary/jdk/test/lib/JDKToolFinder.java @@ -27,6 +27,11 @@ import java.io.FileNotFoundException; import java.nio.file.Path; import java.nio.file.Paths; +/** + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib} + */ +@Deprecated public final class JDKToolFinder { private JDKToolFinder() { diff --git a/hotspot/test/testlibrary/jdk/test/lib/JDKToolLauncher.java b/hotspot/test/testlibrary/jdk/test/lib/JDKToolLauncher.java index 3705834a663..eba1859121a 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/JDKToolLauncher.java +++ b/hotspot/test/testlibrary/jdk/test/lib/JDKToolLauncher.java @@ -46,7 +46,10 @@ import java.util.List; * Process p = pb.start(); * } * + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib} */ +@Deprecated public class JDKToolLauncher { private final String executable; private final List vmArgs = new ArrayList(); diff --git a/hotspot/test/testlibrary/jdk/test/lib/OutputAnalyzer.java b/hotspot/test/testlibrary/jdk/test/lib/OutputAnalyzer.java index ff64d9af78a..37f0209b288 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/OutputAnalyzer.java +++ b/hotspot/test/testlibrary/jdk/test/lib/OutputAnalyzer.java @@ -41,7 +41,11 @@ public final class OutputAnalyzer { * * @param process Process to analyze * @throws IOException If an I/O error occurs. + * + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib/process} */ + @Deprecated public OutputAnalyzer(Process process) throws IOException { OutputBuffer output = ProcessTools.getOutput(process); exitValue = process.exitValue(); diff --git a/hotspot/test/testlibrary/jdk/test/lib/OutputBuffer.java b/hotspot/test/testlibrary/jdk/test/lib/OutputBuffer.java index f073619b461..31ad53b6b56 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/OutputBuffer.java +++ b/hotspot/test/testlibrary/jdk/test/lib/OutputBuffer.java @@ -23,6 +23,11 @@ package jdk.test.lib; +/** + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib/process} + */ +@Deprecated public class OutputBuffer { private final String stdout; private final String stderr; diff --git a/hotspot/test/testlibrary/jdk/test/lib/Platform.java b/hotspot/test/testlibrary/jdk/test/lib/Platform.java index 0980e5cc04d..14f1294715b 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/Platform.java +++ b/hotspot/test/testlibrary/jdk/test/lib/Platform.java @@ -25,6 +25,11 @@ package jdk.test.lib; import java.util.regex.Pattern; +/** + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib} + */ +@Deprecated public class Platform { private static final String osName = System.getProperty("os.name"); private static final String dataModel = System.getProperty("sun.arch.data.model"); diff --git a/hotspot/test/testlibrary/jdk/test/lib/ProcessTools.java b/hotspot/test/testlibrary/jdk/test/lib/ProcessTools.java index 295c793c9b7..70e990ac59d 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/ProcessTools.java +++ b/hotspot/test/testlibrary/jdk/test/lib/ProcessTools.java @@ -31,6 +31,11 @@ import java.util.ArrayList; import java.util.Collections; import java.util.List; +/** + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib/process} + */ +@Deprecated public final class ProcessTools { private ProcessTools() { diff --git a/hotspot/test/testlibrary/jdk/test/lib/StreamPumper.java b/hotspot/test/testlibrary/jdk/test/lib/StreamPumper.java index 35e58af9b4b..399ecb179ce 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/StreamPumper.java +++ b/hotspot/test/testlibrary/jdk/test/lib/StreamPumper.java @@ -27,6 +27,11 @@ import java.io.OutputStream; import java.io.InputStream; import java.io.IOException; +/** + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib/process} + */ +@Deprecated public final class StreamPumper implements Runnable { private static final int BUF_SIZE = 256; diff --git a/hotspot/test/testlibrary/jdk/test/lib/Utils.java b/hotspot/test/testlibrary/jdk/test/lib/Utils.java index 6e9196a7c4e..9dcf0632ffe 100644 --- a/hotspot/test/testlibrary/jdk/test/lib/Utils.java +++ b/hotspot/test/testlibrary/jdk/test/lib/Utils.java @@ -55,7 +55,11 @@ import sun.misc.Unsafe; /** * Common library for various test helper functions. + * + * @deprecated This class is deprecated. Use the one from + * {@code /test/lib/share/classes/jdk/test/lib} */ +@Deprecated public final class Utils { /** From 153eae9ba486fb02906249002a43d1ce84c3ced1 Mon Sep 17 00:00:00 2001 From: Thomas Schatzl Date: Tue, 1 Dec 2015 16:46:55 +0100 Subject: [PATCH 2/7] 8136680: Enable adaptive IHOP by default Reviewed-by: jmasa, mgerdin, ehelin --- hotspot/src/share/vm/gc/g1/g1_globals.hpp | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/hotspot/src/share/vm/gc/g1/g1_globals.hpp b/hotspot/src/share/vm/gc/g1/g1_globals.hpp index 6f7ac1fe840..44501e7fe33 100644 --- a/hotspot/src/share/vm/gc/g1/g1_globals.hpp +++ b/hotspot/src/share/vm/gc/g1/g1_globals.hpp @@ -33,9 +33,11 @@ #define G1_FLAGS(develop, develop_pd, product, product_pd, diagnostic, experimental, notproduct, manageable, product_rw, range, constraint) \ \ - product(bool, G1UseAdaptiveIHOP, false, \ - "Adaptively adjust InitiatingHeapOccupancyPercent from the " \ - "initial value.") \ + product(bool, G1UseAdaptiveIHOP, true, \ + "Adaptively adjust the initiating heap occupancy from the " \ + "initial value of InitiatingHeapOccupancyPercent. The policy " \ + "attempts to start marking in time based on application " \ + "behavior.") \ \ experimental(size_t, G1AdaptiveIHOPNumInitialSamples, 3, \ "How many completed time periods from initial mark to first " \ From a3d8114b45bdfcb02b0d28ac7d2ad339898b0bc9 Mon Sep 17 00:00:00 2001 From: Max Ockner Date: Wed, 2 Dec 2015 14:07:58 -0500 Subject: [PATCH 3/7] 8142976: Reimplement TraceClassInitialization with Unified Logging TraceClassInitialization logging reimplemented with Unified Logging. Reviewed-by: coleenp, hseigel --- hotspot/src/share/vm/classfile/verifier.cpp | 46 ++++-- hotspot/src/share/vm/classfile/verifier.hpp | 1 + hotspot/src/share/vm/logging/logTag.hpp | 1 + hotspot/src/share/vm/oops/instanceKlass.cpp | 15 +- hotspot/src/share/vm/runtime/globals.hpp | 3 - hotspot/test/runtime/logging/BadMap50.jasm | 156 ++++++++++++++++++ .../logging/ClassInitializationTest.java | 62 +++++++ 7 files changed, 257 insertions(+), 27 deletions(-) create mode 100644 hotspot/test/runtime/logging/BadMap50.jasm create mode 100644 hotspot/test/runtime/logging/ClassInitializationTest.java diff --git a/hotspot/src/share/vm/classfile/verifier.cpp b/hotspot/src/share/vm/classfile/verifier.cpp index 1ec5d5bddeb..cc6f22795e6 100644 --- a/hotspot/src/share/vm/classfile/verifier.cpp +++ b/hotspot/src/share/vm/classfile/verifier.cpp @@ -48,6 +48,7 @@ #include "runtime/thread.hpp" #include "services/threadService.hpp" #include "utilities/bytes.hpp" +#include "logging/log.hpp" #define NOFAILOVER_MAJOR_VERSION 51 #define NONZERO_PADDING_BYTES_IN_SWITCH_MAJOR_VERSION 51 @@ -111,6 +112,18 @@ void Verifier::trace_class_resolution(Klass* resolve_class, InstanceKlass* verif } } +// Prints the end-verification message to the appropriate output. +void Verifier::log_end_verification(outputStream* st, const char* klassName, Symbol* exception_name, TRAPS) { + if (HAS_PENDING_EXCEPTION) { + st->print("Verification for %s has", klassName); + st->print_cr(" exception pending %s ", + PENDING_EXCEPTION->klass()->external_name()); + } else if (exception_name != NULL) { + st->print_cr("Verification for %s failed", klassName); + } + st->print_cr("End class verification for: %s", klassName); +} + bool Verifier::verify(instanceKlassHandle klass, Verifier::Mode mode, bool should_verify_class, TRAPS) { HandleMark hm; ResourceMark rm(THREAD); @@ -155,9 +168,7 @@ bool Verifier::verify(instanceKlassHandle klass, Verifier::Mode mode, bool shoul bool can_failover = FailOverToOldVerifier && klass->major_version() < NOFAILOVER_MAJOR_VERSION; - if (TraceClassInitialization) { - tty->print_cr("Start class verification for: %s", klassName); - } + log_info(classinit)("Start class verification for: %s", klassName); if (klass->major_version() >= STACKMAP_ATTRIBUTE_MAJOR_VERSION) { ClassVerifier split_verifier(klass, THREAD); split_verifier.verify_class(THREAD); @@ -165,10 +176,10 @@ bool Verifier::verify(instanceKlassHandle klass, Verifier::Mode mode, bool shoul if (can_failover && !HAS_PENDING_EXCEPTION && (exception_name == vmSymbols::java_lang_VerifyError() || exception_name == vmSymbols::java_lang_ClassFormatError())) { - if (TraceClassInitialization || VerboseVerification) { - tty->print_cr( - "Fail over class verification to old verifier for: %s", klassName); + if (VerboseVerification) { + tty->print_cr("Fail over class verification to old verifier for: %s", klassName); } + log_info(classinit)("Fail over class verification to old verifier for: %s", klassName); exception_name = inference_verify( klass, message_buffer, message_buffer_len, THREAD); } @@ -180,15 +191,11 @@ bool Verifier::verify(instanceKlassHandle klass, Verifier::Mode mode, bool shoul klass, message_buffer, message_buffer_len, THREAD); } - if (TraceClassInitialization || VerboseVerification) { - if (HAS_PENDING_EXCEPTION) { - tty->print("Verification for %s has", klassName); - tty->print_cr(" exception pending %s ", - PENDING_EXCEPTION->klass()->external_name()); - } else if (exception_name != NULL) { - tty->print_cr("Verification for %s failed", klassName); - } - tty->print_cr("End class verification for: %s", klassName); + if (log_is_enabled(Info, classinit)){ + log_end_verification(LogHandle(classinit)::info_stream(), klassName, exception_name, THREAD); + } + if (VerboseVerification){ + log_end_verification(tty, klassName, exception_name, THREAD); } if (HAS_PENDING_EXCEPTION) { @@ -598,10 +605,13 @@ void ClassVerifier::verify_class(TRAPS) { verify_method(methodHandle(THREAD, m), CHECK_VERIFY(this)); } - if (VerboseVerification || TraceClassInitialization) { - if (was_recursively_verified()) + if (was_recursively_verified()){ + if (VerboseVerification){ tty->print_cr("Recursive verification detected for: %s", - _klass->external_name()); + _klass->external_name()); + } + log_info(classinit)("Recursive verification detected for: %s", + _klass->external_name()); } } diff --git a/hotspot/src/share/vm/classfile/verifier.hpp b/hotspot/src/share/vm/classfile/verifier.hpp index e265355e733..ce3d9beaa10 100644 --- a/hotspot/src/share/vm/classfile/verifier.hpp +++ b/hotspot/src/share/vm/classfile/verifier.hpp @@ -50,6 +50,7 @@ class Verifier : AllStatic { * Otherwise, no exception is thrown and the return indicates the * error. */ + static void log_end_verification(outputStream* st, const char* klassName, Symbol* exception_name, TRAPS); static bool verify(instanceKlassHandle klass, Mode mode, bool should_verify_class, TRAPS); // Return false if the class is loaded by the bootstrap loader, diff --git a/hotspot/src/share/vm/logging/logTag.hpp b/hotspot/src/share/vm/logging/logTag.hpp index 4d5d6caf73d..e5ca5de2e68 100644 --- a/hotspot/src/share/vm/logging/logTag.hpp +++ b/hotspot/src/share/vm/logging/logTag.hpp @@ -31,6 +31,7 @@ // (The tags 'all', 'disable' and 'help' are special tags that can // not be used in log calls, and should not be listed below.) #define LOG_TAG_LIST \ + LOG_TAG(classinit) \ LOG_TAG(defaultmethods) \ LOG_TAG(logging) \ LOG_TAG(safepoint) diff --git a/hotspot/src/share/vm/oops/instanceKlass.cpp b/hotspot/src/share/vm/oops/instanceKlass.cpp index 03584ee0320..9fe297aca04 100644 --- a/hotspot/src/share/vm/oops/instanceKlass.cpp +++ b/hotspot/src/share/vm/oops/instanceKlass.cpp @@ -62,6 +62,7 @@ #include "services/threadService.hpp" #include "utilities/dtrace.hpp" #include "utilities/macros.hpp" +#include "logging/log.hpp" #ifdef COMPILER1 #include "c1/c1_Compiler.hpp" #endif @@ -492,9 +493,9 @@ void InstanceKlass::eager_initialize_impl(instanceKlassHandle this_k) { this_k->set_init_state (fully_initialized); this_k->fence_and_clear_init_lock(); // trace - if (TraceClassInitialization) { + if (log_is_enabled(Info, classinit)) { ResourceMark rm(THREAD); - tty->print_cr("[Initialized %s without side effects]", this_k->external_name()); + log_info(classinit)("[Initialized %s without side effects]", this_k->external_name()); } } } @@ -1127,10 +1128,12 @@ void InstanceKlass::call_class_initializer_impl(instanceKlassHandle this_k, TRAP methodHandle h_method(THREAD, this_k->class_initializer()); assert(!this_k->is_initialized(), "we cannot initialize twice"); - if (TraceClassInitialization) { - tty->print("%d Initializing ", call_class_initializer_impl_counter++); - this_k->name()->print_value(); - tty->print_cr("%s (" INTPTR_FORMAT ")", h_method() == NULL ? "(no method)" : "", p2i(this_k())); + if (log_is_enabled(Info, classinit)) { + ResourceMark rm; + outputStream* log = LogHandle(classinit)::info_stream(); + log->print("%d Initializing ", call_class_initializer_impl_counter++); + this_k->name()->print_value_on(log); + log->print_cr("%s (" INTPTR_FORMAT ")", h_method() == NULL ? "(no method)" : "", p2i(this_k())); } if (h_method() != NULL) { JavaCallArguments args; // No arguments diff --git a/hotspot/src/share/vm/runtime/globals.hpp b/hotspot/src/share/vm/runtime/globals.hpp index fb61f555307..f33e6a52db3 100644 --- a/hotspot/src/share/vm/runtime/globals.hpp +++ b/hotspot/src/share/vm/runtime/globals.hpp @@ -1463,9 +1463,6 @@ public: develop(bool, TraceBytecodes, false, \ "Trace bytecode execution") \ \ - develop(bool, TraceClassInitialization, false, \ - "Trace class initialization") \ - \ product(bool, TraceExceptions, false, \ "Trace exceptions") \ \ diff --git a/hotspot/test/runtime/logging/BadMap50.jasm b/hotspot/test/runtime/logging/BadMap50.jasm new file mode 100644 index 00000000000..5e04fb690e4 --- /dev/null +++ b/hotspot/test/runtime/logging/BadMap50.jasm @@ -0,0 +1,156 @@ +/* + * Copyright (c) 2015, 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. + */ + +/* + * This class should throw VerifyError because the StackMap for bytecode + * index 45 is incorrect. The stack maps for bytecode indexes 45 and 49 are + * incompatible because 45 doesn't supply enough locals to satisfy 49. + * + * The astore_2 bytecode at bytecode index 45 changes the type state, + * preventing the stackmap mismatch. But, if the incoming type state is used, + * as required by JVM Spec 8, then the verifier will detected the stackmap + * mismatch, and throw VerifyError. + */ + +super public class BadMap50 + version 50:0 +{ + + +public Method "":"()V" + stack 1 locals 1 +{ + aload_0; + invokespecial Method java/lang/Object."":"()V"; + return; +} + +public static Method main:"([Ljava/lang/String;)V" + throws java/lang/Throwable + stack 0 locals 1 +{ + return; +} + +public static Method foo:"()V" + stack 3 locals 5 +{ + iconst_0; + ifne L5; + nop; + try t7; + L5: stack_frame_type full; + aconst_null; + astore_0; + iconst_3; + istore_1; + try t0; + aconst_null; + astore_0; + endtry t0; + goto L19; + catch t0 java/io/IOException; + stack_frame_type full; + locals_map class java/lang/Object, int; + stack_map class java/io/IOException; + astore_2; + aconst_null; + astore_0; + iconst_2; + istore_1; + try t1; + L19: stack_frame_type full; + locals_map class java/lang/Object, int; + iconst_0; + istore_2; + endtry t1; + iload_1; + ifeq L37; + nop; + goto L37; + catch t1 #0; + catch t2 #0; + try t2; + stack_frame_type full; + locals_map class java/lang/Object, int; + stack_map class java/lang/Throwable; +astore_3; +iconst_2; +istore_2; + endtry t2; + iload_1; + ifeq L35; + nop; + L35: stack_frame_type full; + locals_map class java/lang/Object, int, bogus, class java/lang/Throwable; +aload_3; + athrow; + try t3, t4; + L37: stack_frame_type full; + locals_map class java/lang/Object, int, int; + iload_2; + ifeq L42; + nop; + endtry t3, t4; + L42: stack_frame_type full; + locals_map class java/lang/Object, int, int; + goto L54; + catch t3 java/lang/Exception; + try t5; + stack_frame_type full; + locals_map class java/lang/Object, int; + stack_map class java/lang/Exception; + // astore_2; // astore_2, at bci 45, that changes the type state. +// pop; +iconst_1; + istore_2; // astore_2, at bci 45, that changes the type state. + endtry t5; + goto L54; + catch t4 #0; + catch t5 #0; + catch t6 #0; + try t6; + stack_frame_type full; + locals_map class java/lang/Object, int, int; + stack_map class java/lang/Throwable; +// astore 3; + istore_1; + endtry t6; +// aload 3; +// athrow; + L54: stack_frame_type full; + locals_map class java/lang/Object, int, int; + goto L57; + L57: stack_frame_type full; + locals_map class java/lang/Object, int, int; + nop; + endtry t7; + return; + catch t7 #0; + stack_frame_type full; + stack_map class java/lang/Throwable; + nop; + athrow; +} + +} // end Class BadMap50 diff --git a/hotspot/test/runtime/logging/ClassInitializationTest.java b/hotspot/test/runtime/logging/ClassInitializationTest.java new file mode 100644 index 00000000000..23c17f9aca3 --- /dev/null +++ b/hotspot/test/runtime/logging/ClassInitializationTest.java @@ -0,0 +1,62 @@ +/* + * Copyright (c) 2015, 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. + */ + + +/* + * @test ClassInitializationTest + * @bug 8142976 + * @library /testlibrary + * @compile BadMap50.jasm + * @run driver ClassInitializationTest + */ + +import jdk.test.lib.*; + +public class ClassInitializationTest { + + public static void main(String... args) throws Exception { + + // (1) + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xlog:classinit=info", "-Xverify:all", "-Xmx64m", "BadMap50"); + OutputAnalyzer out = new OutputAnalyzer(pb.start()); + out.shouldContain("Start class verification for:"); + out.shouldContain("End class verification for:"); + out.shouldContain("Initializing"); + out.shouldContain("Verification for BadMap50 failed"); + out.shouldContain("Fail over class verification to old verifier for: BadMap50"); + + // (2) + if (Platform.isDebugBuild()) { + pb = ProcessTools.createJavaProcessBuilder("-Xlog:classinit=info", "-Xverify:all", "-XX:+EagerInitialization", "-Xmx64m", "-version"); + out = new OutputAnalyzer(pb.start()); + out.shouldContain("[Initialized").shouldContain("without side effects]"); + out.shouldHaveExitValue(0); + } + // (3) Ensure that VerboseVerification still triggers appropriate messages. + pb = ProcessTools.createJavaProcessBuilder("-XX:+UnlockDiagnosticVMOptions", "-XX:+VerboseVerification", "-Xverify:all", "-Xmx64m", "BadMap50"); + out = new OutputAnalyzer(pb.start()); + out.shouldContain("End class verification for:"); + out.shouldContain("Verification for BadMap50 failed"); + out.shouldContain("Fail over class verification to old verifier for: BadMap50"); + } +} From 0d75e3c8cd74bdfd2692d3e1e6c06bf96234886e Mon Sep 17 00:00:00 2001 From: Rachel Protacio Date: Thu, 3 Dec 2015 13:08:37 -0500 Subject: [PATCH 4/7] 8144536: Clean up Unified Logging test directory Consolidated two logging options' tests into one file each Reviewed-by: dholmes, ctornqvi --- .../runtime/logging/DefaultMethodsTest.java | 2 +- .../test/runtime/logging/SafepointTest.java | 28 ++++++++++-- .../runtime/logging/SafepointTestMain.java | 45 ------------------- .../test/runtime/logging/VMOperationTest.java | 26 +++++++++-- .../runtime/logging/VMOperationTestMain.java | 43 ------------------ 5 files changed, 48 insertions(+), 96 deletions(-) delete mode 100644 hotspot/test/runtime/logging/SafepointTestMain.java delete mode 100644 hotspot/test/runtime/logging/VMOperationTestMain.java diff --git a/hotspot/test/runtime/logging/DefaultMethodsTest.java b/hotspot/test/runtime/logging/DefaultMethodsTest.java index 50cbbe0e63d..007da7bf8fb 100644 --- a/hotspot/test/runtime/logging/DefaultMethodsTest.java +++ b/hotspot/test/runtime/logging/DefaultMethodsTest.java @@ -28,7 +28,7 @@ * @library /testlibrary * @modules java.base/sun.misc * java.management - * @run main DefaultMethodsTest + * @run driver DefaultMethodsTest */ import jdk.test.lib.*; diff --git a/hotspot/test/runtime/logging/SafepointTest.java b/hotspot/test/runtime/logging/SafepointTest.java index 79ca2e2b2fa..a5bc00ece3d 100644 --- a/hotspot/test/runtime/logging/SafepointTest.java +++ b/hotspot/test/runtime/logging/SafepointTest.java @@ -26,19 +26,18 @@ * @bug 8140348 * @summary safepoint=trace should have output from each log statement in the code * @library /testlibrary - * @compile SafepointTestMain.java * @modules java.base/sun.misc * java.management - * @build SafepointTest - * @run main SafepointTest + * @run driver SafepointTest */ import jdk.test.lib.*; +import java.lang.ref.WeakReference; public class SafepointTest { public static void main(String[] args) throws Exception { ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( - "-Xlog:safepoint=trace", "SafepointTestMain"); + "-Xlog:safepoint=trace", InnerClass.class.getName()); OutputAnalyzer output = new OutputAnalyzer(pb.start()); output.shouldContain("Safepoint synchronization initiated. ("); output.shouldContain("Entering safepoint region: "); @@ -46,4 +45,25 @@ public class SafepointTest { output.shouldContain("_at_poll_safepoint"); output.shouldHaveExitValue(0); } + + public static class InnerClass { + public static byte[] garbage; + public static volatile WeakReference weakref; + + public static void createweakref() { + Object o = new Object(); + weakref = new WeakReference<>(o); + } + + public static void main(String[] args) throws Exception { + // Cause several safepoints to run GC to see safepoint messages + for (int i = 0; i < 2; i++) { + createweakref(); + while(weakref.get() != null) { + garbage = new byte[8192]; + System.gc(); + } + } + } + } } diff --git a/hotspot/test/runtime/logging/SafepointTestMain.java b/hotspot/test/runtime/logging/SafepointTestMain.java deleted file mode 100644 index 4d65d2fcf3d..00000000000 --- a/hotspot/test/runtime/logging/SafepointTestMain.java +++ /dev/null @@ -1,45 +0,0 @@ -/* - * Copyright (c) 2015, 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. - */ - -import java.lang.ref.WeakReference; - -public class SafepointTestMain { - public static byte[] garbage; - public static volatile WeakReference weakref; - - public static void createweakref() { - Object o = new Object(); - weakref = new WeakReference<>(o); - } - - public static void main(String[] args) throws Exception { - // Cause several safepoints to run GC to see safepoint messages - for (int i = 0; i < 2; i++) { - createweakref(); - while(weakref.get() != null) { - garbage = new byte[8192]; - System.gc(); - } - } - } -} diff --git a/hotspot/test/runtime/logging/VMOperationTest.java b/hotspot/test/runtime/logging/VMOperationTest.java index 630dcd14386..4d2ae411e1c 100644 --- a/hotspot/test/runtime/logging/VMOperationTest.java +++ b/hotspot/test/runtime/logging/VMOperationTest.java @@ -26,21 +26,41 @@ * @bug 8143157 * @summary vmoperation=debug should have logging output * @library /testlibrary - * @compile VMOperationTestMain.java * @modules java.base/sun.misc * java.management - * @run main VMOperationTest + * @run driver VMOperationTest */ import jdk.test.lib.*; +import java.lang.ref.WeakReference; public class VMOperationTest { public static void main(String[] args) throws Exception { ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( - "-Xlog:vmoperation=debug", "-Xmx64m", "-Xms64m", "VMOperationTestMain"); + "-Xlog:vmoperation=debug", "-Xmx64m", "-Xms64m", + InternalClass.class.getName()); OutputAnalyzer output = new OutputAnalyzer(pb.start()); output.shouldContain("VM_Operation ("); output.shouldHaveExitValue(0); } + + public static class InternalClass { + public static byte[] garbage; + public static volatile WeakReference weakref; + + public static void createweakref() { + Object o = new Object(); + weakref = new WeakReference<>(o); + } + + // Loop until a GC runs. + public static void main(String[] args) throws Exception { + createweakref(); + while (weakref.get() != null) { + garbage = new byte[8192]; + System.gc(); + } + } + } } diff --git a/hotspot/test/runtime/logging/VMOperationTestMain.java b/hotspot/test/runtime/logging/VMOperationTestMain.java deleted file mode 100644 index 53101892dc2..00000000000 --- a/hotspot/test/runtime/logging/VMOperationTestMain.java +++ /dev/null @@ -1,43 +0,0 @@ -/* - * Copyright (c) 2015, 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. - */ - -import java.lang.ref.WeakReference; - -public class VMOperationTestMain { - public static byte[] garbage; - public static volatile WeakReference weakref; - - public static void createweakref() { - Object o = new Object(); - weakref = new WeakReference<>(o); - } - - // Loop until a GC runs. - public static void main(String[] args) throws Exception { - createweakref(); - while (weakref.get() != null) { - garbage = new byte[8192]; - System.gc(); - } - } -} From 2eb9b07e69aa0162594c870ee4935106648e8e40 Mon Sep 17 00:00:00 2001 From: Dmitry Samersoff Date: Thu, 3 Dec 2015 21:34:23 +0300 Subject: [PATCH 5/7] 8114853: variable tracking size limit exceeded in vmStructs.cpp Turn off var-tracking-assignments for vmStructs.cpp Reviewed-by: tschatzl, mgerdin, coleenp --- hotspot/make/linux/makefiles/gcc.make | 3 +++ 1 file changed, 3 insertions(+) diff --git a/hotspot/make/linux/makefiles/gcc.make b/hotspot/make/linux/makefiles/gcc.make index 7920596193f..4b3c7b1fd9d 100644 --- a/hotspot/make/linux/makefiles/gcc.make +++ b/hotspot/make/linux/makefiles/gcc.make @@ -260,6 +260,9 @@ endif OPT_CFLAGS = $(OPT_CFLAGS/$(OPT_CFLAGS_DEFAULT)) $(OPT_EXTRAS) +# Variable tracking size limit exceeded for VMStructs::init() +OPT_CFLAGS/vmStructs.o += -fno-var-tracking-assignments + # The gcc compiler segv's on ia64 when compiling bytecodeInterpreter.cpp # if we use expensive-optimizations ifeq ($(BUILDARCH), ia64) From 6e0ea9d21c27659c726b5753f0f32f239ba668c2 Mon Sep 17 00:00:00 2001 From: David Holmes Date: Fri, 4 Dec 2015 04:06:37 -0500 Subject: [PATCH 6/7] 8132510: Replace ThreadLocalStorage with compiler/language-based thread-local variables Used compiled-based TLS when available. Additional contributions by Thomas Stufe (AIX) and Andrew Haley (Aarch64) Reviewed-by: stuefe, bdelsart, dcubed --- hotspot/make/aix/makefiles/xlc.make | 3 + .../cpu/aarch64/vm/macroAssembler_aarch64.cpp | 21 ++++ .../src/cpu/sparc/vm/macroAssembler_sparc.cpp | 6 +- .../src/cpu/sparc/vm/stubRoutines_sparc.cpp | 2 +- hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp | 41 ++++++++ hotspot/src/os/aix/vm/os_aix.cpp | 28 +----- hotspot/src/os/aix/vm/os_aix.inline.hpp | 4 - hotspot/src/os/aix/vm/thread_aix.inline.hpp | 36 ------- hotspot/src/os/bsd/vm/os_bsd.cpp | 40 +------- hotspot/src/os/bsd/vm/os_bsd.inline.hpp | 6 +- hotspot/src/os/bsd/vm/thread_bsd.inline.hpp | 39 -------- hotspot/src/os/linux/vm/os_linux.cpp | 39 +------- hotspot/src/os/linux/vm/os_linux.inline.hpp | 6 +- .../src/os/linux/vm/thread_linux.inline.hpp | 39 -------- .../os/posix/vm/threadLocalStorage_posix.cpp | 68 +++++++++++++ hotspot/src/os/solaris/vm/os_solaris.cpp | 5 +- .../os/solaris/vm/thread_solaris.inline.hpp | 50 ---------- hotspot/src/os/windows/vm/os_windows.cpp | 56 +++++------ hotspot/src/os/windows/vm/os_windows.hpp | 11 +++ .../windows/vm/threadLocalStorage_windows.cpp | 63 ++++++++++++ .../os/windows/vm/thread_windows.inline.hpp | 39 -------- hotspot/src/os_cpu/aix_ppc/vm/os_aix_ppc.cpp | 2 +- .../os_cpu/aix_ppc/vm/threadLS_aix_ppc.cpp | 40 -------- .../os_cpu/aix_ppc/vm/threadLS_aix_ppc.hpp | 36 ------- .../os_cpu/bsd_x86/vm/assembler_bsd_x86.cpp | 57 +---------- hotspot/src/os_cpu/bsd_x86/vm/os_bsd_x86.cpp | 2 +- .../os_cpu/bsd_x86/vm/threadLS_bsd_x86.cpp | 92 ----------------- .../os_cpu/bsd_x86/vm/threadLS_bsd_x86.hpp | 56 ----------- .../os_cpu/bsd_zero/vm/assembler_bsd_zero.cpp | 8 +- .../src/os_cpu/bsd_zero/vm/os_bsd_zero.cpp | 2 +- .../os_cpu/bsd_zero/vm/threadLS_bsd_zero.cpp | 40 -------- .../os_cpu/bsd_zero/vm/threadLS_bsd_zero.hpp | 35 ------- .../vm/assembler_linux_aarch64.cpp | 30 +----- .../linux_aarch64/vm/os_linux_aarch64.cpp | 2 +- .../vm/threadLS_linux_aarch64.cpp | 42 -------- .../vm/threadLS_linux_aarch64.hpp | 37 ------- .../linux_aarch64/vm/threadLS_linux_aarch64.s | 44 +++++++++ .../linux_aarch64/vm/thread_linux_aarch64.hpp | 4 +- .../src/os_cpu/linux_ppc/vm/os_linux_ppc.cpp | 2 +- .../linux_ppc/vm/threadLS_linux_ppc.cpp | 39 -------- .../linux_ppc/vm/threadLS_linux_ppc.hpp | 36 ------- .../os_cpu/linux_sparc/vm/os_linux_sparc.cpp | 2 +- .../linux_sparc/vm/threadLS_linux_sparc.cpp | 38 ------- .../linux_sparc/vm/threadLS_linux_sparc.hpp | 33 ------- .../linux_x86/vm/assembler_linux_x86.cpp | 80 +-------------- .../src/os_cpu/linux_x86/vm/os_linux_x86.cpp | 2 +- .../linux_x86/vm/threadLS_linux_x86.cpp | 98 ------------------- .../linux_x86/vm/threadLS_linux_x86.hpp | 58 ----------- .../linux_zero/vm/assembler_linux_zero.cpp | 8 +- .../os_cpu/linux_zero/vm/os_linux_zero.cpp | 2 +- .../linux_zero/vm/threadLS_linux_zero.cpp | 40 -------- .../linux_zero/vm/threadLS_linux_zero.hpp | 35 ------- .../solaris_sparc/vm/os_solaris_sparc.cpp | 2 +- .../vm/threadLS_solaris_sparc.cpp | 52 ---------- .../vm/threadLS_solaris_sparc.hpp | 39 -------- .../solaris_x86/vm/assembler_solaris_x86.cpp | 32 ------ .../os_cpu/solaris_x86/vm/os_solaris_x86.cpp | 2 +- .../solaris_x86/vm/threadLS_solaris_x86.cpp | 52 ---------- .../solaris_x86/vm/threadLS_solaris_x86.hpp | 39 -------- .../windows_x86/vm/assembler_windows_x86.cpp | 43 +------- .../os_cpu/windows_x86/vm/os_windows_x86.cpp | 6 +- .../windows_x86/vm/threadLS_windows_x86.cpp | 49 ---------- .../windows_x86/vm/threadLS_windows_x86.hpp | 49 ---------- hotspot/src/share/vm/code/nmethod.cpp | 2 +- .../vm/gc/cms/concurrentMarkSweepThread.cpp | 3 - hotspot/src/share/vm/gc/g1/g1HotCardCache.hpp | 2 +- .../src/share/vm/gc/parallel/gcTaskThread.cpp | 2 - .../share/vm/gc/shared/concurrentGCThread.cpp | 4 - hotspot/src/share/vm/gc/shared/workgroup.cpp | 1 - hotspot/src/share/vm/memory/allocation.cpp | 2 +- hotspot/src/share/vm/memory/resourceArea.hpp | 4 +- hotspot/src/share/vm/oops/oopsHierarchy.cpp | 4 +- .../src/share/vm/precompiled/precompiled.hpp | 1 - hotspot/src/share/vm/prims/jni.cpp | 12 +-- hotspot/src/share/vm/prims/jniCheck.cpp | 4 +- hotspot/src/share/vm/prims/jvmtiEnter.xsl | 8 +- hotspot/src/share/vm/prims/jvmtiExport.cpp | 6 +- hotspot/src/share/vm/prims/jvmtiUtil.hpp | 4 +- .../src/share/vm/runtime/interfaceSupport.cpp | 1 - .../src/share/vm/runtime/interfaceSupport.hpp | 2 +- hotspot/src/share/vm/runtime/java.cpp | 11 +-- hotspot/src/share/vm/runtime/mutex.cpp | 6 +- hotspot/src/share/vm/runtime/mutexLocker.cpp | 1 - hotspot/src/share/vm/runtime/os.cpp | 24 +---- hotspot/src/share/vm/runtime/os.hpp | 6 -- .../src/share/vm/runtime/sharedRuntime.hpp | 1 - hotspot/src/share/vm/runtime/thread.cpp | 65 +++++------- hotspot/src/share/vm/runtime/thread.hpp | 58 ++++++----- .../src/share/vm/runtime/thread.inline.hpp | 17 +--- .../share/vm/runtime/threadLocalStorage.cpp | 63 ------------ .../share/vm/runtime/threadLocalStorage.hpp | 82 +++------------- hotspot/src/share/vm/runtime/vmThread.cpp | 4 - .../src/share/vm/runtime/vm_operations.cpp | 6 +- hotspot/src/share/vm/utilities/debug.cpp | 12 +-- hotspot/src/share/vm/utilities/events.cpp | 1 - hotspot/src/share/vm/utilities/events.hpp | 4 +- .../vm/utilities/globalDefinitions_gcc.hpp | 4 + .../globalDefinitions_sparcWorks.hpp | 4 + .../vm/utilities/globalDefinitions_visCPP.hpp | 6 +- .../vm/utilities/globalDefinitions_xlc.hpp | 3 + hotspot/src/share/vm/utilities/ostream.cpp | 6 +- 101 files changed, 449 insertions(+), 1966 deletions(-) delete mode 100644 hotspot/src/os/aix/vm/thread_aix.inline.hpp delete mode 100644 hotspot/src/os/bsd/vm/thread_bsd.inline.hpp delete mode 100644 hotspot/src/os/linux/vm/thread_linux.inline.hpp create mode 100644 hotspot/src/os/posix/vm/threadLocalStorage_posix.cpp delete mode 100644 hotspot/src/os/solaris/vm/thread_solaris.inline.hpp create mode 100644 hotspot/src/os/windows/vm/threadLocalStorage_windows.cpp delete mode 100644 hotspot/src/os/windows/vm/thread_windows.inline.hpp delete mode 100644 hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.cpp delete mode 100644 hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.hpp delete mode 100644 hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.cpp delete mode 100644 hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.hpp delete mode 100644 hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.cpp delete mode 100644 hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.hpp delete mode 100644 hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.cpp delete mode 100644 hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.hpp create mode 100644 hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.s delete mode 100644 hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.cpp delete mode 100644 hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.hpp delete mode 100644 hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.cpp delete mode 100644 hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.hpp delete mode 100644 hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.cpp delete mode 100644 hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.hpp delete mode 100644 hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.cpp delete mode 100644 hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.hpp delete mode 100644 hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.cpp delete mode 100644 hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.hpp delete mode 100644 hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.cpp delete mode 100644 hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.hpp delete mode 100644 hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.cpp delete mode 100644 hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.hpp delete mode 100644 hotspot/src/share/vm/runtime/threadLocalStorage.cpp diff --git a/hotspot/make/aix/makefiles/xlc.make b/hotspot/make/aix/makefiles/xlc.make index cf8d085c39a..676ba5c6a6d 100644 --- a/hotspot/make/aix/makefiles/xlc.make +++ b/hotspot/make/aix/makefiles/xlc.make @@ -74,6 +74,9 @@ CFLAGS += $(VM_PICFLAG) CFLAGS += -qnortti CFLAGS += -qnoeh +# for compiler-level tls +CFLAGS += -qtls=default + CFLAGS += -D_REENTRANT # no xlc counterpart for -fcheck-new # CFLAGS += -fcheck-new diff --git a/hotspot/src/cpu/aarch64/vm/macroAssembler_aarch64.cpp b/hotspot/src/cpu/aarch64/vm/macroAssembler_aarch64.cpp index c9608bc7492..eae6417b08b 100644 --- a/hotspot/src/cpu/aarch64/vm/macroAssembler_aarch64.cpp +++ b/hotspot/src/cpu/aarch64/vm/macroAssembler_aarch64.cpp @@ -41,6 +41,7 @@ #include "runtime/icache.hpp" #include "runtime/interfaceSupport.hpp" #include "runtime/sharedRuntime.hpp" +#include "runtime/thread.hpp" #if INCLUDE_ALL_GCS #include "gc/g1/g1CollectedHeap.inline.hpp" @@ -4653,3 +4654,23 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, BIND(DONE); sub(result, result, len); // Return index where we stopped } + +// get_thread() can be called anywhere inside generated code so we +// need to save whatever non-callee save context might get clobbered +// by the call to JavaThread::aarch64_get_thread_helper() or, indeed, +// the call setup code. +// +// aarch64_get_thread_helper() clobbers only r0, r1, and flags. +// +void MacroAssembler::get_thread(Register dst) { + RegSet saved_regs = RegSet::range(r0, r1) + lr - dst; + push(saved_regs, sp); + + mov(lr, CAST_FROM_FN_PTR(address, JavaThread::aarch64_get_thread_helper)); + blrt(lr, 1, 0, 1); + if (dst != c_rarg0) { + mov(dst, c_rarg0); + } + + pop(saved_regs, sp); +} diff --git a/hotspot/src/cpu/sparc/vm/macroAssembler_sparc.cpp b/hotspot/src/cpu/sparc/vm/macroAssembler_sparc.cpp index 59b5b41a0d8..0d069a48ea5 100644 --- a/hotspot/src/cpu/sparc/vm/macroAssembler_sparc.cpp +++ b/hotspot/src/cpu/sparc/vm/macroAssembler_sparc.cpp @@ -360,10 +360,10 @@ void MacroAssembler::leave() { #ifdef ASSERT // a hook for debugging static Thread* reinitialize_thread() { - return ThreadLocalStorage::thread(); + return Thread::current(); } #else -#define reinitialize_thread ThreadLocalStorage::thread +#define reinitialize_thread Thread::current #endif #ifdef ASSERT @@ -393,7 +393,7 @@ void MacroAssembler::get_thread() { } static Thread* verify_thread_subroutine(Thread* gthread_value) { - Thread* correct_value = ThreadLocalStorage::thread(); + Thread* correct_value = Thread::current(); guarantee(gthread_value == correct_value, "G2_thread value must be the thread"); return correct_value; } diff --git a/hotspot/src/cpu/sparc/vm/stubRoutines_sparc.cpp b/hotspot/src/cpu/sparc/vm/stubRoutines_sparc.cpp index 3a9ca6fffc9..f8083c0f273 100644 --- a/hotspot/src/cpu/sparc/vm/stubRoutines_sparc.cpp +++ b/hotspot/src/cpu/sparc/vm/stubRoutines_sparc.cpp @@ -36,7 +36,7 @@ extern "C" { address _flush_reg_windows(); // in .s file. // Flush registers to stack. In case of error we will need to stack walk. address bootstrap_flush_windows(void) { - Thread* thread = ThreadLocalStorage::get_thread_slow(); + Thread* thread = Thread::current_or_null(); // Very early in process there is no thread. if (thread != NULL) { guarantee(thread->is_Java_thread(), "Not a Java thread."); diff --git a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp index 21646e1bc0d..50f85fbfa84 100644 --- a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp +++ b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp @@ -39,6 +39,7 @@ #include "runtime/os.hpp" #include "runtime/sharedRuntime.hpp" #include "runtime/stubRoutines.hpp" +#include "runtime/thread.hpp" #include "utilities/macros.hpp" #if INCLUDE_ALL_GCS #include "gc/g1/g1CollectedHeap.inline.hpp" @@ -10834,3 +10835,43 @@ SkipIfEqual::SkipIfEqual( SkipIfEqual::~SkipIfEqual() { _masm->bind(_label); } + +// 32-bit Windows has its own fast-path implementation +// of get_thread +#if !defined(WIN32) || defined(_LP64) + +// This is simply a call to Thread::current() +void MacroAssembler::get_thread(Register thread) { + if (thread != rax) { + push(rax); + } + LP64_ONLY(push(rdi);) + LP64_ONLY(push(rsi);) + push(rdx); + push(rcx); +#ifdef _LP64 + push(r8); + push(r9); + push(r10); + push(r11); +#endif + + MacroAssembler::call_VM_leaf_base(CAST_FROM_FN_PTR(address, Thread::current), 0); + +#ifdef _LP64 + pop(r11); + pop(r10); + pop(r9); + pop(r8); +#endif + pop(rcx); + pop(rdx); + LP64_ONLY(pop(rsi);) + LP64_ONLY(pop(rdi);) + if (thread != rax) { + mov(thread, rax); + pop(rax); + } +} + +#endif diff --git a/hotspot/src/os/aix/vm/os_aix.cpp b/hotspot/src/os/aix/vm/os_aix.cpp index 04b6ac79f59..50755a1815e 100644 --- a/hotspot/src/os/aix/vm/os_aix.cpp +++ b/hotspot/src/os/aix/vm/os_aix.cpp @@ -902,7 +902,7 @@ static void *java_start(Thread *thread) { int pid = os::current_process_id(); alloca(((pid ^ counter++) & 7) * 128); - ThreadLocalStorage::set_thread(thread); + thread->initialize_thread_current(); OSThread* osthread = thread->osthread(); @@ -1077,32 +1077,6 @@ void os::free_thread(OSThread* osthread) { delete osthread; } -////////////////////////////////////////////////////////////////////////////// -// thread local storage - -int os::allocate_thread_local_storage() { - pthread_key_t key; - int rslt = pthread_key_create(&key, NULL); - assert(rslt == 0, "cannot allocate thread local storage"); - return (int)key; -} - -// Note: This is currently not used by VM, as we don't destroy TLS key -// on VM exit. -void os::free_thread_local_storage(int index) { - int rslt = pthread_key_delete((pthread_key_t)index); - assert(rslt == 0, "invalid index"); -} - -void os::thread_local_storage_at_put(int index, void* value) { - int rslt = pthread_setspecific((pthread_key_t)index, value); - assert(rslt == 0, "pthread_setspecific failed"); -} - -extern "C" Thread* get_thread() { - return ThreadLocalStorage::thread(); -} - //////////////////////////////////////////////////////////////////////////////// // time support diff --git a/hotspot/src/os/aix/vm/os_aix.inline.hpp b/hotspot/src/os/aix/vm/os_aix.inline.hpp index 7de0626dae6..0aa537bf923 100644 --- a/hotspot/src/os/aix/vm/os_aix.inline.hpp +++ b/hotspot/src/os/aix/vm/os_aix.inline.hpp @@ -36,10 +36,6 @@ #include #include -inline void* os::thread_local_storage_at(int index) { - return pthread_getspecific((pthread_key_t)index); -} - // File names are case-sensitive on windows only. inline int os::file_name_strcmp(const char* s1, const char* s2) { return strcmp(s1, s2); diff --git a/hotspot/src/os/aix/vm/thread_aix.inline.hpp b/hotspot/src/os/aix/vm/thread_aix.inline.hpp deleted file mode 100644 index 04623516049..00000000000 --- a/hotspot/src/os/aix/vm/thread_aix.inline.hpp +++ /dev/null @@ -1,36 +0,0 @@ -/* - * Copyright (c) 2002, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2013 SAP AG. 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. - * - */ - -#ifndef OS_AIX_VM_THREAD_AIX_INLINE_HPP -#define OS_AIX_VM_THREAD_AIX_INLINE_HPP - -#include "runtime/thread.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Contains inlined functions for class Thread and ThreadLocalStorage - -inline void ThreadLocalStorage::pd_invalidate_all() {} // nothing to do - -#endif // OS_AIX_VM_THREAD_AIX_INLINE_HPP diff --git a/hotspot/src/os/bsd/vm/os_bsd.cpp b/hotspot/src/os/bsd/vm/os_bsd.cpp index f6b6e43b502..e59bcb7f7ac 100644 --- a/hotspot/src/os/bsd/vm/os_bsd.cpp +++ b/hotspot/src/os/bsd/vm/os_bsd.cpp @@ -674,7 +674,7 @@ static void *java_start(Thread *thread) { int pid = os::current_process_id(); alloca(((pid ^ counter++) & 7) * 128); - ThreadLocalStorage::set_thread(thread); + thread->initialize_thread_current(); OSThread* osthread = thread->osthread(); Monitor* sync = osthread->startThread_lock(); @@ -882,44 +882,6 @@ void os::free_thread(OSThread* osthread) { delete osthread; } -////////////////////////////////////////////////////////////////////////////// -// thread local storage - -// Restore the thread pointer if the destructor is called. This is in case -// someone from JNI code sets up a destructor with pthread_key_create to run -// detachCurrentThread on thread death. Unless we restore the thread pointer we -// will hang or crash. When detachCurrentThread is called the key will be set -// to null and we will not be called again. If detachCurrentThread is never -// called we could loop forever depending on the pthread implementation. -static void restore_thread_pointer(void* p) { - Thread* thread = (Thread*) p; - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} - -int os::allocate_thread_local_storage() { - pthread_key_t key; - int rslt = pthread_key_create(&key, restore_thread_pointer); - assert(rslt == 0, "cannot allocate thread local storage"); - return (int)key; -} - -// Note: This is currently not used by VM, as we don't destroy TLS key -// on VM exit. -void os::free_thread_local_storage(int index) { - int rslt = pthread_key_delete((pthread_key_t)index); - assert(rslt == 0, "invalid index"); -} - -void os::thread_local_storage_at_put(int index, void* value) { - int rslt = pthread_setspecific((pthread_key_t)index, value); - assert(rslt == 0, "pthread_setspecific failed"); -} - -extern "C" Thread* get_thread() { - return ThreadLocalStorage::thread(); -} - - //////////////////////////////////////////////////////////////////////////////// // time support diff --git a/hotspot/src/os/bsd/vm/os_bsd.inline.hpp b/hotspot/src/os/bsd/vm/os_bsd.inline.hpp index 3afb5c21a83..4a654614c2b 100644 --- a/hotspot/src/os/bsd/vm/os_bsd.inline.hpp +++ b/hotspot/src/os/bsd/vm/os_bsd.inline.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -34,10 +34,6 @@ #include #include -inline void* os::thread_local_storage_at(int index) { - return pthread_getspecific((pthread_key_t)index); -} - // File names are case-sensitive on windows only inline int os::file_name_strcmp(const char* s1, const char* s2) { return strcmp(s1, s2); diff --git a/hotspot/src/os/bsd/vm/thread_bsd.inline.hpp b/hotspot/src/os/bsd/vm/thread_bsd.inline.hpp deleted file mode 100644 index 86f125dc5b4..00000000000 --- a/hotspot/src/os/bsd/vm/thread_bsd.inline.hpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 2002, 2011, 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. - * - */ - -#ifndef OS_BSD_VM_THREAD_BSD_INLINE_HPP -#define OS_BSD_VM_THREAD_BSD_INLINE_HPP - -#ifndef SHARE_VM_RUNTIME_THREAD_INLINE_HPP_SCOPE -#error "This file should only be included from thread.inline.hpp" -#endif - -#include "runtime/thread.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Contains inlined functions for class Thread and ThreadLocalStorage - -inline void ThreadLocalStorage::pd_invalidate_all() {} // nothing to do - -#endif // OS_BSD_VM_THREAD_BSD_INLINE_HPP diff --git a/hotspot/src/os/linux/vm/os_linux.cpp b/hotspot/src/os/linux/vm/os_linux.cpp index 2aa2088c2f5..a672c6525b9 100644 --- a/hotspot/src/os/linux/vm/os_linux.cpp +++ b/hotspot/src/os/linux/vm/os_linux.cpp @@ -646,7 +646,7 @@ static void *java_start(Thread *thread) { int pid = os::current_process_id(); alloca(((pid ^ counter++) & 7) * 128); - ThreadLocalStorage::set_thread(thread); + thread->initialize_thread_current(); OSThread* osthread = thread->osthread(); Monitor* sync = osthread->startThread_lock(); @@ -873,43 +873,6 @@ void os::free_thread(OSThread* osthread) { delete osthread; } -////////////////////////////////////////////////////////////////////////////// -// thread local storage - -// Restore the thread pointer if the destructor is called. This is in case -// someone from JNI code sets up a destructor with pthread_key_create to run -// detachCurrentThread on thread death. Unless we restore the thread pointer we -// will hang or crash. When detachCurrentThread is called the key will be set -// to null and we will not be called again. If detachCurrentThread is never -// called we could loop forever depending on the pthread implementation. -static void restore_thread_pointer(void* p) { - Thread* thread = (Thread*) p; - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} - -int os::allocate_thread_local_storage() { - pthread_key_t key; - int rslt = pthread_key_create(&key, restore_thread_pointer); - assert(rslt == 0, "cannot allocate thread local storage"); - return (int)key; -} - -// Note: This is currently not used by VM, as we don't destroy TLS key -// on VM exit. -void os::free_thread_local_storage(int index) { - int rslt = pthread_key_delete((pthread_key_t)index); - assert(rslt == 0, "invalid index"); -} - -void os::thread_local_storage_at_put(int index, void* value) { - int rslt = pthread_setspecific((pthread_key_t)index, value); - assert(rslt == 0, "pthread_setspecific failed"); -} - -extern "C" Thread* get_thread() { - return ThreadLocalStorage::thread(); -} - ////////////////////////////////////////////////////////////////////////////// // initial thread diff --git a/hotspot/src/os/linux/vm/os_linux.inline.hpp b/hotspot/src/os/linux/vm/os_linux.inline.hpp index ba4b777d520..7559cde3d00 100644 --- a/hotspot/src/os/linux/vm/os_linux.inline.hpp +++ b/hotspot/src/os/linux/vm/os_linux.inline.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -34,10 +34,6 @@ #include #include -inline void* os::thread_local_storage_at(int index) { - return pthread_getspecific((pthread_key_t)index); -} - // File names are case-sensitive on windows only inline int os::file_name_strcmp(const char* s1, const char* s2) { return strcmp(s1, s2); diff --git a/hotspot/src/os/linux/vm/thread_linux.inline.hpp b/hotspot/src/os/linux/vm/thread_linux.inline.hpp deleted file mode 100644 index b58dc078948..00000000000 --- a/hotspot/src/os/linux/vm/thread_linux.inline.hpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 2002, 2011, 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. - * - */ - -#ifndef OS_LINUX_VM_THREAD_LINUX_INLINE_HPP -#define OS_LINUX_VM_THREAD_LINUX_INLINE_HPP - -#ifndef SHARE_VM_RUNTIME_THREAD_INLINE_HPP_SCOPE -#error "This file should only be included from thread.inline.hpp" -#endif - -#include "runtime/thread.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Contains inlined functions for class Thread and ThreadLocalStorage - -inline void ThreadLocalStorage::pd_invalidate_all() {} // nothing to do - -#endif // OS_LINUX_VM_THREAD_LINUX_INLINE_HPP diff --git a/hotspot/src/os/posix/vm/threadLocalStorage_posix.cpp b/hotspot/src/os/posix/vm/threadLocalStorage_posix.cpp new file mode 100644 index 00000000000..cc703ef811f --- /dev/null +++ b/hotspot/src/os/posix/vm/threadLocalStorage_posix.cpp @@ -0,0 +1,68 @@ +/* + * Copyright (c) 2015, 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 "runtime/threadLocalStorage.hpp" +#include + +static pthread_key_t _thread_key; +static bool _initialized = false; + +// Restore the thread pointer if the destructor is called. This is in case +// someone from JNI code sets up a destructor with pthread_key_create to run +// detachCurrentThread on thread death. Unless we restore the thread pointer we +// will hang or crash. When detachCurrentThread is called the key will be set +// to null and we will not be called again. If detachCurrentThread is never +// called we could loop forever depending on the pthread implementation. +extern "C" void restore_thread_pointer(void* p) { + ThreadLocalStorage::set_thread((Thread*) p); +} + +void ThreadLocalStorage::init() { + assert(!_initialized, "initializing TLS more than once!"); + int rslt = pthread_key_create(&_thread_key, restore_thread_pointer); + // If this assert fails we will get a recursive assertion failure + // and not see the actual error message or get a hs_err file + assert_status(rslt == 0, rslt, "pthread_key_create"); + _initialized = true; +} + +bool ThreadLocalStorage::is_initialized() { + return _initialized; +} + +Thread* ThreadLocalStorage::thread() { + // If this assert fails we will get a recursive assertion failure + // and not see the actual error message or get a hs_err file. + // Which most likely indicates we have taken an error path early in + // the initialization process, which is using Thread::current without + // checking TLS is initialized - see java.cpp vm_exit + assert(_initialized, "TLS not initialized yet!"); + return (Thread*) pthread_getspecific(_thread_key); // may be NULL +} + +void ThreadLocalStorage::set_thread(Thread* current) { + assert(_initialized, "TLS not initialized yet!"); + int rslt = pthread_setspecific(_thread_key, current); + assert_status(rslt == 0, rslt, "pthread_setspecific"); +} diff --git a/hotspot/src/os/solaris/vm/os_solaris.cpp b/hotspot/src/os/solaris/vm/os_solaris.cpp index 96a0831949b..0a78ca64304 100644 --- a/hotspot/src/os/solaris/vm/os_solaris.cpp +++ b/hotspot/src/os/solaris/vm/os_solaris.cpp @@ -728,6 +728,9 @@ extern "C" void* java_start(void* thread_addr) { int prio; Thread* thread = (Thread*)thread_addr; + + thread->initialize_thread_current(); + OSThread* osthr = thread->osthread(); osthr->set_lwp_id(_lwp_self()); // Store lwp in case we are bound @@ -5579,7 +5582,7 @@ int os::fork_and_exec(char* cmd) { // fork is async-safe, fork1 is not so can't use in signal handler pid_t pid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); if (t != NULL && t->is_inside_signal_handler()) { pid = fork(); } else { diff --git a/hotspot/src/os/solaris/vm/thread_solaris.inline.hpp b/hotspot/src/os/solaris/vm/thread_solaris.inline.hpp deleted file mode 100644 index fdbc553cd22..00000000000 --- a/hotspot/src/os/solaris/vm/thread_solaris.inline.hpp +++ /dev/null @@ -1,50 +0,0 @@ -/* - * Copyright (c) 2002, 2015, 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. - * - */ - -#ifndef OS_SOLARIS_VM_THREAD_SOLARIS_INLINE_HPP -#define OS_SOLARIS_VM_THREAD_SOLARIS_INLINE_HPP - -#ifndef SHARE_VM_RUNTIME_THREAD_INLINE_HPP_SCOPE -#error "This file should only be included from thread.inline.hpp" -#endif - -#include "runtime/thread.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Thread::current is "hot" it's called > 128K times in the 1st 500 msecs of -// startup. -// ThreadLocalStorage::thread is warm -- it's called > 16K times in the same -// period. Thread::current() now calls ThreadLocalStorage::thread() directly. -// For SPARC, to avoid excessive register window spill-fill faults, -// we aggressively inline these routines. - -inline void ThreadLocalStorage::set_thread(Thread* thread) { - _thr_current = thread; -} - -inline Thread* ThreadLocalStorage::thread() { - return _thr_current; -} - -#endif // OS_SOLARIS_VM_THREAD_SOLARIS_INLINE_HPP diff --git a/hotspot/src/os/windows/vm/os_windows.cpp b/hotspot/src/os/windows/vm/os_windows.cpp index 94b4bc3bfe6..a0366810289 100644 --- a/hotspot/src/os/windows/vm/os_windows.cpp +++ b/hotspot/src/os/windows/vm/os_windows.cpp @@ -419,6 +419,8 @@ static unsigned __stdcall java_start(Thread* thread) { int pid = os::current_process_id(); _alloca(((pid ^ counter++) & 7) * 128); + thread->initialize_thread_current(); + OSThread* osthr = thread->osthread(); assert(osthr->get_state() == RUNNABLE, "invalid os thread state"); @@ -2146,7 +2148,7 @@ int os::signal_wait() { LONG Handle_Exception(struct _EXCEPTION_POINTERS* exceptionInfo, address handler) { - JavaThread* thread = JavaThread::current(); + JavaThread* thread = (JavaThread*) Thread::current_or_null(); // Save pc in thread #ifdef _M_IA64 // Do not blow up if no thread info available. @@ -2384,7 +2386,7 @@ LONG WINAPI topLevelExceptionFilter(struct _EXCEPTION_POINTERS* exceptionInfo) { address pc = (address) exceptionInfo->ContextRecord->Eip; #endif #endif - Thread* t = ThreadLocalStorage::get_thread_slow(); // slow & steady + Thread* t = Thread::current_or_null_safe(); // Handle SafeFetch32 and SafeFetchN exceptions. if (StubRoutines::is_safefetch_fault(pc)) { @@ -4011,27 +4013,6 @@ bool os::message_box(const char* title, const char* message) { return result == IDYES; } -int os::allocate_thread_local_storage() { - return TlsAlloc(); -} - - -void os::free_thread_local_storage(int index) { - TlsFree(index); -} - - -void os::thread_local_storage_at_put(int index, void* value) { - TlsSetValue(index, value); - assert(thread_local_storage_at(index) == value, "Just checking"); -} - - -void* os::thread_local_storage_at(int index) { - return TlsGetValue(index); -} - - #ifndef PRODUCT #ifndef _WIN64 // Helpers to check whether NX protection is enabled @@ -4079,6 +4060,9 @@ void os::init(void) { fatal("DuplicateHandle failed\n"); } main_thread_id = (int) GetCurrentThreadId(); + + // initialize fast thread access - only used for 32-bit + win32::initialize_thread_ptr_offset(); } // To install functions for atexit processing @@ -5177,9 +5161,7 @@ void Parker::park(bool isAbsolute, jlong time) { } } - JavaThread* thread = (JavaThread*)(Thread::current()); - assert(thread->is_Java_thread(), "Must be JavaThread"); - JavaThread *jt = (JavaThread *)thread; + JavaThread* thread = JavaThread::current(); // Don't wait if interrupted or already triggered if (Thread::is_interrupted(thread, false) || @@ -5187,16 +5169,16 @@ void Parker::park(bool isAbsolute, jlong time) { ResetEvent(_ParkEvent); return; } else { - ThreadBlockInVM tbivm(jt); + ThreadBlockInVM tbivm(thread); OSThreadWaitState osts(thread->osthread(), false /* not Object.wait() */); - jt->set_suspend_equivalent(); + thread->set_suspend_equivalent(); WaitForSingleObject(_ParkEvent, time); ResetEvent(_ParkEvent); // If externally suspended while waiting, re-suspend - if (jt->handle_special_suspend_equivalent_condition()) { - jt->java_suspend_self(); + if (thread->handle_special_suspend_equivalent_condition()) { + thread->java_suspend_self(); } } } @@ -5299,7 +5281,7 @@ LONG WINAPI os::win32::serialize_fault_filter(struct _EXCEPTION_POINTERS* e) { DWORD exception_code = e->ExceptionRecord->ExceptionCode; if (exception_code == EXCEPTION_ACCESS_VIOLATION) { - JavaThread* thread = (JavaThread*)ThreadLocalStorage::get_thread_slow(); + JavaThread* thread = JavaThread::current(); PEXCEPTION_RECORD exceptionRecord = e->ExceptionRecord; address addr = (address) exceptionRecord->ExceptionInformation[1]; @@ -6066,3 +6048,15 @@ int os::get_signal_number(const char* name) { return -1; } +// Fast current thread access + +int os::win32::_thread_ptr_offset = 0; + +static void call_wrapper_dummy() {} + +// We need to call the os_exception_wrapper once so that it sets +// up the offset from FS of the thread pointer. +void os::win32::initialize_thread_ptr_offset() { + os::os_exception_wrapper((java_call_t)call_wrapper_dummy, + NULL, NULL, NULL, NULL); +} diff --git a/hotspot/src/os/windows/vm/os_windows.hpp b/hotspot/src/os/windows/vm/os_windows.hpp index 08be4d06f70..19de558ce85 100644 --- a/hotspot/src/os/windows/vm/os_windows.hpp +++ b/hotspot/src/os/windows/vm/os_windows.hpp @@ -117,6 +117,17 @@ class win32 { // filter function to ignore faults on serializations page static LONG WINAPI serialize_fault_filter(struct _EXCEPTION_POINTERS* e); + + // Fast access to current thread +protected: + static int _thread_ptr_offset; +private: + static void initialize_thread_ptr_offset(); +public: + static inline void set_thread_ptr_offset(int offset) { + _thread_ptr_offset = offset; + } + static inline int get_thread_ptr_offset() { return _thread_ptr_offset; } }; /* diff --git a/hotspot/src/os/windows/vm/threadLocalStorage_windows.cpp b/hotspot/src/os/windows/vm/threadLocalStorage_windows.cpp new file mode 100644 index 00000000000..5648a672d71 --- /dev/null +++ b/hotspot/src/os/windows/vm/threadLocalStorage_windows.cpp @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2015, 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 "precompiled.hpp" +#include "runtime/threadLocalStorage.hpp" +#include + +static DWORD _thread_key; +static bool _initialized = false; + + +void ThreadLocalStorage::init() { + assert(!_initialized, "initializing TLS more than once!"); + _thread_key = TlsAlloc(); + // If this assert fails we will get a recursive assertion failure + // and not see the actual error message or get a hs_err file + assert(_thread_key != TLS_OUT_OF_INDEXES, "TlsAlloc failed: out of indices"); + _initialized = true; +} + +bool ThreadLocalStorage::is_initialized() { + return _initialized; +} + +Thread* ThreadLocalStorage::thread() { + // If this assert fails we will get a recursive assertion failure + // and not see the actual error message or get a hs_err file. + // Which most likely indicates we have taken an error path early in + // the initialization process, which is using Thread::current without + // checking TLS is initialized - see java.cpp vm_exit + assert(_initialized, "TLS not initialized yet!"); + Thread* current = (Thread*) TlsGetValue(_thread_key); + assert(current != 0 || GetLastError() == ERROR_SUCCESS, + "TlsGetValue failed with error code: %lu", GetLastError()); + return current; +} + +void ThreadLocalStorage::set_thread(Thread* current) { + assert(_initialized, "TLS not initialized yet!"); + BOOL res = TlsSetValue(_thread_key, current); + assert(res, "TlsSetValue failed with error code: %lu", GetLastError()); +} diff --git a/hotspot/src/os/windows/vm/thread_windows.inline.hpp b/hotspot/src/os/windows/vm/thread_windows.inline.hpp deleted file mode 100644 index 95dd17cecc7..00000000000 --- a/hotspot/src/os/windows/vm/thread_windows.inline.hpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 2002, 2010, 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. - * - */ - -#ifndef OS_WINDOWS_VM_THREAD_WINDOWS_INLINE_HPP -#define OS_WINDOWS_VM_THREAD_WINDOWS_INLINE_HPP - -#ifndef SHARE_VM_RUNTIME_THREAD_INLINE_HPP_SCOPE -#error "This file should only be included from thread.inline.hpp" -#endif - -#include "runtime/thread.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Contains inlined functions for class Thread and ThreadLocalStorage - -inline void ThreadLocalStorage::pd_invalidate_all() { return; } - -#endif // OS_WINDOWS_VM_THREAD_WINDOWS_INLINE_HPP diff --git a/hotspot/src/os_cpu/aix_ppc/vm/os_aix_ppc.cpp b/hotspot/src/os_cpu/aix_ppc/vm/os_aix_ppc.cpp index f0b867e88da..2369f2b09c2 100644 --- a/hotspot/src/os_cpu/aix_ppc/vm/os_aix_ppc.cpp +++ b/hotspot/src/os_cpu/aix_ppc/vm/os_aix_ppc.cpp @@ -167,7 +167,7 @@ JVM_handle_aix_signal(int sig, siginfo_t* info, void* ucVoid, int abort_if_unrec ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); // slow & steady + Thread* t = Thread::current_or_null_safe(); SignalHandlerMark shm(t); diff --git a/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.cpp b/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.cpp deleted file mode 100644 index 69d2f1dd5ca..00000000000 --- a/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.cpp +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2014 SAP AG. 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 "precompiled.hpp" -#include "runtime/threadLocalStorage.hpp" -#include "runtime/thread.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { - // Nothing we can do here for user-level thread. -} - -void ThreadLocalStorage::pd_init() { - // Nothing to do. -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.hpp b/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.hpp deleted file mode 100644 index 825b0b8435a..00000000000 --- a/hotspot/src/os_cpu/aix_ppc/vm/threadLS_aix_ppc.hpp +++ /dev/null @@ -1,36 +0,0 @@ -/* - * Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2013 SAP AG. 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. - * - */ - -#ifndef OS_CPU_AIX_PPC_VM_THREADLS_AIX_PPC_HPP -#define OS_CPU_AIX_PPC_VM_THREADLS_AIX_PPC_HPP - - // Processor dependent parts of ThreadLocalStorage - -public: - static Thread* thread() { - return (Thread *) os::thread_local_storage_at(thread_index()); - } - -#endif // OS_CPU_AIX_PPC_VM_THREADLS_AIX_PPC_HPP diff --git a/hotspot/src/os_cpu/bsd_x86/vm/assembler_bsd_x86.cpp b/hotspot/src/os_cpu/bsd_x86/vm/assembler_bsd_x86.cpp index 18bed6b0a6c..dd20ea833c8 100644 --- a/hotspot/src/os_cpu/bsd_x86/vm/assembler_bsd_x86.cpp +++ b/hotspot/src/os_cpu/bsd_x86/vm/assembler_bsd_x86.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -26,62 +26,7 @@ #include "asm/macroAssembler.hpp" #include "asm/macroAssembler.inline.hpp" #include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" -#ifndef _LP64 void MacroAssembler::int3() { call(RuntimeAddress(CAST_FROM_FN_PTR(address, os::breakpoint))); } - -void MacroAssembler::get_thread(Register thread) { - movl(thread, rsp); - shrl(thread, PAGE_SHIFT); - - ExternalAddress tls_base((address)ThreadLocalStorage::sp_map_addr()); - Address index(noreg, thread, Address::times_4); - ArrayAddress tls(tls_base, index); - - movptr(thread, tls); -} -#else -void MacroAssembler::int3() { - call(RuntimeAddress(CAST_FROM_FN_PTR(address, os::breakpoint))); -} - -void MacroAssembler::get_thread(Register thread) { - // call pthread_getspecific - // void * pthread_getspecific(pthread_key_t key); - if (thread != rax) { - push(rax); - } - push(rdi); - push(rsi); - push(rdx); - push(rcx); - push(r8); - push(r9); - push(r10); - // XXX - mov(r10, rsp); - andq(rsp, -16); - push(r10); - push(r11); - - movl(rdi, ThreadLocalStorage::thread_index()); - call(RuntimeAddress(CAST_FROM_FN_PTR(address, pthread_getspecific))); - - pop(r11); - pop(rsp); - pop(r10); - pop(r9); - pop(r8); - pop(rcx); - pop(rdx); - pop(rsi); - pop(rdi); - if (thread != rax) { - mov(thread, rax); - pop(rax); - } -} -#endif diff --git a/hotspot/src/os_cpu/bsd_x86/vm/os_bsd_x86.cpp b/hotspot/src/os_cpu/bsd_x86/vm/os_bsd_x86.cpp index ba2c97185a6..e9a10a61593 100644 --- a/hotspot/src/os_cpu/bsd_x86/vm/os_bsd_x86.cpp +++ b/hotspot/src/os_cpu/bsd_x86/vm/os_bsd_x86.cpp @@ -405,7 +405,7 @@ JVM_handle_bsd_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.cpp b/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.cpp deleted file mode 100644 index 0515fbc3fd0..00000000000 --- a/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.cpp +++ /dev/null @@ -1,92 +0,0 @@ -/* - * Copyright (c) 1999, 2010, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Map stack pointer (%esp) to thread pointer for faster TLS access -// -// Here we use a flat table for better performance. Getting current thread -// is down to one memory access (read _sp_map[%esp>>12]) in generated code -// and two in runtime code (-fPIC code needs an extra load for _sp_map). -// -// This code assumes stack page is not shared by different threads. It works -// in 32-bit VM when page size is 4K (or a multiple of 4K, if that matters). -// -// Notice that _sp_map is allocated in the bss segment, which is ZFOD -// (zero-fill-on-demand). While it reserves 4M address space upfront, -// actual memory pages are committed on demand. -// -// If an application creates and destroys a lot of threads, usually the -// stack space freed by a thread will soon get reused by new thread -// (this is especially true in NPTL or BsdThreads in fixed-stack mode). -// No memory page in _sp_map is wasted. -// -// However, it's still possible that we might end up populating & -// committing a large fraction of the 4M table over time, but the actual -// amount of live data in the table could be quite small. The max wastage -// is less than 4M bytes. If it becomes an issue, we could use madvise() -// with MADV_DONTNEED to reclaim unused (i.e. all-zero) pages in _sp_map. -// MADV_DONTNEED on Bsd keeps the virtual memory mapping, but zaps the -// physical memory page (i.e. similar to MADV_FREE on Solaris). - -#ifndef AMD64 -Thread* ThreadLocalStorage::_sp_map[1UL << (SP_BITLENGTH - PAGE_SHIFT)]; -#endif // !AMD64 - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing we can do here for user-level thread -} - -void ThreadLocalStorage::pd_init() { -#ifndef AMD64 - assert(align_size_down(os::vm_page_size(), PAGE_SIZE) == os::vm_page_size(), - "page size must be multiple of PAGE_SIZE"); -#endif // !AMD64 -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); - -#ifndef AMD64 - address stack_top = os::current_stack_base(); - size_t stack_size = os::current_stack_size(); - - for (address p = stack_top - stack_size; p < stack_top; p += PAGE_SIZE) { - // pd_set_thread() is called with non-NULL value when a new thread is - // created/attached, or with NULL value when a thread is about to exit. - // If both "thread" and the corresponding _sp_map[] entry are non-NULL, - // they should have the same value. Otherwise it might indicate that the - // stack page is shared by multiple threads. However, a more likely cause - // for this assertion to fail is that an attached thread exited without - // detaching itself from VM, which is a program error and could cause VM - // to crash. - assert(thread == NULL || _sp_map[(uintptr_t)p >> PAGE_SHIFT] == NULL || - thread == _sp_map[(uintptr_t)p >> PAGE_SHIFT], - "thread exited without detaching from VM??"); - _sp_map[(uintptr_t)p >> PAGE_SHIFT] = thread; - } -#endif // !AMD64 -} diff --git a/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.hpp b/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.hpp deleted file mode 100644 index 7fedb95f6e5..00000000000 --- a/hotspot/src/os_cpu/bsd_x86/vm/threadLS_bsd_x86.hpp +++ /dev/null @@ -1,56 +0,0 @@ -/* - * Copyright (c) 1999, 2010, 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. - * - */ - -#ifndef OS_CPU_BSD_X86_VM_THREADLS_BSD_X86_HPP -#define OS_CPU_BSD_X86_VM_THREADLS_BSD_X86_HPP - - // Processor dependent parts of ThreadLocalStorage - -#ifndef AMD64 - // map stack pointer to thread pointer - see notes in threadLS_bsd_x86.cpp - #define SP_BITLENGTH 32 -#ifndef PAGE_SHIFT - #define PAGE_SHIFT 12 - #define PAGE_SIZE (1UL << PAGE_SHIFT) -#endif - static Thread* _sp_map[1UL << (SP_BITLENGTH - PAGE_SHIFT)]; -#endif // !AMD64 - -public: - -#ifndef AMD64 - static Thread** sp_map_addr() { return _sp_map; } -#endif // !AMD64 - - static Thread* thread() { -#ifdef AMD64 - return (Thread*) os::thread_local_storage_at(thread_index()); -#else - uintptr_t sp; - __asm__ volatile ("movl %%esp, %0" : "=r" (sp)); - return _sp_map[sp >> PAGE_SHIFT]; -#endif // AMD64 - } - -#endif // OS_CPU_BSD_X86_VM_THREADLS_BSD_X86_HPP diff --git a/hotspot/src/os_cpu/bsd_zero/vm/assembler_bsd_zero.cpp b/hotspot/src/os_cpu/bsd_zero/vm/assembler_bsd_zero.cpp index ec56a35cd6b..cb6fbce1579 100644 --- a/hotspot/src/os_cpu/bsd_zero/vm/assembler_bsd_zero.cpp +++ b/hotspot/src/os_cpu/bsd_zero/vm/assembler_bsd_zero.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright 2009 Red Hat, Inc. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -23,10 +23,4 @@ * */ -#include "precompiled.hpp" -#include "asm/assembler.hpp" -#include "assembler_zero.inline.hpp" -#include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" - // This file is intentionally empty diff --git a/hotspot/src/os_cpu/bsd_zero/vm/os_bsd_zero.cpp b/hotspot/src/os_cpu/bsd_zero/vm/os_bsd_zero.cpp index 40a455688f5..ff580acb164 100644 --- a/hotspot/src/os_cpu/bsd_zero/vm/os_bsd_zero.cpp +++ b/hotspot/src/os_cpu/bsd_zero/vm/os_bsd_zero.cpp @@ -134,7 +134,7 @@ JVM_handle_bsd_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); SignalHandlerMark shm(t); diff --git a/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.cpp b/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.cpp deleted file mode 100644 index 73e5e829f46..00000000000 --- a/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.cpp +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved. - * Copyright 2007 Red Hat, Inc. - * 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing to do -} - -void ThreadLocalStorage::pd_init() { - // nothing to do -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.hpp b/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.hpp deleted file mode 100644 index 0360b581acb..00000000000 --- a/hotspot/src/os_cpu/bsd_zero/vm/threadLS_bsd_zero.hpp +++ /dev/null @@ -1,35 +0,0 @@ -/* - * Copyright (c) 2003, 2010, 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. - * - */ - -#ifndef OS_CPU_BSD_ZERO_VM_THREADLS_BSD_ZERO_HPP -#define OS_CPU_BSD_ZERO_VM_THREADLS_BSD_ZERO_HPP - -// Processor dependent parts of ThreadLocalStorage - - public: - static Thread* thread() { - return (Thread*) os::thread_local_storage_at(thread_index()); - } - -#endif // OS_CPU_BSD_ZERO_VM_THREADLS_BSD_ZERO_HPP diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/assembler_linux_aarch64.cpp b/hotspot/src/os_cpu/linux_aarch64/vm/assembler_linux_aarch64.cpp index 96dcfcf2992..920d94da7f1 100644 --- a/hotspot/src/os_cpu/linux_aarch64/vm/assembler_linux_aarch64.cpp +++ b/hotspot/src/os_cpu/linux_aarch64/vm/assembler_linux_aarch64.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, Red Hat Inc. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -23,32 +23,6 @@ * */ -#include "precompiled.hpp" -#include "asm/macroAssembler.hpp" -#include "asm/macroAssembler.inline.hpp" -#include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" +// nothing required here -// get_thread can be called anywhere inside generated code so we need -// to save whatever non-callee save context might get clobbered by the -// call to the C thread_local lookup call or, indeed, the call setup -// code. x86 appears to save C arg registers. - -void MacroAssembler::get_thread(Register dst) { - // call pthread_getspecific - // void * pthread_getspecific(pthread_key_t key); - - // Save all call-clobbered regs except dst, plus r19 and r20. - RegSet saved_regs = RegSet::range(r0, r20) + lr - dst; - push(saved_regs, sp); - mov(c_rarg0, ThreadLocalStorage::thread_index()); - mov(r19, CAST_FROM_FN_PTR(address, pthread_getspecific)); - blrt(r19, 1, 0, 1); - if (dst != c_rarg0) { - mov(dst, c_rarg0); - } - // restore pushed registers - pop(saved_regs, sp); -} - diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/os_linux_aarch64.cpp b/hotspot/src/os_cpu/linux_aarch64/vm/os_linux_aarch64.cpp index 7ae7e616052..20bc506de69 100644 --- a/hotspot/src/os_cpu/linux_aarch64/vm/os_linux_aarch64.cpp +++ b/hotspot/src/os_cpu/linux_aarch64/vm/os_linux_aarch64.cpp @@ -249,7 +249,7 @@ JVM_handle_linux_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.cpp b/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.cpp deleted file mode 100644 index e7b0d4dfa1b..00000000000 --- a/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.cpp +++ /dev/null @@ -1,42 +0,0 @@ -/* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 2014, Red Hat Inc. 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 "precompiled.hpp" -#include "runtime/threadLocalStorage.hpp" -#include "runtime/thread.inline.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing we can do here for user-level thread -} - -void ThreadLocalStorage::pd_init() { -} - -__thread Thread *aarch64_currentThread; - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); - aarch64_currentThread = thread; -} diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.hpp b/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.hpp deleted file mode 100644 index 73c6afc5ea8..00000000000 --- a/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.hpp +++ /dev/null @@ -1,37 +0,0 @@ -/* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. - * Copyright (c) 2014, Red Hat Inc. 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. - * - */ - -#ifndef OS_CPU_LINUX_AARCH64_VM_THREADLS_LINUX_AARCH64_HPP -#define OS_CPU_LINUX_AARCH64_VM_THREADLS_LINUX_AARCH64_HPP - - // Processor dependent parts of ThreadLocalStorage - -public: - - static Thread *thread() { - return aarch64_currentThread; - } - -#endif // OS_CPU_LINUX_AARCH64_VM_THREADLS_LINUX_AARCH64_HPP diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.s b/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.s new file mode 100644 index 00000000000..f541844b9d6 --- /dev/null +++ b/hotspot/src/os_cpu/linux_aarch64/vm/threadLS_linux_aarch64.s @@ -0,0 +1,44 @@ +// Copyright (c) 2015, Red Hat Inc. 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. + + // JavaThread::aarch64_get_thread_helper() + // + // Return the current thread pointer in x0. + // Clobber x1, flags. + // All other registers are preserved, + + .global _ZN10JavaThread25aarch64_get_thread_helperEv + .type _ZN10JavaThread25aarch64_get_thread_helperEv, %function + +_ZN10JavaThread25aarch64_get_thread_helperEv: + stp x29, x30, [sp, -16]! + adrp x0, :tlsdesc:_ZN6Thread12_thr_currentE + ldr x1, [x0, #:tlsdesc_lo12:_ZN6Thread12_thr_currentE] + add x0, x0, :tlsdesc_lo12:_ZN6Thread12_thr_currentE + .tlsdesccall _ZN6Thread12_thr_currentE + blr x1 + mrs x1, tpidr_el0 + add x0, x1, x0 + ldr x0, [x0] + ldp x29, x30, [sp], 16 + ret + + .size _ZN10JavaThread25aarch64_get_thread_helperEv, .-_ZN10JavaThread25aarch64_get_thread_helperEv diff --git a/hotspot/src/os_cpu/linux_aarch64/vm/thread_linux_aarch64.hpp b/hotspot/src/os_cpu/linux_aarch64/vm/thread_linux_aarch64.hpp index 96053ec31bd..70e37cecaeb 100644 --- a/hotspot/src/os_cpu/linux_aarch64/vm/thread_linux_aarch64.hpp +++ b/hotspot/src/os_cpu/linux_aarch64/vm/thread_linux_aarch64.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, Red Hat Inc. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -77,6 +77,8 @@ private: bool pd_get_top_frame(frame* fr_addr, void* ucontext, bool isInJava); public: + static Thread *aarch64_get_thread_helper(); + // These routines are only used on cpu architectures that // have separate register stacks (Itanium). static bool register_stack_overflow() { return false; } diff --git a/hotspot/src/os_cpu/linux_ppc/vm/os_linux_ppc.cpp b/hotspot/src/os_cpu/linux_ppc/vm/os_linux_ppc.cpp index 1c42912c4a7..4f87f6809d5 100644 --- a/hotspot/src/os_cpu/linux_ppc/vm/os_linux_ppc.cpp +++ b/hotspot/src/os_cpu/linux_ppc/vm/os_linux_ppc.cpp @@ -182,7 +182,7 @@ JVM_handle_linux_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); SignalHandlerMark shm(t); diff --git a/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.cpp b/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.cpp deleted file mode 100644 index 8b6aa99f89d..00000000000 --- a/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.cpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2013 SAP AG. 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 "precompiled.hpp" -#include "runtime/threadLocalStorage.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing we can do here for user-level thread -} - -void ThreadLocalStorage::pd_init() { - // Nothing to do -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.hpp b/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.hpp deleted file mode 100644 index ac9beffcc85..00000000000 --- a/hotspot/src/os_cpu/linux_ppc/vm/threadLS_linux_ppc.hpp +++ /dev/null @@ -1,36 +0,0 @@ -/* - * Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2013 SAP AG. 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. - * - */ - -#ifndef OS_CPU_LINUX_PPC_VM_THREADLS_LINUX_PPC_HPP -#define OS_CPU_LINUX_PPC_VM_THREADLS_LINUX_PPC_HPP - - // Processor dependent parts of ThreadLocalStorage - -public: - static Thread* thread() { - return (Thread *) os::thread_local_storage_at(thread_index()); - } - -#endif // OS_CPU_LINUX_PPC_VM_THREADLS_LINUX_PPC_HPP diff --git a/hotspot/src/os_cpu/linux_sparc/vm/os_linux_sparc.cpp b/hotspot/src/os_cpu/linux_sparc/vm/os_linux_sparc.cpp index 16457399fca..07942dfd3f9 100644 --- a/hotspot/src/os_cpu/linux_sparc/vm/os_linux_sparc.cpp +++ b/hotspot/src/os_cpu/linux_sparc/vm/os_linux_sparc.cpp @@ -541,7 +541,7 @@ JVM_handle_linux_signal(int sig, ucontext_t* ucFake = (ucontext_t*) ucVoid; sigcontext* uc = (sigcontext*)ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.cpp b/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.cpp deleted file mode 100644 index 3f291a06aa6..00000000000 --- a/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.cpp +++ /dev/null @@ -1,38 +0,0 @@ -/* - * Copyright (c) 1998, 2010, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { -} - -void ThreadLocalStorage::pd_init() { - // Nothing to do -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.hpp b/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.hpp deleted file mode 100644 index 0584108833a..00000000000 --- a/hotspot/src/os_cpu/linux_sparc/vm/threadLS_linux_sparc.hpp +++ /dev/null @@ -1,33 +0,0 @@ -/* - * Copyright (c) 1998, 2010, 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. - * - */ - -#ifndef OS_CPU_LINUX_SPARC_VM_THREADLS_LINUX_SPARC_HPP -#define OS_CPU_LINUX_SPARC_VM_THREADLS_LINUX_SPARC_HPP - -public: - static Thread* thread() { - return (Thread*) os::thread_local_storage_at(thread_index()); - } - -#endif // OS_CPU_LINUX_SPARC_VM_THREADLS_LINUX_SPARC_HPP diff --git a/hotspot/src/os_cpu/linux_x86/vm/assembler_linux_x86.cpp b/hotspot/src/os_cpu/linux_x86/vm/assembler_linux_x86.cpp index 5c0c07a0c91..dd20ea833c8 100644 --- a/hotspot/src/os_cpu/linux_x86/vm/assembler_linux_x86.cpp +++ b/hotspot/src/os_cpu/linux_x86/vm/assembler_linux_x86.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -26,85 +26,7 @@ #include "asm/macroAssembler.hpp" #include "asm/macroAssembler.inline.hpp" #include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" -#ifndef _LP64 void MacroAssembler::int3() { call(RuntimeAddress(CAST_FROM_FN_PTR(address, os::breakpoint))); } - -#ifdef MINIMIZE_RAM_USAGE - -void MacroAssembler::get_thread(Register thread) { - // call pthread_getspecific - // void * pthread_getspecific(pthread_key_t key); - if (thread != rax) push(rax); - push(rcx); - push(rdx); - - push(ThreadLocalStorage::thread_index()); - call(RuntimeAddress(CAST_FROM_FN_PTR(address, pthread_getspecific))); - increment(rsp, wordSize); - - pop(rdx); - pop(rcx); - if (thread != rax) { - mov(thread, rax); - pop(rax); - } -} - -#else -void MacroAssembler::get_thread(Register thread) { - movl(thread, rsp); - shrl(thread, PAGE_SHIFT); - - ExternalAddress tls_base((address)ThreadLocalStorage::sp_map_addr()); - Address index(noreg, thread, Address::times_4); - ArrayAddress tls(tls_base, index); - - movptr(thread, tls); -} -#endif // MINIMIZE_RAM_USAGE -#else -void MacroAssembler::int3() { - call(RuntimeAddress(CAST_FROM_FN_PTR(address, os::breakpoint))); -} - -void MacroAssembler::get_thread(Register thread) { - // call pthread_getspecific - // void * pthread_getspecific(pthread_key_t key); - if (thread != rax) { - push(rax); - } - push(rdi); - push(rsi); - push(rdx); - push(rcx); - push(r8); - push(r9); - push(r10); - // XXX - mov(r10, rsp); - andq(rsp, -16); - push(r10); - push(r11); - - movl(rdi, ThreadLocalStorage::thread_index()); - call(RuntimeAddress(CAST_FROM_FN_PTR(address, pthread_getspecific))); - - pop(r11); - pop(rsp); - pop(r10); - pop(r9); - pop(r8); - pop(rcx); - pop(rdx); - pop(rsi); - pop(rdi); - if (thread != rax) { - mov(thread, rax); - pop(rax); - } -} -#endif diff --git a/hotspot/src/os_cpu/linux_x86/vm/os_linux_x86.cpp b/hotspot/src/os_cpu/linux_x86/vm/os_linux_x86.cpp index fe1c4c59068..3a8de947a87 100644 --- a/hotspot/src/os_cpu/linux_x86/vm/os_linux_x86.cpp +++ b/hotspot/src/os_cpu/linux_x86/vm/os_linux_x86.cpp @@ -221,7 +221,7 @@ JVM_handle_linux_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.cpp b/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.cpp deleted file mode 100644 index addb46d6059..00000000000 --- a/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.cpp +++ /dev/null @@ -1,98 +0,0 @@ -/* - * Copyright (c) 1999, 2015, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Map stack pointer (%esp) to thread pointer for faster TLS access -// -// Here we use a flat table for better performance. Getting current thread -// is down to one memory access (read _sp_map[%esp>>12]) in generated code -// and two in runtime code (-fPIC code needs an extra load for _sp_map). -// -// This code assumes stack page is not shared by different threads. It works -// in 32-bit VM when page size is 4K (or a multiple of 4K, if that matters). -// -// Notice that _sp_map is allocated in the bss segment, which is ZFOD -// (zero-fill-on-demand). While it reserves 4M address space upfront, -// actual memory pages are committed on demand. -// -// If an application creates and destroys a lot of threads, usually the -// stack space freed by a thread will soon get reused by new thread. -// No memory page in _sp_map is wasted. -// -// However, it's still possible that we might end up populating & -// committing a large fraction of the 4M table over time, but the actual -// amount of live data in the table could be quite small. The max wastage -// is less than 4M bytes. If it becomes an issue, we could use madvise() -// with MADV_DONTNEED to reclaim unused (i.e. all-zero) pages in _sp_map. -// MADV_DONTNEED on Linux keeps the virtual memory mapping, but zaps the -// physical memory page (i.e. similar to MADV_FREE on Solaris). - -#if !defined(AMD64) && !defined(MINIMIZE_RAM_USAGE) -Thread* ThreadLocalStorage::_sp_map[1UL << (SP_BITLENGTH - PAGE_SHIFT)]; - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing we can do here for user-level thread -} - -void ThreadLocalStorage::pd_init() { - assert(align_size_down(os::vm_page_size(), PAGE_SIZE) == os::vm_page_size(), - "page size must be multiple of PAGE_SIZE"); -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); - address stack_top = os::current_stack_base(); - size_t stack_size = os::current_stack_size(); - - for (address p = stack_top - stack_size; p < stack_top; p += PAGE_SIZE) { - // pd_set_thread() is called with non-NULL value when a new thread is - // created/attached, or with NULL value when a thread is about to exit. - // If both "thread" and the corresponding _sp_map[] entry are non-NULL, - // they should have the same value. Otherwise it might indicate that the - // stack page is shared by multiple threads. However, a more likely cause - // for this assertion to fail is that an attached thread exited without - // detaching itself from VM, which is a program error and could cause VM - // to crash. - assert(thread == NULL || _sp_map[(uintptr_t)p >> PAGE_SHIFT] == NULL || - thread == _sp_map[(uintptr_t)p >> PAGE_SHIFT], - "thread exited without detaching from VM??"); - _sp_map[(uintptr_t)p >> PAGE_SHIFT] = thread; - } -} -#else - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing we can do here for user-level thread -} - -void ThreadLocalStorage::pd_init() { -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} -#endif // !AMD64 && !MINIMIZE_RAM_USAGE diff --git a/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.hpp b/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.hpp deleted file mode 100644 index f3f2f26f88a..00000000000 --- a/hotspot/src/os_cpu/linux_x86/vm/threadLS_linux_x86.hpp +++ /dev/null @@ -1,58 +0,0 @@ -/* - * Copyright (c) 1999, 2010, 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. - * - */ - -#ifndef OS_CPU_LINUX_X86_VM_THREADLS_LINUX_X86_HPP -#define OS_CPU_LINUX_X86_VM_THREADLS_LINUX_X86_HPP - - // Processor dependent parts of ThreadLocalStorage - -#if !defined(AMD64) && !defined(MINIMIZE_RAM_USAGE) - - // map stack pointer to thread pointer - see notes in threadLS_linux_x86.cpp - #define SP_BITLENGTH 32 - #define PAGE_SHIFT 12 - #define PAGE_SIZE (1UL << PAGE_SHIFT) - static Thread* _sp_map[1UL << (SP_BITLENGTH - PAGE_SHIFT)]; - -public: - - static Thread** sp_map_addr() { return _sp_map; } - - static Thread* thread() { - uintptr_t sp; - __asm__ volatile ("movl %%esp, %0" : "=r" (sp)); - return _sp_map[sp >> PAGE_SHIFT]; - } - -#else - -public: - - static Thread* thread() { - return (Thread*) os::thread_local_storage_at(thread_index()); - } - -#endif // AMD64 || MINIMIZE_RAM_USAGE - -#endif // OS_CPU_LINUX_X86_VM_THREADLS_LINUX_X86_HPP diff --git a/hotspot/src/os_cpu/linux_zero/vm/assembler_linux_zero.cpp b/hotspot/src/os_cpu/linux_zero/vm/assembler_linux_zero.cpp index ec56a35cd6b..cb6fbce1579 100644 --- a/hotspot/src/os_cpu/linux_zero/vm/assembler_linux_zero.cpp +++ b/hotspot/src/os_cpu/linux_zero/vm/assembler_linux_zero.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright 2009 Red Hat, Inc. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -23,10 +23,4 @@ * */ -#include "precompiled.hpp" -#include "asm/assembler.hpp" -#include "assembler_zero.inline.hpp" -#include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" - // This file is intentionally empty diff --git a/hotspot/src/os_cpu/linux_zero/vm/os_linux_zero.cpp b/hotspot/src/os_cpu/linux_zero/vm/os_linux_zero.cpp index b17d58e60b0..2b4ea0bd7b8 100644 --- a/hotspot/src/os_cpu/linux_zero/vm/os_linux_zero.cpp +++ b/hotspot/src/os_cpu/linux_zero/vm/os_linux_zero.cpp @@ -125,7 +125,7 @@ JVM_handle_linux_signal(int sig, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); SignalHandlerMark shm(t); diff --git a/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.cpp b/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.cpp deleted file mode 100644 index 73e5e829f46..00000000000 --- a/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.cpp +++ /dev/null @@ -1,40 +0,0 @@ -/* - * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved. - * Copyright 2007 Red Hat, Inc. - * 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -void ThreadLocalStorage::generate_code_for_get_thread() { - // nothing to do -} - -void ThreadLocalStorage::pd_init() { - // nothing to do -} - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.hpp b/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.hpp deleted file mode 100644 index 3fd4ccf973f..00000000000 --- a/hotspot/src/os_cpu/linux_zero/vm/threadLS_linux_zero.hpp +++ /dev/null @@ -1,35 +0,0 @@ -/* - * Copyright (c) 2003, 2010, 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. - * - */ - -#ifndef OS_CPU_LINUX_ZERO_VM_THREADLS_LINUX_ZERO_HPP -#define OS_CPU_LINUX_ZERO_VM_THREADLS_LINUX_ZERO_HPP - -// Processor dependent parts of ThreadLocalStorage - - public: - static Thread* thread() { - return (Thread*) os::thread_local_storage_at(thread_index()); - } - -#endif // OS_CPU_LINUX_ZERO_VM_THREADLS_LINUX_ZERO_HPP diff --git a/hotspot/src/os_cpu/solaris_sparc/vm/os_solaris_sparc.cpp b/hotspot/src/os_cpu/solaris_sparc/vm/os_solaris_sparc.cpp index d139566c72d..8ade2f62a5e 100644 --- a/hotspot/src/os_cpu/solaris_sparc/vm/os_solaris_sparc.cpp +++ b/hotspot/src/os_cpu/solaris_sparc/vm/os_solaris_sparc.cpp @@ -290,7 +290,7 @@ JVM_handle_solaris_signal(int sig, siginfo_t* info, void* ucVoid, int abort_if_unrecognized) { ucontext_t* uc = (ucontext_t*) ucVoid; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.cpp b/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.cpp deleted file mode 100644 index 30210a453cc..00000000000 --- a/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.cpp +++ /dev/null @@ -1,52 +0,0 @@ -/* - * Copyright (c) 1998, 2015, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// True thread-local variable -__thread Thread * ThreadLocalStorage::_thr_current = NULL; - -// Implementations needed to support the shared API - -void ThreadLocalStorage::pd_invalidate_all() {} // nothing to do - -bool ThreadLocalStorage::_initialized = false; - -void ThreadLocalStorage::init() { - _initialized = true; -} - -bool ThreadLocalStorage::is_initialized() { - return _initialized; -} - -Thread* ThreadLocalStorage::get_thread_slow() { - return thread(); -} - -extern "C" Thread* get_thread() { - return ThreadLocalStorage::thread(); -} diff --git a/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.hpp b/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.hpp deleted file mode 100644 index e3d96c87ae7..00000000000 --- a/hotspot/src/os_cpu/solaris_sparc/vm/threadLS_solaris_sparc.hpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 1998, 2015, 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. - * - */ - -#ifndef OS_CPU_SOLARIS_SPARC_VM_THREADLS_SOLARIS_SPARC_HPP -#define OS_CPU_SOLARIS_SPARC_VM_THREADLS_SOLARIS_SPARC_HPP - -// Solaris specific implementation involves simple, direct use -// of a compiler-based thread-local variable - -private: - static __thread Thread * _thr_current; - - static bool _initialized; // needed for shared API - -public: - static inline Thread* thread(); - -#endif // OS_CPU_SOLARIS_SPARC_VM_THREADLS_SOLARIS_SPARC_HPP diff --git a/hotspot/src/os_cpu/solaris_x86/vm/assembler_solaris_x86.cpp b/hotspot/src/os_cpu/solaris_x86/vm/assembler_solaris_x86.cpp index d4c0feccaa3..8d43f0378f4 100644 --- a/hotspot/src/os_cpu/solaris_x86/vm/assembler_solaris_x86.cpp +++ b/hotspot/src/os_cpu/solaris_x86/vm/assembler_solaris_x86.cpp @@ -25,8 +25,6 @@ #include "precompiled.hpp" #include "asm/macroAssembler.inline.hpp" #include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" -#include "runtime/thread.inline.hpp" void MacroAssembler::int3() { push(rax); @@ -37,33 +35,3 @@ void MacroAssembler::int3() { pop(rdx); pop(rax); } - -// This is simply a call to ThreadLocalStorage::thread() -void MacroAssembler::get_thread(Register thread) { - if (thread != rax) { - push(rax); - } - push(rdi); - push(rsi); - push(rdx); - push(rcx); - push(r8); - push(r9); - push(r10); - push(r11); - - call(RuntimeAddress(CAST_FROM_FN_PTR(address, ThreadLocalStorage::thread))); - - pop(r11); - pop(r10); - pop(r9); - pop(r8); - pop(rcx); - pop(rdx); - pop(rsi); - pop(rdi); - if (thread != rax) { - movl(thread, rax); - pop(rax); - } -} diff --git a/hotspot/src/os_cpu/solaris_x86/vm/os_solaris_x86.cpp b/hotspot/src/os_cpu/solaris_x86/vm/os_solaris_x86.cpp index 28845c326b8..5f140a596de 100644 --- a/hotspot/src/os_cpu/solaris_x86/vm/os_solaris_x86.cpp +++ b/hotspot/src/os_cpu/solaris_x86/vm/os_solaris_x86.cpp @@ -346,7 +346,7 @@ JVM_handle_solaris_signal(int sig, siginfo_t* info, void* ucVoid, } #endif // !AMD64 - Thread* t = ThreadLocalStorage::get_thread_slow(); // slow & steady + Thread* t = Thread::current_or_null_safe(); // Must do this before SignalHandlerMark, if crash protection installed we will longjmp away // (no destructors can be run) diff --git a/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.cpp b/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.cpp deleted file mode 100644 index 30210a453cc..00000000000 --- a/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.cpp +++ /dev/null @@ -1,52 +0,0 @@ -/* - * Copyright (c) 1998, 2015, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// True thread-local variable -__thread Thread * ThreadLocalStorage::_thr_current = NULL; - -// Implementations needed to support the shared API - -void ThreadLocalStorage::pd_invalidate_all() {} // nothing to do - -bool ThreadLocalStorage::_initialized = false; - -void ThreadLocalStorage::init() { - _initialized = true; -} - -bool ThreadLocalStorage::is_initialized() { - return _initialized; -} - -Thread* ThreadLocalStorage::get_thread_slow() { - return thread(); -} - -extern "C" Thread* get_thread() { - return ThreadLocalStorage::thread(); -} diff --git a/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.hpp b/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.hpp deleted file mode 100644 index 4f8da7bcb65..00000000000 --- a/hotspot/src/os_cpu/solaris_x86/vm/threadLS_solaris_x86.hpp +++ /dev/null @@ -1,39 +0,0 @@ -/* - * Copyright (c) 1998, 2015, 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. - * - */ - -#ifndef OS_CPU_SOLARIS_X86_VM_THREADLS_SOLARIS_X86_HPP -#define OS_CPU_SOLARIS_X86_VM_THREADLS_SOLARIS_X86_HPP - -// Solaris specific implementation involves simple, direct use -// of a compiler-based thread-local variable - -private: - static __thread Thread * _thr_current; - - static bool _initialized; // needed for shared API - -public: - static inline Thread* thread(); - -#endif // OS_CPU_SOLARIS_X86_VM_THREADLS_SOLARIS_X86_HPP diff --git a/hotspot/src/os_cpu/windows_x86/vm/assembler_windows_x86.cpp b/hotspot/src/os_cpu/windows_x86/vm/assembler_windows_x86.cpp index ce48421c6d4..71e1e8b6aad 100644 --- a/hotspot/src/os_cpu/windows_x86/vm/assembler_windows_x86.cpp +++ b/hotspot/src/os_cpu/windows_x86/vm/assembler_windows_x86.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -26,8 +26,6 @@ #include "asm/macroAssembler.hpp" #include "asm/macroAssembler.inline.hpp" #include "runtime/os.hpp" -#include "runtime/threadLocalStorage.hpp" - void MacroAssembler::int3() { emit_int8((unsigned char)0xCC); @@ -58,44 +56,11 @@ void MacroAssembler::get_thread(Register thread) { prefix(FS_segment); movptr(thread, null); - assert(ThreadLocalStorage::get_thread_ptr_offset() != 0, + assert(os::win32::get_thread_ptr_offset() != 0, "Thread Pointer Offset has not been initialized"); - movl(thread, Address(thread, ThreadLocalStorage::get_thread_ptr_offset())); + movl(thread, Address(thread, os::win32::get_thread_ptr_offset())); } -#else -// call (Thread*)TlsGetValue(thread_index()); -void MacroAssembler::get_thread(Register thread) { - if (thread != rax) { - push(rax); - } - push(rdi); - push(rsi); - push(rdx); - push(rcx); - push(r8); - push(r9); - push(r10); - // XXX - mov(r10, rsp); - andq(rsp, -16); - push(r10); - push(r11); - movl(c_rarg0, ThreadLocalStorage::thread_index()); - call(RuntimeAddress((address)TlsGetValue)); +// #else - use shared x86 implementation in cpu/x86/vm/macroAssembler_x86.cpp - pop(r11); - pop(rsp); - pop(r10); - pop(r9); - pop(r8); - pop(rcx); - pop(rdx); - pop(rsi); - pop(rdi); - if (thread != rax) { - mov(thread, rax); - pop(rax); - } -} #endif diff --git a/hotspot/src/os_cpu/windows_x86/vm/os_windows_x86.cpp b/hotspot/src/os_cpu/windows_x86/vm/os_windows_x86.cpp index f17330ace65..66933305a0c 100644 --- a/hotspot/src/os_cpu/windows_x86/vm/os_windows_x86.cpp +++ b/hotspot/src/os_cpu/windows_x86/vm/os_windows_x86.cpp @@ -85,14 +85,14 @@ void os::os_exception_wrapper(java_call_t f, JavaValue* value, const methodHandl // volatile Thread* wrapperthread = thread; - if ( ThreadLocalStorage::get_thread_ptr_offset() == 0 ) { + if (os::win32::get_thread_ptr_offset() == 0) { int thread_ptr_offset; __asm { lea eax, dword ptr wrapperthread; sub eax, dword ptr FS:[0H]; mov thread_ptr_offset, eax }; - ThreadLocalStorage::set_thread_ptr_offset(thread_ptr_offset); + os::win32::set_thread_ptr_offset(thread_ptr_offset); } #ifdef ASSERT // Verify that the offset hasn't changed since we initally captured @@ -105,7 +105,7 @@ void os::os_exception_wrapper(java_call_t f, JavaValue* value, const methodHandl sub eax, dword ptr FS:[0H]; mov test_thread_ptr_offset, eax }; - assert(test_thread_ptr_offset == ThreadLocalStorage::get_thread_ptr_offset(), + assert(test_thread_ptr_offset == os::win32::get_thread_ptr_offset(), "thread pointer offset from SEH changed"); } #endif // ASSERT diff --git a/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.cpp b/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.cpp deleted file mode 100644 index f363816d973..00000000000 --- a/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.cpp +++ /dev/null @@ -1,49 +0,0 @@ -/* - * Copyright (c) 1998, 2010, 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 "precompiled.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Provides an entry point we can link against and -// a buffer we can emit code into. The buffer is -// filled by ThreadLocalStorage::generate_code_for_get_thread -// and called from ThreadLocalStorage::thread() - -int ThreadLocalStorage::_thread_ptr_offset = 0; - -static void call_wrapper_dummy() {} - -// We need to call the os_exception_wrapper once so that it sets -// up the offset from FS of the thread pointer. -void ThreadLocalStorage::generate_code_for_get_thread() { - os::os_exception_wrapper( (java_call_t)call_wrapper_dummy, - NULL, NULL, NULL, NULL); -} - -void ThreadLocalStorage::pd_init() { } - -void ThreadLocalStorage::pd_set_thread(Thread* thread) { - os::thread_local_storage_at_put(ThreadLocalStorage::thread_index(), thread); -} diff --git a/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.hpp b/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.hpp deleted file mode 100644 index a764d7376f5..00000000000 --- a/hotspot/src/os_cpu/windows_x86/vm/threadLS_windows_x86.hpp +++ /dev/null @@ -1,49 +0,0 @@ -/* - * Copyright (c) 1998, 2010, 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. - * - */ - -#ifndef OS_CPU_WINDOWS_X86_VM_THREADLS_WINDOWS_X86_HPP -#define OS_CPU_WINDOWS_X86_VM_THREADLS_WINDOWS_X86_HPP - -// Processor dependent parts of ThreadLocalStorage - -protected: - - static int _thread_ptr_offset; - -public: - - // Java Thread - static inline Thread* thread() { - return (Thread*)TlsGetValue(thread_index()); - } - - static inline Thread* get_thread() { - return (Thread*)TlsGetValue(thread_index()); - } - - static inline void set_thread_ptr_offset( int offset ) { _thread_ptr_offset = offset; } - - static inline int get_thread_ptr_offset() { return _thread_ptr_offset; } - -#endif // OS_CPU_WINDOWS_X86_VM_THREADLS_WINDOWS_X86_HPP diff --git a/hotspot/src/share/vm/code/nmethod.cpp b/hotspot/src/share/vm/code/nmethod.cpp index 307f0f71b98..60a864b38e0 100644 --- a/hotspot/src/share/vm/code/nmethod.cpp +++ b/hotspot/src/share/vm/code/nmethod.cpp @@ -2609,7 +2609,7 @@ address nmethod::continuation_for_implicit_exception(address pc) { int cont_offset = ImplicitExceptionTable(this).at( exception_offset ); #ifdef ASSERT if (cont_offset == 0) { - Thread* thread = ThreadLocalStorage::get_thread_slow(); + Thread* thread = Thread::current(); ResetNoHandleMark rnm; // Might be called from LEAF/QUICK ENTRY HandleMark hm(thread); ResourceMark rm(thread); diff --git a/hotspot/src/share/vm/gc/cms/concurrentMarkSweepThread.cpp b/hotspot/src/share/vm/gc/cms/concurrentMarkSweepThread.cpp index 8d19ad31278..bb1ff998d8d 100644 --- a/hotspot/src/share/vm/gc/cms/concurrentMarkSweepThread.cpp +++ b/hotspot/src/share/vm/gc/cms/concurrentMarkSweepThread.cpp @@ -144,9 +144,6 @@ void ConcurrentMarkSweepThread::run() { _cmst = NULL; Terminator_lock->notify(); } - - // Thread destructor usually does this.. - ThreadLocalStorage::set_thread(NULL); } #ifndef PRODUCT diff --git a/hotspot/src/share/vm/gc/g1/g1HotCardCache.hpp b/hotspot/src/share/vm/gc/g1/g1HotCardCache.hpp index 36a609db30d..5ec0fbbcb2a 100644 --- a/hotspot/src/share/vm/gc/g1/g1HotCardCache.hpp +++ b/hotspot/src/share/vm/gc/g1/g1HotCardCache.hpp @@ -123,7 +123,7 @@ class G1HotCardCache: public CHeapObj { // Resets the hot card cache and discards the entries. void reset_hot_cache() { assert(SafepointSynchronize::is_at_safepoint(), "Should be at a safepoint"); - assert(Thread::current_noinline()->is_VM_thread(), "Current thread should be the VMthread"); + assert(Thread::current()->is_VM_thread(), "Current thread should be the VMthread"); if (default_use_cache()) { reset_hot_cache_internal(); } diff --git a/hotspot/src/share/vm/gc/parallel/gcTaskThread.cpp b/hotspot/src/share/vm/gc/parallel/gcTaskThread.cpp index 8e3f62c0c57..c8ce9969b38 100644 --- a/hotspot/src/share/vm/gc/parallel/gcTaskThread.cpp +++ b/hotspot/src/share/vm/gc/parallel/gcTaskThread.cpp @@ -1,4 +1,3 @@ - /* * Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. @@ -96,7 +95,6 @@ void GCTaskThread::print_task_time_stamps() { void GCTaskThread::run() { // Set up the thread for stack overflow support this->record_stack_base_and_size(); - this->initialize_thread_local_storage(); this->initialize_named_thread(); // Bind yourself to your processor. if (processor_id() != GCTaskManager::sentinel_worker()) { diff --git a/hotspot/src/share/vm/gc/shared/concurrentGCThread.cpp b/hotspot/src/share/vm/gc/shared/concurrentGCThread.cpp index 11146d8750e..bf095b43e42 100644 --- a/hotspot/src/share/vm/gc/shared/concurrentGCThread.cpp +++ b/hotspot/src/share/vm/gc/shared/concurrentGCThread.cpp @@ -51,7 +51,6 @@ void ConcurrentGCThread::create_and_start() { void ConcurrentGCThread::initialize_in_thread() { this->record_stack_base_and_size(); - this->initialize_thread_local_storage(); this->initialize_named_thread(); this->set_active_handles(JNIHandleBlock::allocate_block()); // From this time Thread::current() should be working. @@ -74,9 +73,6 @@ void ConcurrentGCThread::terminate() { _has_terminated = true; Terminator_lock->notify(); } - - // Thread destructor usually does this.. - ThreadLocalStorage::set_thread(NULL); } static void _sltLoop(JavaThread* thread, TRAPS) { diff --git a/hotspot/src/share/vm/gc/shared/workgroup.cpp b/hotspot/src/share/vm/gc/shared/workgroup.cpp index 24295694d09..22e231ffc0c 100644 --- a/hotspot/src/share/vm/gc/shared/workgroup.cpp +++ b/hotspot/src/share/vm/gc/shared/workgroup.cpp @@ -275,7 +275,6 @@ void AbstractGangWorker::run() { } void AbstractGangWorker::initialize() { - this->initialize_thread_local_storage(); this->record_stack_base_and_size(); this->initialize_named_thread(); assert(_gang != NULL, "No gang to run in"); diff --git a/hotspot/src/share/vm/memory/allocation.cpp b/hotspot/src/share/vm/memory/allocation.cpp index b6287bd5ed1..c86eb5174ab 100644 --- a/hotspot/src/share/vm/memory/allocation.cpp +++ b/hotspot/src/share/vm/memory/allocation.cpp @@ -790,7 +790,7 @@ void Arena::free_malloced_objects(Chunk* chunk, char* hwm, char* max, char* hwm2 ReallocMark::ReallocMark() { #ifdef ASSERT - Thread *thread = ThreadLocalStorage::get_thread_slow(); + Thread *thread = Thread::current(); _nesting = thread->resource_area()->nesting(); #endif } diff --git a/hotspot/src/share/vm/memory/resourceArea.hpp b/hotspot/src/share/vm/memory/resourceArea.hpp index 02dffc7a4b4..65ae36a1646 100644 --- a/hotspot/src/share/vm/memory/resourceArea.hpp +++ b/hotspot/src/share/vm/memory/resourceArea.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -121,7 +121,7 @@ protected: debug_only(_area->_nesting++;) assert( _area->_nesting > 0, "must stack allocate RMs" ); #ifdef ASSERT - Thread* thread = ThreadLocalStorage::thread(); + Thread* thread = Thread::current_or_null(); if (thread != NULL) { _thread = thread; _previous_resource_mark = thread->current_resource_mark(); diff --git a/hotspot/src/share/vm/oops/oopsHierarchy.cpp b/hotspot/src/share/vm/oops/oopsHierarchy.cpp index b2fdba28d3f..7a0bf50d6e0 100644 --- a/hotspot/src/share/vm/oops/oopsHierarchy.cpp +++ b/hotspot/src/share/vm/oops/oopsHierarchy.cpp @@ -35,7 +35,7 @@ void oop::register_oop() { assert (CheckUnhandledOops, "should only call when CheckUnhandledOops"); if (!Universe::is_fully_initialized()) return; // This gets expensive, which is why checking unhandled oops is on a switch. - Thread* t = ThreadLocalStorage::thread(); + Thread* t = Thread::current_or_null(); if (t != NULL && t->is_Java_thread()) { frame fr = os::current_frame(); // This points to the oop creator, I guess current frame points to caller @@ -48,7 +48,7 @@ void oop::unregister_oop() { assert (CheckUnhandledOops, "should only call when CheckUnhandledOops"); if (!Universe::is_fully_initialized()) return; // This gets expensive, which is why checking unhandled oops is on a switch. - Thread* t = ThreadLocalStorage::thread(); + Thread* t = Thread::current_or_null(); if (t != NULL && t->is_Java_thread()) { t->unhandled_oops()->unregister_unhandled_oop(this); } diff --git a/hotspot/src/share/vm/precompiled/precompiled.hpp b/hotspot/src/share/vm/precompiled/precompiled.hpp index cb729d28212..bd79327acc2 100644 --- a/hotspot/src/share/vm/precompiled/precompiled.hpp +++ b/hotspot/src/share/vm/precompiled/precompiled.hpp @@ -203,7 +203,6 @@ # include "runtime/stubRoutines.hpp" # include "runtime/synchronizer.hpp" # include "runtime/thread.hpp" -# include "runtime/threadLocalStorage.hpp" # include "runtime/timer.hpp" # include "runtime/unhandledOops.hpp" # include "runtime/vframe.hpp" diff --git a/hotspot/src/share/vm/prims/jni.cpp b/hotspot/src/share/vm/prims/jni.cpp index b4a278faa3a..8fa5b4e83b7 100644 --- a/hotspot/src/share/vm/prims/jni.cpp +++ b/hotspot/src/share/vm/prims/jni.cpp @@ -4175,7 +4175,7 @@ static jint attach_current_thread(JavaVM *vm, void **penv, void *_args, bool dae } */ - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null(); if (t != NULL) { // If the thread has been attached this operation is a no-op *(JNIEnv**)penv = ((JavaThread*) t)->jni_environment(); @@ -4190,10 +4190,8 @@ static jint attach_current_thread(JavaVM *vm, void **penv, void *_args, bool dae // initializing the Java level thread object. Hence, the correct state must // be set in order for the Safepoint code to deal with it correctly. thread->set_thread_state(_thread_in_vm); - // Must do this before initialize_thread_local_storage thread->record_stack_base_and_size(); - - thread->initialize_thread_local_storage(); + thread->initialize_thread_current(); if (!os::create_attached_thread(thread)) { delete thread; @@ -4300,8 +4298,8 @@ jint JNICALL jni_DetachCurrentThread(JavaVM *vm) { JNIWrapper("DetachCurrentThread"); - // If the thread has been deattacted the operations is a no-op - if (ThreadLocalStorage::thread() == NULL) { + // If the thread has already been detached the operation is a no-op + if (Thread::current_or_null() == NULL) { HOTSPOT_JNI_DETACHCURRENTTHREAD_RETURN(JNI_OK); return JNI_OK; } @@ -4358,7 +4356,7 @@ jint JNICALL jni_GetEnv(JavaVM *vm, void **penv, jint version) { #define JVMPI_VERSION_1_2 ((jint)0x10000003) #endif // !JVMPI_VERSION_1 - Thread* thread = ThreadLocalStorage::thread(); + Thread* thread = Thread::current_or_null(); if (thread != NULL && thread->is_Java_thread()) { if (Threads::is_supported_jni_version_including_1_1(version)) { *(JNIEnv**)penv = ((JavaThread*) thread)->jni_environment(); diff --git a/hotspot/src/share/vm/prims/jniCheck.cpp b/hotspot/src/share/vm/prims/jniCheck.cpp index d8839050411..8ba2212ac78 100644 --- a/hotspot/src/share/vm/prims/jniCheck.cpp +++ b/hotspot/src/share/vm/prims/jniCheck.cpp @@ -87,9 +87,9 @@ static struct JNINativeInterface_ * unchecked_jni_NativeInterface; #define JNI_ENTRY_CHECKED(result_type, header) \ extern "C" { \ result_type JNICALL header { \ - JavaThread* thr = (JavaThread*)ThreadLocalStorage::get_thread_slow();\ + JavaThread* thr = (JavaThread*) Thread::current_or_null(); \ if (thr == NULL || !thr->is_Java_thread()) { \ - tty->print_cr("%s", fatal_using_jnienv_in_nonjava); \ + tty->print_cr("%s", fatal_using_jnienv_in_nonjava); \ os::abort(true); \ } \ JNIEnv* xenv = thr->jni_environment(); \ diff --git a/hotspot/src/share/vm/prims/jvmtiEnter.xsl b/hotspot/src/share/vm/prims/jvmtiEnter.xsl index 6842d797b23..cee47ec40e2 100644 --- a/hotspot/src/share/vm/prims/jvmtiEnter.xsl +++ b/hotspot/src/share/vm/prims/jvmtiEnter.xsl @@ -494,7 +494,7 @@ static jvmtiError JNICALL } - Thread* this_thread = (Thread*)ThreadLocalStorage::thread(); + Thread* this_thread = Thread::current_or_null(); @@ -528,7 +528,7 @@ static jvmtiError JNICALL return JVMTI_ERROR_WRONG_PHASE; } - Thread* this_thread = (Thread*)ThreadLocalStorage::thread(); + Thread* this_thread = Thread::current_or_null(); @@ -558,7 +558,7 @@ static jvmtiError JNICALL if (Threads::number_of_threads() != 0) { - Thread* this_thread = (Thread*)ThreadLocalStorage::thread(); + Thread* this_thread = Thread::current_or_null(); @@ -567,7 +567,7 @@ static jvmtiError JNICALL if (Threads::number_of_threads() == 0) { transition = false; } else { - this_thread = (Thread*)ThreadLocalStorage::thread(); + this_thread = Thread::current_or_null(); transition = ((this_thread != NULL) && !this_thread->is_VM_thread() && !this_thread->is_ConcurrentGC_thread()); } if (transition) { diff --git a/hotspot/src/share/vm/prims/jvmtiExport.cpp b/hotspot/src/share/vm/prims/jvmtiExport.cpp index 1e061689c05..54a4bac002d 100644 --- a/hotspot/src/share/vm/prims/jvmtiExport.cpp +++ b/hotspot/src/share/vm/prims/jvmtiExport.cpp @@ -374,7 +374,7 @@ JvmtiExport::get_jvmti_interface(JavaVM *jvm, void **penv, jint version) { } if (JvmtiEnv::get_phase() == JVMTI_PHASE_LIVE) { - JavaThread* current_thread = (JavaThread*) ThreadLocalStorage::thread(); + JavaThread* current_thread = JavaThread::current(); // transition code: native to VM ThreadInVMfromNative __tiv(current_thread); VM_ENTRY_BASE(jvmtiEnv*, JvmtiExport::get_jvmti_interface, current_thread) @@ -1901,7 +1901,7 @@ void JvmtiExport::post_dynamic_code_generated_while_holding_locks(const char* na // Collect all the vm internally allocated objects which are visible to java world void JvmtiExport::record_vm_internal_object_allocation(oop obj) { - Thread* thread = ThreadLocalStorage::thread(); + Thread* thread = Thread::current_or_null(); if (thread != NULL && thread->is_Java_thread()) { // Can not take safepoint here. No_Safepoint_Verifier no_sfpt; @@ -2436,7 +2436,7 @@ NoJvmtiVMObjectAllocMark::NoJvmtiVMObjectAllocMark() : _collector(NULL) { if (!JvmtiExport::should_post_vm_object_alloc()) { return; } - Thread* thread = ThreadLocalStorage::thread(); + Thread* thread = Thread::current_or_null(); if (thread != NULL && thread->is_Java_thread()) { JavaThread* current_thread = (JavaThread*)thread; JvmtiThreadState *state = current_thread->jvmti_thread_state(); diff --git a/hotspot/src/share/vm/prims/jvmtiUtil.hpp b/hotspot/src/share/vm/prims/jvmtiUtil.hpp index f840d985d51..c3c4a32084b 100644 --- a/hotspot/src/share/vm/prims/jvmtiUtil.hpp +++ b/hotspot/src/share/vm/prims/jvmtiUtil.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2010, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -79,7 +79,7 @@ class SafeResourceMark : public ResourceMark { if (Threads::number_of_threads() == 0) { return JvmtiUtil::single_threaded_resource_area(); } - thread = ThreadLocalStorage::thread(); + thread = Thread::current_or_null(); if (thread == NULL) { return JvmtiUtil::single_threaded_resource_area(); } diff --git a/hotspot/src/share/vm/runtime/interfaceSupport.cpp b/hotspot/src/share/vm/runtime/interfaceSupport.cpp index a65fca64e40..e8dcb72d2fb 100644 --- a/hotspot/src/share/vm/runtime/interfaceSupport.cpp +++ b/hotspot/src/share/vm/runtime/interfaceSupport.cpp @@ -32,7 +32,6 @@ #include "runtime/interfaceSupport.hpp" #include "runtime/orderAccess.inline.hpp" #include "runtime/os.inline.hpp" -#include "runtime/threadLocalStorage.hpp" #include "runtime/vframe.hpp" #include "utilities/preserveException.hpp" diff --git a/hotspot/src/share/vm/runtime/interfaceSupport.hpp b/hotspot/src/share/vm/runtime/interfaceSupport.hpp index fc5374df4fc..aa628e3a7fa 100644 --- a/hotspot/src/share/vm/runtime/interfaceSupport.hpp +++ b/hotspot/src/share/vm/runtime/interfaceSupport.hpp @@ -562,7 +562,7 @@ extern "C" { \ #define JVM_ENTRY_NO_ENV(result_type, header) \ extern "C" { \ result_type JNICALL header { \ - JavaThread* thread = (JavaThread*)ThreadLocalStorage::thread(); \ + JavaThread* thread = JavaThread::current(); \ ThreadInVMfromNative __tiv(thread); \ debug_only(VMNativeEntryWrapper __vew;) \ VM_ENTRY_BASE(result_type, header, thread) diff --git a/hotspot/src/share/vm/runtime/java.cpp b/hotspot/src/share/vm/runtime/java.cpp index 1ab815c7d52..386935633a9 100644 --- a/hotspot/src/share/vm/runtime/java.cpp +++ b/hotspot/src/share/vm/runtime/java.cpp @@ -512,10 +512,10 @@ void before_exit(JavaThread * thread) { } void vm_exit(int code) { - Thread* thread = ThreadLocalStorage::is_initialized() ? - ThreadLocalStorage::get_thread_slow() : NULL; + Thread* thread = + ThreadLocalStorage::is_initialized() ? Thread::current_or_null() : NULL; if (thread == NULL) { - // we have serious problems -- just exit + // very early initialization failure -- just exit vm_direct_exit(code); } @@ -551,8 +551,7 @@ void vm_perform_shutdown_actions() { // Calling 'exit_globals()' will disable thread-local-storage and cause all // kinds of assertions to trigger in debug mode. if (is_init_completed()) { - Thread* thread = ThreadLocalStorage::is_initialized() ? - ThreadLocalStorage::get_thread_slow() : NULL; + Thread* thread = Thread::current_or_null(); if (thread != NULL && thread->is_Java_thread()) { // We are leaving the VM, set state to native (in case any OS exit // handlers call back to the VM) @@ -606,7 +605,7 @@ void vm_exit_during_initialization(Handle exception) { // If there are exceptions on this thread it must be cleared // first and here. Any future calls to EXCEPTION_MARK requires // that no pending exceptions exist. - Thread *THREAD = Thread::current(); + Thread *THREAD = Thread::current(); // can't be NULL if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; } diff --git a/hotspot/src/share/vm/runtime/mutex.cpp b/hotspot/src/share/vm/runtime/mutex.cpp index 954ad8e330e..81fa378d05c 100644 --- a/hotspot/src/share/vm/runtime/mutex.cpp +++ b/hotspot/src/share/vm/runtime/mutex.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 2015, 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 @@ -1035,10 +1035,10 @@ void Monitor::jvm_raw_lock() { Exeunt: assert(ILocked(), "invariant"); assert(_owner == NULL, "invariant"); - // This can potentially be called by non-java Threads. Thus, the ThreadLocalStorage + // This can potentially be called by non-java Threads. Thus, the Thread::current_or_null() // might return NULL. Don't call set_owner since it will break on an NULL owner // Consider installing a non-null "ANON" distinguished value instead of just NULL. - _owner = ThreadLocalStorage::thread(); + _owner = Thread::current_or_null(); return; } diff --git a/hotspot/src/share/vm/runtime/mutexLocker.cpp b/hotspot/src/share/vm/runtime/mutexLocker.cpp index 3c8c4df149e..71557ddd910 100644 --- a/hotspot/src/share/vm/runtime/mutexLocker.cpp +++ b/hotspot/src/share/vm/runtime/mutexLocker.cpp @@ -27,7 +27,6 @@ #include "runtime/os.inline.hpp" #include "runtime/safepoint.hpp" #include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" #include "runtime/vmThread.hpp" // Mutexes used in the VM (see comment in mutexLocker.hpp): diff --git a/hotspot/src/share/vm/runtime/os.cpp b/hotspot/src/share/vm/runtime/os.cpp index d1cc3d76074..1eb8323c464 100644 --- a/hotspot/src/share/vm/runtime/os.cpp +++ b/hotspot/src/share/vm/runtime/os.cpp @@ -420,28 +420,6 @@ void* os::native_java_library() { } #endif } - static jboolean onLoaded = JNI_FALSE; - if (onLoaded) { - // We may have to wait to fire OnLoad until TLS is initialized. - if (ThreadLocalStorage::is_initialized()) { - // The JNI_OnLoad handling is normally done by method load in - // java.lang.ClassLoader$NativeLibrary, but the VM loads the base library - // explicitly so we have to check for JNI_OnLoad as well - const char *onLoadSymbols[] = JNI_ONLOAD_SYMBOLS; - JNI_OnLoad_t JNI_OnLoad = CAST_TO_FN_PTR( - JNI_OnLoad_t, dll_lookup(_native_java_library, onLoadSymbols[0])); - if (JNI_OnLoad != NULL) { - JavaThread* thread = JavaThread::current(); - ThreadToNativeFromVM ttn(thread); - HandleMark hm(thread); - jint ver = (*JNI_OnLoad)(&main_vm, NULL); - onLoaded = JNI_TRUE; - if (!Threads::is_supported_jni_version_including_1_1(ver)) { - vm_exit_during_initialization("Unsupported JNI version"); - } - } - } - } return _native_java_library; } @@ -574,7 +552,7 @@ void* os::malloc(size_t size, MEMFLAGS memflags, const NativeCallStack& stack) { // exists and has crash protection. WatcherThread *wt = WatcherThread::watcher_thread(); if (wt != NULL && wt->has_crash_protection()) { - Thread* thread = ThreadLocalStorage::get_thread_slow(); + Thread* thread = Thread::current_or_null(); if (thread == wt) { assert(!wt->has_crash_protection(), "Can't malloc with crash protection from WatcherThread"); diff --git a/hotspot/src/share/vm/runtime/os.hpp b/hotspot/src/share/vm/runtime/os.hpp index a26cc9ab3f9..ef8b2954518 100644 --- a/hotspot/src/share/vm/runtime/os.hpp +++ b/hotspot/src/share/vm/runtime/os.hpp @@ -670,12 +670,6 @@ class os: AllStatic { static jlong current_file_offset(int fd); static jlong seek_to_file_offset(int fd, jlong offset); - // Thread Local Storage - static int allocate_thread_local_storage(); - static void thread_local_storage_at_put(int index, void* value); - static void* thread_local_storage_at(int index); - static void free_thread_local_storage(int index); - // Retrieve native stack frames. // Parameter: // stack: an array to storage stack pointers. diff --git a/hotspot/src/share/vm/runtime/sharedRuntime.hpp b/hotspot/src/share/vm/runtime/sharedRuntime.hpp index acf822f6eb1..cfb63f37dee 100644 --- a/hotspot/src/share/vm/runtime/sharedRuntime.hpp +++ b/hotspot/src/share/vm/runtime/sharedRuntime.hpp @@ -30,7 +30,6 @@ #include "interpreter/linkResolver.hpp" #include "memory/allocation.hpp" #include "memory/resourceArea.hpp" -#include "runtime/threadLocalStorage.hpp" #include "utilities/hashtable.hpp" #include "utilities/macros.hpp" diff --git a/hotspot/src/share/vm/runtime/thread.cpp b/hotspot/src/share/vm/runtime/thread.cpp index db0f5d9812c..6ee4c82dc73 100644 --- a/hotspot/src/share/vm/runtime/thread.cpp +++ b/hotspot/src/share/vm/runtime/thread.cpp @@ -78,7 +78,6 @@ #include "runtime/task.hpp" #include "runtime/thread.inline.hpp" #include "runtime/threadCritical.hpp" -#include "runtime/threadLocalStorage.hpp" #include "runtime/vframe.hpp" #include "runtime/vframeArray.hpp" #include "runtime/vframe_hp.hpp" @@ -142,6 +141,10 @@ #endif // ndef DTRACE_ENABLED +#ifndef USE_LIBRARY_BASED_TLS_ONLY +// Current thread is maintained as a thread-local variable +THREAD_LOCAL_DECL Thread* Thread::_thr_current = NULL; +#endif // Class hierarchy // - Thread @@ -281,22 +284,22 @@ Thread::Thread() { #endif // ASSERT } -// Non-inlined version to be used where thread.inline.hpp shouldn't be included. -Thread* Thread::current_noinline() { - return Thread::current(); +void Thread::initialize_thread_current() { +#ifndef USE_LIBRARY_BASED_TLS_ONLY + assert(_thr_current == NULL, "Thread::current already initialized"); + _thr_current = this; +#endif + assert(ThreadLocalStorage::thread() == NULL, "ThreadLocalStorage::thread already initialized"); + ThreadLocalStorage::set_thread(this); + assert(Thread::current() == ThreadLocalStorage::thread(), "TLS mismatch!"); } -void Thread::initialize_thread_local_storage() { - // Note: Make sure this method only calls - // non-blocking operations. Otherwise, it might not work - // with the thread-startup/safepoint interaction. - - // During Java thread startup, safepoint code should allow this - // method to complete because it may need to allocate memory to - // store information for the new thread. - - // initialize structure dependent on thread local storage - ThreadLocalStorage::set_thread(this); +void Thread::clear_thread_current() { + assert(Thread::current() == ThreadLocalStorage::thread(), "TLS mismatch!"); +#ifndef USE_LIBRARY_BASED_TLS_ONLY + _thr_current = NULL; +#endif + ThreadLocalStorage::set_thread(NULL); } void Thread::record_stack_base_and_size() { @@ -364,15 +367,12 @@ Thread::~Thread() { delete _SR_lock; - // clear thread local storage if the Thread is deleting itself + // clear Thread::current if thread is deleting itself. + // Needed to ensure JNI correctly detects non-attached threads. if (this == Thread::current()) { - ThreadLocalStorage::set_thread(NULL); - } else { - // In the case where we're not the current thread, invalidate all the - // caches in case some code tries to get the current thread or the - // thread that was destroyed, and gets stale information. - ThreadLocalStorage::invalidate_all(); + clear_thread_current(); } + CHECK_UNHANDLED_OOPS_ONLY(if (CheckUnhandledOops) delete unhandled_oops();) } @@ -1273,7 +1273,6 @@ void WatcherThread::run() { assert(this == watcher_thread(), "just checking"); this->record_stack_base_and_size(); - this->initialize_thread_local_storage(); this->set_native_thread_name(this->name()); this->set_active_handles(JNIHandleBlock::allocate_block()); while (true) { @@ -1326,9 +1325,6 @@ void WatcherThread::run() { _watcher_thread = NULL; Terminator_lock->notify(); } - - // Thread destructor usually does this.. - ThreadLocalStorage::set_thread(NULL); } void WatcherThread::start() { @@ -1663,9 +1659,6 @@ void JavaThread::run() { // Record real stack base and size. this->record_stack_base_and_size(); - // Initialize thread local storage; set before calling MutexLocker - this->initialize_thread_local_storage(); - this->create_stack_guard_pages(); this->cache_global_variables(); @@ -1997,8 +1990,7 @@ void JavaThread::cleanup_failed_attach_current_thread() { JavaThread* JavaThread::active() { - Thread* thread = ThreadLocalStorage::thread(); - assert(thread != NULL, "just checking"); + Thread* thread = Thread::current(); if (thread->is_Java_thread()) { return (JavaThread*) thread; } else { @@ -3407,7 +3399,7 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { jint adjust_after_os_result = Arguments::adjust_after_os(); if (adjust_after_os_result != JNI_OK) return adjust_after_os_result; - // initialize TLS + // Initialize library-based TLS ThreadLocalStorage::init(); // Initialize output stream logging @@ -3444,14 +3436,9 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) { // Attach the main thread to this os thread JavaThread* main_thread = new JavaThread(); main_thread->set_thread_state(_thread_in_vm); - // must do this before set_active_handles and initialize_thread_local_storage - // Note: on solaris initialize_thread_local_storage() will (indirectly) - // change the stack size recorded here to one based on the java thread - // stacksize. This adjusted size is what is used to figure the placement - // of the guard pages. + main_thread->initialize_thread_current(); + // must do this before set_active_handles main_thread->record_stack_base_and_size(); - main_thread->initialize_thread_local_storage(); - main_thread->set_active_handles(JNIHandleBlock::allocate_block()); if (!main_thread->set_as_starting_thread()) { diff --git a/hotspot/src/share/vm/runtime/thread.hpp b/hotspot/src/share/vm/runtime/thread.hpp index 27c81f00346..51ea77a3039 100644 --- a/hotspot/src/share/vm/runtime/thread.hpp +++ b/hotspot/src/share/vm/runtime/thread.hpp @@ -102,6 +102,12 @@ class WorkerThread; class Thread: public ThreadShadow { friend class VMStructs; private: + +#ifndef USE_LIBRARY_BASED_TLS_ONLY + // Current thread is maintained as a thread-local variable + static THREAD_LOCAL_DECL Thread* _thr_current; +#endif + // Exception handling // (Note: _pending_exception and friends are in ThreadShadow) //oop _pending_exception; // pending exception for current thread @@ -260,7 +266,6 @@ class Thread: public ThreadShadow { friend class No_Alloc_Verifier; friend class No_Safepoint_Verifier; friend class Pause_No_Safepoint_Verifier; - friend class ThreadLocalStorage; friend class GC_locker; ThreadLocalAllocBuffer _tlab; // Thread-local eden @@ -307,9 +312,12 @@ class Thread: public ThreadShadow { Thread(); virtual ~Thread(); - // initializtion - void initialize_thread_local_storage(); + // Manage Thread::current() + void initialize_thread_current(); + private: + void clear_thread_current(); // needed for detaching JNI threads + public: // thread entry point virtual void run(); @@ -337,10 +345,13 @@ class Thread: public ThreadShadow { virtual char* name() const { return (char*)"Unknown thread"; } - // Returns the current thread + // Returns the current thread (ASSERTS if NULL) static inline Thread* current(); - // ... without having to include thread.inline.hpp. - static Thread* current_noinline(); + // Returns the current thread, or NULL if not attached + static inline Thread* current_or_null(); + // Returns the current thread, or NULL if not attached, and is + // safe for use from signal-handlers + static inline Thread* current_or_null_safe(); // Common thread operations static void set_priority(Thread* thread, ThreadPriority priority); @@ -649,25 +660,22 @@ protected: }; // Inline implementation of Thread::current() -// Thread::current is "hot" it's called > 128K times in the 1st 500 msecs of -// startup. -// ThreadLocalStorage::thread is warm -- it's called > 16K times in the same -// period. This is inlined in thread_.inline.hpp. - inline Thread* Thread::current() { -#ifdef ASSERT - // This function is very high traffic. Define PARANOID to enable expensive - // asserts. -#ifdef PARANOID - // Signal handler should call ThreadLocalStorage::get_thread_slow() - Thread* t = ThreadLocalStorage::get_thread_slow(); - assert(t != NULL && !t->is_inside_signal_handler(), - "Don't use Thread::current() inside signal handler"); + Thread* current = current_or_null(); + assert(current != NULL, "Thread::current() called on detached thread"); + return current; +} + +inline Thread* Thread::current_or_null() { +#ifndef USE_LIBRARY_BASED_TLS_ONLY + return _thr_current; +#else + return ThreadLocalStorage::thread(); #endif -#endif - Thread* thread = ThreadLocalStorage::thread(); - assert(thread != NULL, "just checking"); - return thread; +} + +inline Thread* Thread::current_or_null_safe() { + return ThreadLocalStorage::thread(); } // Name support for threads. non-JavaThread subclasses with multiple @@ -1842,8 +1850,8 @@ class JavaThread: public Thread { // Inline implementation of JavaThread::current inline JavaThread* JavaThread::current() { - Thread* thread = ThreadLocalStorage::thread(); - assert(thread != NULL && thread->is_Java_thread(), "just checking"); + Thread* thread = Thread::current(); + assert(thread->is_Java_thread(), "just checking"); return (JavaThread*)thread; } diff --git a/hotspot/src/share/vm/runtime/thread.inline.hpp b/hotspot/src/share/vm/runtime/thread.inline.hpp index 0e75347d2a1..a6fb4a63979 100644 --- a/hotspot/src/share/vm/runtime/thread.inline.hpp +++ b/hotspot/src/share/vm/runtime/thread.inline.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -30,21 +30,6 @@ #include "runtime/atomic.inline.hpp" #include "runtime/os.inline.hpp" #include "runtime/thread.hpp" -#ifdef TARGET_OS_FAMILY_linux -# include "thread_linux.inline.hpp" -#endif -#ifdef TARGET_OS_FAMILY_solaris -# include "thread_solaris.inline.hpp" -#endif -#ifdef TARGET_OS_FAMILY_windows -# include "thread_windows.inline.hpp" -#endif -#ifdef TARGET_OS_FAMILY_aix -# include "thread_aix.inline.hpp" -#endif -#ifdef TARGET_OS_FAMILY_bsd -# include "thread_bsd.inline.hpp" -#endif #undef SHARE_VM_RUNTIME_THREAD_INLINE_HPP_SCOPE diff --git a/hotspot/src/share/vm/runtime/threadLocalStorage.cpp b/hotspot/src/share/vm/runtime/threadLocalStorage.cpp deleted file mode 100644 index 61d40889615..00000000000 --- a/hotspot/src/share/vm/runtime/threadLocalStorage.cpp +++ /dev/null @@ -1,63 +0,0 @@ -/* - * Copyright (c) 1997, 2015, 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 "precompiled.hpp" -#include "runtime/os.inline.hpp" -#include "runtime/thread.inline.hpp" -#include "runtime/threadLocalStorage.hpp" - -// Solaris no longer has this kind of ThreadLocalStorage implementation. -// This will be removed from all platforms in the near future. - -#ifndef SOLARIS - -// static member initialization -int ThreadLocalStorage::_thread_index = -1; - -Thread* ThreadLocalStorage::get_thread_slow() { - return (Thread*) os::thread_local_storage_at(ThreadLocalStorage::thread_index()); -} - -void ThreadLocalStorage::set_thread(Thread* thread) { - pd_set_thread(thread); - - // The following ensure that any optimization tricks we have tried - // did not backfire on us: - guarantee(get_thread() == thread, "must be the same thread, quickly"); - guarantee(get_thread_slow() == thread, "must be the same thread, slowly"); -} - -void ThreadLocalStorage::init() { - assert(!is_initialized(), - "More than one attempt to initialize threadLocalStorage"); - pd_init(); - set_thread_index(os::allocate_thread_local_storage()); - generate_code_for_get_thread(); -} - -bool ThreadLocalStorage::is_initialized() { - return (thread_index() != -1); -} - -#endif // SOLARIS diff --git a/hotspot/src/share/vm/runtime/threadLocalStorage.hpp b/hotspot/src/share/vm/runtime/threadLocalStorage.hpp index 128dd98067f..016e1fc1196 100644 --- a/hotspot/src/share/vm/runtime/threadLocalStorage.hpp +++ b/hotspot/src/share/vm/runtime/threadLocalStorage.hpp @@ -25,86 +25,26 @@ #ifndef SHARE_VM_RUNTIME_THREADLOCALSTORAGE_HPP #define SHARE_VM_RUNTIME_THREADLOCALSTORAGE_HPP -#include "gc/shared/gcUtil.hpp" -#include "runtime/os.hpp" #include "utilities/top.hpp" -// Interface for thread local storage +// forward-decl as we can't have an include cycle +class Thread; -// Fast variant of ThreadLocalStorage::get_thread_slow -extern "C" Thread* get_thread(); - -// Get raw thread id: e.g., %g7 on sparc, fs or gs on x86 -extern "C" uintptr_t _raw_thread_id(); +// Wrapper class for library-based (as opposed to compiler-based) +// thread-local storage (TLS). All platforms require this for +// signal-handler based TLS access (which while not strictly async-signal +// safe in theory, is and has-been for a long time, in practice). +// Platforms without compiler-based TLS (i.e. __thread storage-class modifier) +// will use this implementation for all TLS access - see thread.hpp/cpp class ThreadLocalStorage : AllStatic { // Exported API public: - static void set_thread(Thread* thread); - static Thread* get_thread_slow(); - static void invalidate_all() { pd_invalidate_all(); } + static Thread* thread(); // return current thread, if attached + static void set_thread(Thread* thread); // set current thread static void init(); - static bool is_initialized(); - - // Machine dependent stuff -#ifdef TARGET_OS_ARCH_linux_x86 -# include "threadLS_linux_x86.hpp" -#endif -#ifdef TARGET_OS_ARCH_linux_sparc -# include "threadLS_linux_sparc.hpp" -#endif -#ifdef TARGET_OS_ARCH_linux_zero -# include "threadLS_linux_zero.hpp" -#endif -#ifdef TARGET_OS_ARCH_solaris_x86 -# include "threadLS_solaris_x86.hpp" -#endif -#ifdef TARGET_OS_ARCH_solaris_sparc -# include "threadLS_solaris_sparc.hpp" -#endif -#ifdef TARGET_OS_ARCH_windows_x86 -# include "threadLS_windows_x86.hpp" -#endif -#ifdef TARGET_OS_ARCH_linux_arm -# include "threadLS_linux_arm.hpp" -#endif -#ifdef TARGET_OS_ARCH_linux_ppc -# include "threadLS_linux_ppc.hpp" -#endif -#ifdef TARGET_OS_ARCH_linux_aarch64 -# include "threadLS_linux_aarch64.hpp" -#endif -#ifdef TARGET_OS_ARCH_aix_ppc -# include "threadLS_aix_ppc.hpp" -#endif -#ifdef TARGET_OS_ARCH_bsd_x86 -# include "threadLS_bsd_x86.hpp" -#endif -#ifdef TARGET_OS_ARCH_bsd_zero -# include "threadLS_bsd_zero.hpp" -#endif - -#ifndef SOLARIS - public: - // Accessor - static inline int thread_index() { return _thread_index; } - static inline void set_thread_index(int index) { _thread_index = index; } - - private: - static int _thread_index; - - static void generate_code_for_get_thread(); - - // Processor dependent parts of set_thread and initialization - static void pd_set_thread(Thread* thread); - static void pd_init(); - -#endif // SOLARIS - - // Invalidate any thread cacheing or optimization schemes. - static void pd_invalidate_all(); - + static bool is_initialized(); // can't use TLS prior to initialization }; #endif // SHARE_VM_RUNTIME_THREADLOCALSTORAGE_HPP diff --git a/hotspot/src/share/vm/runtime/vmThread.cpp b/hotspot/src/share/vm/runtime/vmThread.cpp index bb2e3a058a2..e91faff6b28 100644 --- a/hotspot/src/share/vm/runtime/vmThread.cpp +++ b/hotspot/src/share/vm/runtime/vmThread.cpp @@ -240,7 +240,6 @@ void VMThread::destroy() { void VMThread::run() { assert(this == vm_thread(), "check"); - this->initialize_thread_local_storage(); this->initialize_named_thread(); this->record_stack_base_and_size(); // Notify_lock wait checks on active_handles() to rewait in @@ -308,9 +307,6 @@ void VMThread::run() { _terminate_lock->notify(); } - // Thread destructor usually does this. - ThreadLocalStorage::set_thread(NULL); - // Deletion must be done synchronously by the JNI DestroyJavaVM thread // so that the VMThread deletion completes before the main thread frees // up the CodeHeap. diff --git a/hotspot/src/share/vm/runtime/vm_operations.cpp b/hotspot/src/share/vm/runtime/vm_operations.cpp index 92e855b9e83..edc3876a78f 100644 --- a/hotspot/src/share/vm/runtime/vm_operations.cpp +++ b/hotspot/src/share/vm/runtime/vm_operations.cpp @@ -378,7 +378,7 @@ Thread * VM_Exit::_shutdown_thread = NULL; int VM_Exit::set_vm_exited() { CodeCacheExtensions::complete_step(CodeCacheExtensionsSteps::LastStep); - Thread * thr_cur = ThreadLocalStorage::get_thread_slow(); + Thread * thr_cur = Thread::current(); assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint already"); @@ -400,7 +400,7 @@ int VM_Exit::wait_for_threads_in_native_to_block() { // to wait for threads in _thread_in_native state to be quiescent. assert(SafepointSynchronize::is_at_safepoint(), "must be at safepoint already"); - Thread * thr_cur = ThreadLocalStorage::get_thread_slow(); + Thread * thr_cur = Thread::current(); Monitor timer(Mutex::leaf, "VM_Exit timer", true, Monitor::_safepoint_check_never); @@ -477,7 +477,7 @@ void VM_Exit::doit() { void VM_Exit::wait_if_vm_exited() { if (_vm_exited && - ThreadLocalStorage::get_thread_slow() != _shutdown_thread) { + Thread::current_or_null() != _shutdown_thread) { // _vm_exited is set at safepoint, and the Threads_lock is never released // we will block here until the process dies Threads_lock->lock_without_safepoint_check(); diff --git a/hotspot/src/share/vm/utilities/debug.cpp b/hotspot/src/share/vm/utilities/debug.cpp index 241aa6f139e..da09fcbf2ab 100644 --- a/hotspot/src/share/vm/utilities/debug.cpp +++ b/hotspot/src/share/vm/utilities/debug.cpp @@ -215,7 +215,7 @@ void report_vm_error(const char* file, int line, const char* error_msg, const ch if (Debugging || error_is_suppressed(file, line)) return; va_list detail_args; va_start(detail_args, detail_fmt); - VMError::report_and_die(ThreadLocalStorage::get_thread_slow(), file, line, error_msg, detail_fmt, detail_args); + VMError::report_and_die(Thread::current_or_null(), file, line, error_msg, detail_fmt, detail_args); va_end(detail_args); } @@ -224,7 +224,7 @@ void report_fatal(const char* file, int line, const char* detail_fmt, ...) if (Debugging || error_is_suppressed(file, line)) return; va_list detail_args; va_start(detail_args, detail_fmt); - VMError::report_and_die(ThreadLocalStorage::get_thread_slow(), file, line, "fatal error", detail_fmt, detail_args); + VMError::report_and_die(Thread::current_or_null(), file, line, "fatal error", detail_fmt, detail_args); va_end(detail_args); } @@ -233,7 +233,7 @@ void report_vm_out_of_memory(const char* file, int line, size_t size, if (Debugging) return; va_list detail_args; va_start(detail_args, detail_fmt); - VMError::report_and_die(ThreadLocalStorage::get_thread_slow(), file, line, size, vm_err_type, detail_fmt, detail_args); + VMError::report_and_die(Thread::current_or_null(), file, line, size, vm_err_type, detail_fmt, detail_args); va_end(detail_args); // The UseOSErrorReporting option in report_and_die() may allow a return @@ -536,7 +536,7 @@ extern "C" void findpc(intptr_t x); #endif // !PRODUCT extern "C" void ps() { // print stack - if (Thread::current() == NULL) return; + if (Thread::current_or_null() == NULL) return; Command c("ps"); @@ -615,7 +615,7 @@ extern "C" void safepoints() { #endif // !PRODUCT extern "C" void pss() { // print all stacks - if (Thread::current() == NULL) return; + if (Thread::current_or_null() == NULL) return; Command c("pss"); Threads::print(true, PRODUCT_ONLY(false) NOT_PRODUCT(true)); } @@ -772,7 +772,7 @@ void print_native_stack(outputStream* st, frame fr, Thread* t, char* buf, int bu extern "C" void pns(void* sp, void* fp, void* pc) { // print native stack Command c("pns"); static char buf[O_BUFLEN]; - Thread* t = ThreadLocalStorage::get_thread_slow(); + Thread* t = Thread::current_or_null(); // Call generic frame constructor (certain arguments may be ignored) frame fr(sp, fp, pc); print_native_stack(tty, fr, t, buf, sizeof(buf)); diff --git a/hotspot/src/share/vm/utilities/events.cpp b/hotspot/src/share/vm/utilities/events.cpp index 5e56da303b2..278afdc5f6e 100644 --- a/hotspot/src/share/vm/utilities/events.cpp +++ b/hotspot/src/share/vm/utilities/events.cpp @@ -29,7 +29,6 @@ #include "runtime/osThread.hpp" #include "runtime/thread.inline.hpp" #include "runtime/threadCritical.hpp" -#include "runtime/threadLocalStorage.hpp" #include "runtime/timer.hpp" #include "utilities/events.hpp" diff --git a/hotspot/src/share/vm/utilities/events.hpp b/hotspot/src/share/vm/utilities/events.hpp index 1f228efab3b..5b65fcc1a6f 100644 --- a/hotspot/src/share/vm/utilities/events.hpp +++ b/hotspot/src/share/vm/utilities/events.hpp @@ -248,8 +248,8 @@ inline void Events::log_deopt_message(Thread* thread, const char* format, ...) { template inline void EventLogBase::print_log_on(outputStream* out) { - if (ThreadLocalStorage::get_thread_slow() == NULL) { - // Not a regular Java thread so don't bother locking + if (Thread::current_or_null() == NULL) { + // Not yet attached? Don't try to use locking print_log_impl(out); } else { MutexLockerEx ml(&_mutex, Mutex::_no_safepoint_check_flag); diff --git a/hotspot/src/share/vm/utilities/globalDefinitions_gcc.hpp b/hotspot/src/share/vm/utilities/globalDefinitions_gcc.hpp index 836e25b4f89..afeafa384ee 100644 --- a/hotspot/src/share/vm/utilities/globalDefinitions_gcc.hpp +++ b/hotspot/src/share/vm/utilities/globalDefinitions_gcc.hpp @@ -326,4 +326,8 @@ inline int wcslen(const jchar* x) { return wcslen((const wchar_t*)x); } #define JLONG_FORMAT "%ld" #endif // _LP64 && __APPLE__ +#ifndef USE_LIBRARY_BASED_TLS_ONLY +#define THREAD_LOCAL_DECL __thread +#endif + #endif // SHARE_VM_UTILITIES_GLOBALDEFINITIONS_GCC_HPP diff --git a/hotspot/src/share/vm/utilities/globalDefinitions_sparcWorks.hpp b/hotspot/src/share/vm/utilities/globalDefinitions_sparcWorks.hpp index 48007e93008..7e1fa1917e9 100644 --- a/hotspot/src/share/vm/utilities/globalDefinitions_sparcWorks.hpp +++ b/hotspot/src/share/vm/utilities/globalDefinitions_sparcWorks.hpp @@ -273,4 +273,8 @@ inline int wcslen(const jchar* x) { return wcslen((const wchar_t*)x); } #define offset_of(klass,field) offsetof(klass,field) +#ifndef USE_LIBRARY_BASED_TLS_ONLY +#define THREAD_LOCAL_DECL __thread +#endif + #endif // SHARE_VM_UTILITIES_GLOBALDEFINITIONS_SPARCWORKS_HPP diff --git a/hotspot/src/share/vm/utilities/globalDefinitions_visCPP.hpp b/hotspot/src/share/vm/utilities/globalDefinitions_visCPP.hpp index f4a87437199..f594bafbc2f 100644 --- a/hotspot/src/share/vm/utilities/globalDefinitions_visCPP.hpp +++ b/hotspot/src/share/vm/utilities/globalDefinitions_visCPP.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -234,4 +234,8 @@ inline int vsnprintf(char* buf, size_t count, const char* fmt, va_list argptr) { #define offset_of(klass,field) offsetof(klass,field) +#ifndef USE_LIBRARY_BASED_TLS_ONLY +#define THREAD_LOCAL_DECL __declspec( thread ) +#endif + #endif // SHARE_VM_UTILITIES_GLOBALDEFINITIONS_VISCPP_HPP diff --git a/hotspot/src/share/vm/utilities/globalDefinitions_xlc.hpp b/hotspot/src/share/vm/utilities/globalDefinitions_xlc.hpp index 8a489d32576..93f2eaaac98 100644 --- a/hotspot/src/share/vm/utilities/globalDefinitions_xlc.hpp +++ b/hotspot/src/share/vm/utilities/globalDefinitions_xlc.hpp @@ -180,5 +180,8 @@ inline int wcslen(const jchar* x) { return wcslen((const wchar_t*)x); } #define SIZE_64G ((uint64_t) UCONST64( 0x1000000000)) #define SIZE_1T ((uint64_t) UCONST64(0x10000000000)) +#ifndef USE_LIBRARY_BASED_TLS_ONLY +#define THREAD_LOCAL_DECL __thread +#endif #endif // SHARE_VM_UTILITIES_GLOBALDEFINITIONS_XLC_HPP diff --git a/hotspot/src/share/vm/utilities/ostream.cpp b/hotspot/src/share/vm/utilities/ostream.cpp index 0e9584e46b0..0c09556e72a 100644 --- a/hotspot/src/share/vm/utilities/ostream.cpp +++ b/hotspot/src/share/vm/utilities/ostream.cpp @@ -738,7 +738,7 @@ void gcLogFileStream::rotate_log(bool force, outputStream* out) { } #ifdef ASSERT - Thread *thread = Thread::current(); + Thread *thread = Thread::current_or_null(); assert(thread == NULL || (thread->is_VM_thread() && SafepointSynchronize::is_at_safepoint()), "Must be VMThread at safepoint"); @@ -1058,8 +1058,8 @@ intx defaultStream::hold(intx writer_id) { // bootstrap problem tty_lock == NULL || - // can't grab a lock or call Thread::current() if TLS isn't initialized - ThreadLocalStorage::thread() == NULL || + // can't grab a lock if current Thread isn't set + Thread::current_or_null() == NULL || // developer hook !SerializeVMOutput || From 343aae742fb92e1f0802077e7a58eea3d1775ae9 Mon Sep 17 00:00:00 2001 From: Dmitry Dmitriev Date: Fri, 4 Dec 2015 13:39:59 +0300 Subject: [PATCH 7/7] 8144629: runtime/thread/Fibonacci.java test should ran in othervm mode Reviewed-by: ctornqvi --- hotspot/test/runtime/Thread/Fibonacci.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/hotspot/test/runtime/Thread/Fibonacci.java b/hotspot/test/runtime/Thread/Fibonacci.java index 18c58fc9de0..c8bbed1f980 100644 --- a/hotspot/test/runtime/Thread/Fibonacci.java +++ b/hotspot/test/runtime/Thread/Fibonacci.java @@ -29,7 +29,7 @@ * make this test inherently unstable on Windows with 32-bit VM data model. * @requires !(os.family == "windows" & sun.arch.data.model == "32") * @library /testlibrary - * @run main Fibonacci 15 + * @run main/othervm Fibonacci 15 */ import jdk.test.lib.Asserts;