From e13c212339d6af9f5698efdc090a1e9690c94bf7 Mon Sep 17 00:00:00 2001 From: Severin Gehwolf Date: Wed, 7 Jan 2015 16:43:04 +0100 Subject: [PATCH 01/77] 8067331: Zero: Atomic::xchg and Atomic::xchg_ptr need full memory barrier Reviewed-by: dholmes, coleenp --- .../os_cpu/bsd_zero/vm/atomic_bsd_zero.inline.hpp | 14 +++++++++++--- .../linux_zero/vm/atomic_linux_zero.inline.hpp | 14 +++++++++++--- 2 files changed, 22 insertions(+), 6 deletions(-) diff --git a/hotspot/src/os_cpu/bsd_zero/vm/atomic_bsd_zero.inline.hpp b/hotspot/src/os_cpu/bsd_zero/vm/atomic_bsd_zero.inline.hpp index 60969caa962..4e7e702ad66 100644 --- a/hotspot/src/os_cpu/bsd_zero/vm/atomic_bsd_zero.inline.hpp +++ b/hotspot/src/os_cpu/bsd_zero/vm/atomic_bsd_zero.inline.hpp @@ -1,6 +1,6 @@ /* * Copyright (c) 2003, 2014, Oracle and/or its affiliates. All rights reserved. - * Copyright 2007, 2008, 2011 Red Hat, Inc. + * Copyright 2007, 2008, 2011, 2015, 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 @@ -237,7 +237,13 @@ inline jint Atomic::xchg(jint exchange_value, volatile jint* dest) { // operation. Note that some platforms only support this with the // limitation that the only valid value to store is the immediate // constant 1. There is a test for this in JNI_CreateJavaVM(). - return __sync_lock_test_and_set (dest, exchange_value); + jint result = __sync_lock_test_and_set (dest, exchange_value); + // All atomic operations are expected to be full memory barriers + // (see atomic.hpp). However, __sync_lock_test_and_set is not + // a full memory barrier, but an acquire barrier. Hence, this added + // barrier. + __sync_synchronize(); + return result; #endif // M68K #endif // ARM } @@ -250,7 +256,9 @@ inline intptr_t Atomic::xchg_ptr(intptr_t exchange_value, #ifdef M68K return m68k_lock_test_and_set(dest, exchange_value); #else - return __sync_lock_test_and_set (dest, exchange_value); + intptr_t result = __sync_lock_test_and_set (dest, exchange_value); + __sync_synchronize(); + return result; #endif // M68K #endif // ARM } diff --git a/hotspot/src/os_cpu/linux_zero/vm/atomic_linux_zero.inline.hpp b/hotspot/src/os_cpu/linux_zero/vm/atomic_linux_zero.inline.hpp index d9df41034d4..266c950422b 100644 --- a/hotspot/src/os_cpu/linux_zero/vm/atomic_linux_zero.inline.hpp +++ b/hotspot/src/os_cpu/linux_zero/vm/atomic_linux_zero.inline.hpp @@ -1,6 +1,6 @@ /* * Copyright (c) 2003, 2014, Oracle and/or its affiliates. All rights reserved. - * Copyright 2007, 2008, 2011 Red Hat, Inc. + * Copyright 2007, 2008, 2011, 2015, 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 @@ -231,7 +231,13 @@ inline jint Atomic::xchg(jint exchange_value, volatile jint* dest) { // operation. Note that some platforms only support this with the // limitation that the only valid value to store is the immediate // constant 1. There is a test for this in JNI_CreateJavaVM(). - return __sync_lock_test_and_set (dest, exchange_value); + jint result = __sync_lock_test_and_set (dest, exchange_value); + // All atomic operations are expected to be full memory barriers + // (see atomic.hpp). However, __sync_lock_test_and_set is not + // a full memory barrier, but an acquire barrier. Hence, this added + // barrier. + __sync_synchronize(); + return result; #endif // M68K #endif // ARM } @@ -244,7 +250,9 @@ inline intptr_t Atomic::xchg_ptr(intptr_t exchange_value, #ifdef M68K return m68k_lock_test_and_set(dest, exchange_value); #else - return __sync_lock_test_and_set (dest, exchange_value); + intptr_t result = __sync_lock_test_and_set (dest, exchange_value); + __sync_synchronize(); + return result; #endif // M68K #endif // ARM } From f7241915b433ab3d83ed49d1156eff4978eb24a5 Mon Sep 17 00:00:00 2001 From: George Triantafillou Date: Tue, 6 Jan 2015 16:44:53 -0800 Subject: [PATCH 02/77] 8068540: [TESTBUG] Exclude failing nightly tests Reviewed-by: ctornqvi, coleenp --- hotspot/test/runtime/NMT/ChangeTrackingLevel.java | 1 + hotspot/test/runtime/NMT/PrintNMTStatistics.java | 1 + 2 files changed, 2 insertions(+) diff --git a/hotspot/test/runtime/NMT/ChangeTrackingLevel.java b/hotspot/test/runtime/NMT/ChangeTrackingLevel.java index 2794b05a833..d0df079cb66 100644 --- a/hotspot/test/runtime/NMT/ChangeTrackingLevel.java +++ b/hotspot/test/runtime/NMT/ChangeTrackingLevel.java @@ -27,6 +27,7 @@ * @summary Test that you can decrease NMT tracking level but not increase it. * @key nmt * @library /testlibrary /../../test/lib + * @ignore 8067167 * @build ChangeTrackingLevel * @run main ClassFileInstaller sun.hotspot.WhiteBox * sun.hotspot.WhiteBox$WhiteBoxPermission diff --git a/hotspot/test/runtime/NMT/PrintNMTStatistics.java b/hotspot/test/runtime/NMT/PrintNMTStatistics.java index 4b77303e026..ba46c20ec6f 100644 --- a/hotspot/test/runtime/NMT/PrintNMTStatistics.java +++ b/hotspot/test/runtime/NMT/PrintNMTStatistics.java @@ -27,6 +27,7 @@ * @bug 8005936 8058606 * @summary Verify PrintNMTStatistics on normal JVM exit for detail and summary tracking level * @library /testlibrary + * @ignore 8067167 */ import com.oracle.java.testlibrary.*; From 3b3dd985de385a8dc6aca09abf8dd8f3bdf537cb Mon Sep 17 00:00:00 2001 From: Roland Westrelin Date: Wed, 7 Jan 2015 16:40:02 +0100 Subject: [PATCH 03/77] 8063086: Math.pow yields different results upon repeated calls C2 treats x^2 as a special case and computes x * x while the interpreter and c1 don't have special case code for X^2. Reviewed-by: kvn --- hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp | 19 ++++- .../test/compiler/floatingpoint/TestPow2.java | 73 +++++++++++++++++++ 2 files changed, 91 insertions(+), 1 deletion(-) create mode 100644 hotspot/test/compiler/floatingpoint/TestPow2.java diff --git a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp index a6a5b43c07b..24587840f98 100644 --- a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp +++ b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp @@ -3184,7 +3184,24 @@ void MacroAssembler::pow_or_exp(bool is_exp, int num_fpu_regs_in_use) { jmp(done); } else { // Stack: X Y - Label x_negative, y_odd; + Label x_negative, y_not_2; + + static double two = 2.0; + ExternalAddress two_addr((address)&two); + + // constant maybe too far on 64 bit + lea(tmp2, two_addr); + fld_d(Address(tmp2, 0)); // Stack: 2 X Y + fcmp(tmp, 2, true, false); // Stack: X Y + jcc(Assembler::parity, y_not_2); + jcc(Assembler::notEqual, y_not_2); + + fxch(); fpop(); // Stack: X + fmul(0); // Stack: X*X + + jmp(done); + + bind(y_not_2); fldz(); // Stack: 0 X Y fcmp(tmp, 1, true, false); // Stack: X Y diff --git a/hotspot/test/compiler/floatingpoint/TestPow2.java b/hotspot/test/compiler/floatingpoint/TestPow2.java new file mode 100644 index 00000000000..699904b6e4f --- /dev/null +++ b/hotspot/test/compiler/floatingpoint/TestPow2.java @@ -0,0 +1,73 @@ +/* + * Copyright (c) 2014, 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 + * @bug 8063086 + * @summary X^2 special case for C2 yields different result than interpreter + * @library /testlibrary /../../test/lib /compiler/whitebox + * @build TestPow2 + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI + * -XX:-BackgroundCompilation -XX:-UseOnStackReplacement TestPow2 + * + */ + +import java.lang.reflect.*; +import sun.hotspot.WhiteBox; + +public class TestPow2 { + + private static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox(); + + private static final double base = 5350.456329377186; + private static final double exp = 2.0; + + static double m() { + return Math.pow(base, exp); + } + + static public void main(String[] args) throws NoSuchMethodException { + Method test_method = TestPow2.class.getDeclaredMethod("m"); + + double interpreter_result = m(); + + // Compile with C1 if possible + WHITE_BOX.enqueueMethodForCompilation(test_method, CompilerWhiteBoxTest.COMP_LEVEL_SIMPLE); + + double c1_result = m(); + + WHITE_BOX.deoptimizeMethod(test_method); + + // Compile it with C2 if possible + WHITE_BOX.enqueueMethodForCompilation(test_method, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION); + + double c2_result = m(); + + if (interpreter_result != c1_result || interpreter_result != c2_result || + c1_result != c2_result) { + System.out.println("interpreter = " + interpreter_result + " c1 = " + c1_result + " c2 = " + c2_result); + throw new RuntimeException("Test Failed"); + } + } +} From d2891a37e78457f367cfb0e4bdf315a9233f8e85 Mon Sep 17 00:00:00 2001 From: Roland Westrelin Date: Thu, 8 Jan 2015 11:00:38 +0100 Subject: [PATCH 04/77] 8027626: assert(Opcode() != Op_If || outcnt() == 2) failed: bad if #1 IGVN encounters IfNode with single projection when optimizing dying subgraph Reviewed-by: kvn --- hotspot/src/share/vm/opto/cfgnode.hpp | 26 ++++++++++++++++++++------ hotspot/src/share/vm/opto/ifnode.cpp | 27 +++++++++++++-------------- hotspot/src/share/vm/opto/node.cpp | 9 ++++++--- 3 files changed, 39 insertions(+), 23 deletions(-) diff --git a/hotspot/src/share/vm/opto/cfgnode.hpp b/hotspot/src/share/vm/opto/cfgnode.hpp index e795483e3cb..fc3235afecd 100644 --- a/hotspot/src/share/vm/opto/cfgnode.hpp +++ b/hotspot/src/share/vm/opto/cfgnode.hpp @@ -361,22 +361,36 @@ public: #endif }; -class IfTrueNode : public CProjNode { +class IfProjNode : public CProjNode { public: - IfTrueNode( IfNode *ifnode ) : CProjNode(ifnode,1) { + IfProjNode(IfNode *ifnode, uint idx) : CProjNode(ifnode,idx) {} + virtual Node *Identity(PhaseTransform *phase); + +protected: + // Type of If input when this branch is always taken + virtual bool always_taken(const TypeTuple* t) const = 0; +}; + +class IfTrueNode : public IfProjNode { +public: + IfTrueNode( IfNode *ifnode ) : IfProjNode(ifnode,1) { init_class_id(Class_IfTrue); } virtual int Opcode() const; - virtual Node *Identity( PhaseTransform *phase ); + +protected: + virtual bool always_taken(const TypeTuple* t) const { return t == TypeTuple::IFTRUE; } }; -class IfFalseNode : public CProjNode { +class IfFalseNode : public IfProjNode { public: - IfFalseNode( IfNode *ifnode ) : CProjNode(ifnode,0) { + IfFalseNode( IfNode *ifnode ) : IfProjNode(ifnode,0) { init_class_id(Class_IfFalse); } virtual int Opcode() const; - virtual Node *Identity( PhaseTransform *phase ); + +protected: + virtual bool always_taken(const TypeTuple* t) const { return t == TypeTuple::IFFALSE; } }; diff --git a/hotspot/src/share/vm/opto/ifnode.cpp b/hotspot/src/share/vm/opto/ifnode.cpp index 7b5f0d7b4be..493d4a021b8 100644 --- a/hotspot/src/share/vm/opto/ifnode.cpp +++ b/hotspot/src/share/vm/opto/ifnode.cpp @@ -1122,12 +1122,21 @@ void IfNode::dominated_by( Node *prev_dom, PhaseIterGVN *igvn ) { //------------------------------Identity--------------------------------------- // If the test is constant & we match, then we are the input Control -Node *IfTrueNode::Identity( PhaseTransform *phase ) { +Node *IfProjNode::Identity(PhaseTransform *phase) { // Can only optimize if cannot go the other way const TypeTuple *t = phase->type(in(0))->is_tuple(); - return ( t == TypeTuple::IFNEITHER || t == TypeTuple::IFTRUE ) - ? in(0)->in(0) // IfNode control - : this; // no progress + if (t == TypeTuple::IFNEITHER || + // kill dead branch first otherwise the IfNode's control will + // have 2 control uses (the IfNode that doesn't go away because + // it still has uses and this branch of the + // If). Node::has_special_unique_user() will cause this node to + // be reprocessed once the dead branch is killed. + (always_taken(t) && in(0)->outcnt() == 1)) { + // IfNode control + return in(0)->in(0); + } + // no progress + return this; } //------------------------------dump_spec-------------------------------------- @@ -1195,13 +1204,3 @@ static IfNode* idealize_test(PhaseGVN* phase, IfNode* iff) { // Progress return iff; } - -//------------------------------Identity--------------------------------------- -// If the test is constant & we match, then we are the input Control -Node *IfFalseNode::Identity( PhaseTransform *phase ) { - // Can only optimize if cannot go the other way - const TypeTuple *t = phase->type(in(0))->is_tuple(); - return ( t == TypeTuple::IFNEITHER || t == TypeTuple::IFFALSE ) - ? in(0)->in(0) // IfNode control - : this; // no progress -} diff --git a/hotspot/src/share/vm/opto/node.cpp b/hotspot/src/share/vm/opto/node.cpp index 29cd82e4acc..8ed950a021e 100644 --- a/hotspot/src/share/vm/opto/node.cpp +++ b/hotspot/src/share/vm/opto/node.cpp @@ -1080,18 +1080,21 @@ bool Node::has_special_unique_user() const { assert(outcnt() == 1, "match only for unique out"); Node* n = unique_out(); int op = Opcode(); - if( this->is_Store() ) { + if (this->is_Store()) { // Condition for back-to-back stores folding. return n->Opcode() == op && n->in(MemNode::Memory) == this; } else if (this->is_Load()) { // Condition for removing an unused LoadNode from the MemBarAcquire precedence input return n->Opcode() == Op_MemBarAcquire; - } else if( op == Op_AddL ) { + } else if (op == Op_AddL) { // Condition for convL2I(addL(x,y)) ==> addI(convL2I(x),convL2I(y)) return n->Opcode() == Op_ConvL2I && n->in(1) == this; - } else if( op == Op_SubI || op == Op_SubL ) { + } else if (op == Op_SubI || op == Op_SubL) { // Condition for subI(x,subI(y,z)) ==> subI(addI(x,z),y) return n->Opcode() == op && n->in(2) == this; + } else if (is_If() && (n->is_IfFalse() || n->is_IfTrue())) { + // See IfProjNode::Identity() + return true; } return false; }; From 8d30377d508939a2e5bd3a87b0ce51f974bee5d0 Mon Sep 17 00:00:00 2001 From: Claes Redestad Date: Thu, 8 Jan 2015 14:13:03 +0100 Subject: [PATCH 05/77] 8068037: Remove dead code in G1CollectedHeap Reviewed-by: stefank, tschatzl --- hotspot/src/share/vm/gc_implementation/g1/concurrentMark.cpp | 1 - hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.cpp | 2 -- hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.hpp | 1 - hotspot/src/share/vm/gc_implementation/g1/g1MarkSweep.hpp | 3 --- hotspot/src/share/vm/gc_implementation/g1/g1RemSet.hpp | 1 - 5 files changed, 8 deletions(-) diff --git a/hotspot/src/share/vm/gc_implementation/g1/concurrentMark.cpp b/hotspot/src/share/vm/gc_implementation/g1/concurrentMark.cpp index 6bb8b5de63c..0119858fe5b 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/concurrentMark.cpp +++ b/hotspot/src/share/vm/gc_implementation/g1/concurrentMark.cpp @@ -1910,7 +1910,6 @@ public: } void work(uint worker_id) { - double start = os::elapsedTime(); FreeRegionList local_cleanup_list("Local Cleanup List"); HRRSCleanupTask hrrs_cleanup_task; G1NoteEndOfConcMarkClosure g1_note_end(_g1h, &local_cleanup_list, diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.cpp b/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.cpp index 335ba2265d5..0d81bbfec83 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.cpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.cpp @@ -1229,7 +1229,6 @@ bool G1CollectedHeap::do_collection(bool explicit_gc, TraceCollectorStats tcs(g1mm()->full_collection_counters()); TraceMemoryManagerStats tms(true /* fullGC */, gc_cause()); - double start = os::elapsedTime(); g1_policy()->record_full_collection_start(); // Note: When we have a more flexible GC logging framework that @@ -1436,7 +1435,6 @@ bool G1CollectedHeap::do_collection(bool explicit_gc, _allocator->init_mutator_alloc_region(); - double end = os::elapsedTime(); g1_policy()->record_full_collection_end(); if (G1Log::fine()) { diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.hpp b/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.hpp index ab79bcd05ad..3286453095c 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.hpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1CollectedHeap.hpp @@ -56,7 +56,6 @@ class HRRSCleanupTask; class GenerationSpec; class OopsInHeapRegionClosure; class G1KlassScanClosure; -class G1ScanHeapEvacClosure; class ObjectClosure; class SpaceClosure; class CompactibleSpaceClosure; diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1MarkSweep.hpp b/hotspot/src/share/vm/gc_implementation/g1/g1MarkSweep.hpp index cdde980d306..e8d8c307676 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1MarkSweep.hpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1MarkSweep.hpp @@ -46,9 +46,6 @@ class ReferenceProcessor; class G1PrepareCompactClosure; class G1MarkSweep : AllStatic { - friend class VM_G1MarkSweep; - friend class Scavenge; - public: static void invoke_at_safepoint(ReferenceProcessor* rp, diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1RemSet.hpp b/hotspot/src/share/vm/gc_implementation/g1/g1RemSet.hpp index 468d109803b..fb9d348789e 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1RemSet.hpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1RemSet.hpp @@ -31,7 +31,6 @@ // collection set. class G1CollectedHeap; -class CardTableModRefBarrierSet; class ConcurrentG1Refine; class G1ParPushHeapRSClosure; From 6173a83d5defac07c9d481066985b73177f9f698 Mon Sep 17 00:00:00 2001 From: Christian Tornqvist Date: Thu, 8 Jan 2015 11:40:36 -0800 Subject: [PATCH 06/77] 8058897: Unsafe.reallocateMemory() ignores -XX:MallocMaxTestWords setting Reviewed-by: dcubed, dholmes --- hotspot/src/share/vm/runtime/os.cpp | 38 ++++++------ hotspot/test/runtime/Unsafe/Reallocate.java | 67 +++++++++++++++++++++ 2 files changed, 87 insertions(+), 18 deletions(-) create mode 100644 hotspot/test/runtime/Unsafe/Reallocate.java diff --git a/hotspot/src/share/vm/runtime/os.cpp b/hotspot/src/share/vm/runtime/os.cpp index 665688b6808..31cffe27467 100644 --- a/hotspot/src/share/vm/runtime/os.cpp +++ b/hotspot/src/share/vm/runtime/os.cpp @@ -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 @@ -547,20 +547,16 @@ static void verify_memory(void* ptr) { // This function supports testing of the malloc out of memory // condition without really running the system out of memory. // -static u_char* testMalloc(size_t alloc_size) { - assert(MallocMaxTestWords > 0, "sanity check"); +static bool has_reached_max_malloc_test_peak(size_t alloc_size) { + if (MallocMaxTestWords > 0) { + jint words = (jint)(alloc_size / BytesPerWord); - if ((cur_malloc_words + (alloc_size / BytesPerWord)) > MallocMaxTestWords) { - return NULL; + if ((cur_malloc_words + words) > MallocMaxTestWords) { + return true; + } + Atomic::add(words, (volatile jint *)&cur_malloc_words); } - - u_char* ptr = (u_char*)::malloc(alloc_size); - - if (ptr != NULL) { - Atomic::add(((jint) (alloc_size / BytesPerWord)), - (volatile jint *) &cur_malloc_words); - } - return ptr; + return false; } void* os::malloc(size_t size, MEMFLAGS flags) { @@ -608,13 +604,14 @@ void* os::malloc(size_t size, MEMFLAGS memflags, const NativeCallStack& stack) { NOT_PRODUCT(if (MallocVerifyInterval > 0) check_heap()); - u_char* ptr; - if (MallocMaxTestWords > 0) { - ptr = testMalloc(alloc_size); - } else { - ptr = (u_char*)::malloc(alloc_size); + // For the test flag -XX:MallocMaxTestWords + if (has_reached_max_malloc_test_peak(size)) { + return NULL; } + u_char* ptr; + ptr = (u_char*)::malloc(alloc_size); + #ifdef ASSERT if (ptr == NULL) { return NULL; @@ -642,6 +639,11 @@ void* os::realloc(void *memblock, size_t size, MEMFLAGS flags) { void* os::realloc(void *memblock, size_t size, MEMFLAGS memflags, const NativeCallStack& stack) { + // For the test flag -XX:MallocMaxTestWords + if (has_reached_max_malloc_test_peak(size)) { + return NULL; + } + #ifndef ASSERT NOT_PRODUCT(inc_stat_counter(&num_mallocs, 1)); NOT_PRODUCT(inc_stat_counter(&alloc_bytes, size)); diff --git a/hotspot/test/runtime/Unsafe/Reallocate.java b/hotspot/test/runtime/Unsafe/Reallocate.java new file mode 100644 index 00000000000..e5abe9aa9a6 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/Reallocate.java @@ -0,0 +1,67 @@ +/* + * 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 + * @bug 8058897 + * @library /testlibrary + * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:MallocMaxTestWords=20m Reallocate + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class Reallocate { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + + long address = unsafe.allocateMemory(1); + assertNotEquals(address, 0L); + + // Make sure we reallocate correctly + unsafe.putByte(address, Byte.MAX_VALUE); + address = unsafe.reallocateMemory(address, 2); + assertNotEquals(address, 0L); + assertEquals(unsafe.getByte(address), Byte.MAX_VALUE); + + // Reallocating with a 0 size should return a null pointer + address = unsafe.reallocateMemory(address, 0); + assertEquals(address, 0L); + + // Reallocating with a null pointer should result in a normal allocation + address = unsafe.reallocateMemory(0L, 1); + assertNotEquals(address, 0L); + unsafe.putByte(address, Byte.MAX_VALUE); + assertEquals(unsafe.getByte(address), Byte.MAX_VALUE); + + // Make sure we can throw an OOME when we fail to reallocate due to OOM + try { + unsafe.reallocateMemory(address, 20 * 1024 * 1024 * 8); + } catch (OutOfMemoryError e) { + // Expected + return; + } + throw new RuntimeException("Did not get expected OOM"); + } +} From c5853aabdd84800e6b2e8b5e5a694762b86fc109 Mon Sep 17 00:00:00 2001 From: Christian Tornqvist Date: Thu, 8 Jan 2015 11:42:05 -0800 Subject: [PATCH 07/77] 8060219: [TESTBUG] runtime/7194254/Test7194254.java fails to find jstack with modular image build Reviewed-by: gtriantafill, lfoltan, hseigel --- hotspot/test/TEST.groups | 4 +- .../ThreadPriorities.java} | 82 +++++++------------ 2 files changed, 31 insertions(+), 55 deletions(-) rename hotspot/test/runtime/{7194254/Test7194254.java => Thread/ThreadPriorities.java} (50%) diff --git a/hotspot/test/TEST.groups b/hotspot/test/TEST.groups index 261e0cf073c..cbd13fce4c6 100644 --- a/hotspot/test/TEST.groups +++ b/hotspot/test/TEST.groups @@ -1,5 +1,5 @@ # -# Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2013, 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 @@ -69,7 +69,6 @@ needs_jdk = \ gc/metaspace/TestPerfCountersAndMemoryPools.java \ runtime/6819213/TestBootNativeLibraryPath.java \ runtime/7158988/FieldMonitor.java \ - runtime/7194254/Test7194254.java \ runtime/Metaspace/FragmentMetaspace.java \ runtime/NMT/BaselineWithParameter.java \ runtime/NMT/JcmdBaselineDetail.java \ @@ -94,6 +93,7 @@ needs_jdk = \ runtime/NMT/VirtualAllocTestType.java \ runtime/RedefineObject/TestRedefineObject.java \ runtime/Thread/TestThreadDumpMonitorContention.java \ + runtime/Thread/ThreadPriorities.java \ runtime/XCheckJniJsig/XCheckJSig.java \ serviceability/attach/AttachWithStalePidFile.java \ serviceability/sa/jmap-hprof/JMapHProfLargeHeapTest.java \ diff --git a/hotspot/test/runtime/7194254/Test7194254.java b/hotspot/test/runtime/Thread/ThreadPriorities.java similarity index 50% rename from hotspot/test/runtime/7194254/Test7194254.java rename to hotspot/test/runtime/Thread/ThreadPriorities.java index 08f55e55b22..18a75c1d35e 100644 --- a/hotspot/test/runtime/7194254/Test7194254.java +++ b/hotspot/test/runtime/Thread/ThreadPriorities.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 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 @@ -27,23 +27,21 @@ * @summary Creates several threads with different java priorities and checks * whether jstack reports correct priorities for them. * - * @ignore 8060219 - * @run main Test7194254 + * @library /testlibrary + * @run main ThreadPriorities */ -import java.io.BufferedReader; -import java.io.InputStreamReader; -import java.lang.management.ManagementFactory; -import java.lang.management.RuntimeMXBean; import java.util.ArrayList; -import java.util.List; import java.util.concurrent.CyclicBarrier; import java.util.regex.Matcher; import java.util.regex.Pattern; -public class Test7194254 { +import com.oracle.java.testlibrary.*; +import static com.oracle.java.testlibrary.Asserts.*; - public static void main(String[] args) throws Exception { +public class ThreadPriorities { + + public static void main(String[] args) throws Throwable { final int NUMBER_OF_JAVA_PRIORITIES = Thread.MAX_PRIORITY - Thread.MIN_PRIORITY + 1; final CyclicBarrier barrier = @@ -68,53 +66,31 @@ public class Test7194254 { barrier.await(); // 1st int matches = 0; - List failed = new ArrayList<>(); - try { - String pid = getPid(); - String jstack = System.getProperty("java.home") + "/../bin/jstack"; - Process process = new ProcessBuilder(jstack, pid) - .redirectErrorStream(true).start(); - Pattern pattern = Pattern.compile( - "\\\"Priority=(\\d+)\\\".* prio=(\\d+).*"); - try (BufferedReader reader = new BufferedReader( - new InputStreamReader(process.getInputStream()))) { - String line; - while((line = reader.readLine()) != null) { - Matcher matcher = pattern.matcher(line); - if (matcher.matches()) { - matches += 1; - String expected = matcher.group(1); - String actual = matcher.group(2); - if (!expected.equals(actual)) { - failed.add(line); - } - } + ArrayList failed = new ArrayList<>(); + ProcessBuilder pb = new ProcessBuilder( + JDKToolFinder.getJDKTool("jstack"), + String.valueOf(ProcessTools.getProcessId())); + + String[] output = new OutputAnalyzer(pb.start()).getOutput().split("\\n+"); + + Pattern pattern = Pattern.compile( + "\\\"Priority=(\\d+)\\\".* prio=(\\d+).*"); + for (String line : output) { + Matcher matcher = pattern.matcher(line); + if (matcher.matches()) { + matches += 1; + String expected = matcher.group(1); + String actual = matcher.group(2); + if (!expected.equals(actual)) { + failed.add(line); } } - barrier.await(); // 2nd - } finally { - barrier.reset(); } + barrier.await(); // 2nd + barrier.reset(); - if (matches != NUMBER_OF_JAVA_PRIORITIES) { - throw new AssertionError("matches: expected " + - NUMBER_OF_JAVA_PRIORITIES + ", but was " + matches); - } - if (!failed.isEmpty()) { - throw new AssertionError(failed.size() + ":" + failed); - } - System.out.println("Test passes."); + assertEquals(matches, NUMBER_OF_JAVA_PRIORITIES); + assertTrue(failed.isEmpty(), failed.size() + ":" + failed); } - - static String getPid() { - RuntimeMXBean runtimebean = ManagementFactory.getRuntimeMXBean(); - String vmname = runtimebean.getName(); - int i = vmname.indexOf('@'); - if (i != -1) { - vmname = vmname.substring(0, i); - } - return vmname; - } - } From 24d3bb517b29f93c7ea7784047f485bfe83108de Mon Sep 17 00:00:00 2001 From: Bengt Rutisson Date: Fri, 9 Jan 2015 08:38:23 +0100 Subject: [PATCH 08/77] 8068653: TestSmalllHeap.java fails when the page size is 64k Reviewed-by: tschatzl --- hotspot/test/gc/TestSmallHeap.java | 50 ++++++++++++++++++------------ 1 file changed, 31 insertions(+), 19 deletions(-) diff --git a/hotspot/test/gc/TestSmallHeap.java b/hotspot/test/gc/TestSmallHeap.java index 321e33bf3b6..170b5397740 100644 --- a/hotspot/test/gc/TestSmallHeap.java +++ b/hotspot/test/gc/TestSmallHeap.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 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,34 +26,46 @@ * @bug 8067438 * @requires vm.gc=="null" * @summary Verify that starting the VM with a small heap works - * @library /testlibrary - * @run main/othervm -Xmx4m -XX:+UseParallelGC TestSmallHeap - * @run main/othervm -Xmx4m -XX:+UseSerialGC TestSmallHeap - * @run main/othervm -Xmx4m -XX:+UseG1GC TestSmallHeap - * @run main/othervm -Xmx4m -XX:+UseConcMarkSweepGC -XX:CMSMarkStackSizeMax=1032 TestSmallHeap + * @library /testlibrary /../../test/lib + * @build TestSmallHeap + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx2m -XX:+UseParallelGC TestSmallHeap + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx2m -XX:+UseSerialGC TestSmallHeap + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx2m -XX:+UseG1GC TestSmallHeap + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -Xmx2m -XX:+UseConcMarkSweepGC TestSmallHeap + * + * Note: It would be nice to verify the minimal supported heap size (2m) here, + * but we align the heap size based on the card table size. And the card table + * size is aligned based on the minimal pages size provided by the os. This + * means that on most platforms, where the minimal page size is 4k, we get a + * minimal heap size of 2m but on Solaris/Sparc we have a page size of 8k and + * get a minimal heap size of 4m. And on platforms where the page size is 64k + * we get a minimal heap size of 32m. We never use large pages for the card table. * - * Note: It would be nice to verify the minimal supported heap size here, - * but that turns out to be quite tricky since we align the heap size based - * on the card table size. And the card table size is aligned based on the - * minimal pages size provided by the os. This means that on most platforms, - * where the minimal page size is 4k, we get a minimal heap size of 2m but - * on Solaris/Sparc we have a page size of 8k and get a minimal heap size - * of 8m. * There is also no check in the VM for verifying that the maximum heap size * is larger than the supported minimal heap size. This means that specifying - * -Xmx1m on the command line is fine but will give a heap of 2m (or 4m). - * To work around these rather strange behaviors this test uses 4m for all - * platforms. + * -Xmx1m on the command line is fine but will give a heap of 2m (or 4m or 32m). + * + * To work around these rather strange behaviors this test uses -Xmx2m but then + * calculates what the expected heap size should be. The calculation is a + * simplified version of the code in the VM. We assume that the card table will + * use one page. Each byte in the card table corresponds to 512 bytes on the heap. + * So, the expected heap size is page_size * 512. */ -import sun.management.ManagementFactoryHelper; +import com.oracle.java.testlibrary.*; import static com.oracle.java.testlibrary.Asserts.*; +import sun.hotspot.WhiteBox; +import sun.management.ManagementFactoryHelper; public class TestSmallHeap { public static void main(String[] args) { + WhiteBox wb = WhiteBox.getWhiteBox(); + int pageSize = wb.getVMPageSize(); + int heapBytesPerCard = 512; + long expectedMaxHeap = pageSize * heapBytesPerCard; String maxHeap = ManagementFactoryHelper.getDiagnosticMXBean().getVMOption("MaxHeapSize").getValue(); - String expectedMaxHeap = "4194304"; - assertEQ(maxHeap, expectedMaxHeap); + assertEQ(Long.parseLong(maxHeap), expectedMaxHeap); } } From 8467a04febe657f861a6bcdde48ee869d8abdb89 Mon Sep 17 00:00:00 2001 From: Zoltan Majo Date: Fri, 9 Jan 2015 09:26:58 +0100 Subject: [PATCH 09/77] 8068505: interpreter profiling incorrect on PPC64 Reviewed-by: simonis, goetz --- hotspot/src/cpu/ppc/vm/templateInterpreter_ppc.cpp | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/hotspot/src/cpu/ppc/vm/templateInterpreter_ppc.cpp b/hotspot/src/cpu/ppc/vm/templateInterpreter_ppc.cpp index 2a58b36d420..432a96d8268 100644 --- a/hotspot/src/cpu/ppc/vm/templateInterpreter_ppc.cpp +++ b/hotspot/src/cpu/ppc/vm/templateInterpreter_ppc.cpp @@ -264,11 +264,11 @@ void TemplateInterpreterGenerator::generate_counter_incr(Label* overflow, Label* __ cmpdi(CCR0, Rmdo, 0); __ beq(CCR0, no_mdo); - // Increment backedge counter in the MDO. - const int mdo_bc_offs = in_bytes(MethodData::backedge_counter_offset()) + in_bytes(InvocationCounter::counter_offset()); - __ lwz(Rscratch2, mdo_bc_offs, Rmdo); + // Increment invocation counter in the MDO. + const int mdo_ic_offs = in_bytes(MethodData::invocation_counter_offset()) + in_bytes(InvocationCounter::counter_offset()); + __ lwz(Rscratch2, mdo_ic_offs, Rmdo); __ addi(Rscratch2, Rscratch2, increment); - __ stw(Rscratch2, mdo_bc_offs, Rmdo); + __ stw(Rscratch2, mdo_ic_offs, Rmdo); __ load_const_optimized(Rscratch1, mask, R0); __ and_(Rscratch1, Rscratch2, Rscratch1); __ bne(CCR0, done); @@ -276,12 +276,12 @@ void TemplateInterpreterGenerator::generate_counter_incr(Label* overflow, Label* } // Increment counter in MethodCounters*. - const int mo_bc_offs = in_bytes(MethodCounters::backedge_counter_offset()) + in_bytes(InvocationCounter::counter_offset()); + const int mo_ic_offs = in_bytes(MethodCounters::invocation_counter_offset()) + in_bytes(InvocationCounter::counter_offset()); __ bind(no_mdo); __ get_method_counters(R19_method, R3_counters, done); - __ lwz(Rscratch2, mo_bc_offs, R3_counters); + __ lwz(Rscratch2, mo_ic_offs, R3_counters); __ addi(Rscratch2, Rscratch2, increment); - __ stw(Rscratch2, mo_bc_offs, R3_counters); + __ stw(Rscratch2, mo_ic_offs, R3_counters); __ load_const_optimized(Rscratch1, mask, R0); __ and_(Rscratch1, Rscratch2, Rscratch1); __ beq(CCR0, *overflow); From dad0da797d0f47403af05cb2d26ab5cf085a200c Mon Sep 17 00:00:00 2001 From: Albert Noll Date: Fri, 9 Jan 2015 09:52:00 +0100 Subject: [PATCH 10/77] 8068661: Exclude compiler/whitebox/ForceNMethodSweepTest.java from nightly runs The test is unstable and is therefore removed from nightly testing Reviewed-by: kvn, drchase --- hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java | 1 + 1 file changed, 1 insertion(+) diff --git a/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java b/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java index 5038e7cfa8f..c08067ec4be 100644 --- a/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java +++ b/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java @@ -34,6 +34,7 @@ import com.oracle.java.testlibrary.InfiniteLoop; /* * @test * @bug 8059624 8064669 + * @ignore 8066998 * @library /testlibrary /../../test/lib * @build ForceNMethodSweepTest * @run main ClassFileInstaller sun.hotspot.WhiteBox From 1b1ac860df2cf8ea464bc5ff58c70e5bbcfd7fd0 Mon Sep 17 00:00:00 2001 From: Goetz Lindenmaier Date: Fri, 9 Jan 2015 05:45:13 -0800 Subject: [PATCH 11/77] 8068013: [TESTBUG] Aix support in hotspot jtreg tests Reviewed-by: ctornqvi, fzhinkin, farvidsson --- hotspot/test/runtime/6888954/vmerrors.sh | 5 ++-- .../serviceability/dcmd/DynLibDcmdTest.java | 14 ++++++----- hotspot/test/test_env.sh | 23 +++++++++++-------- .../com/oracle/java/testlibrary/Platform.java | 22 +++++++++++------- ...stMutuallyExclusivePlatformPredicates.java | 2 +- 5 files changed, 40 insertions(+), 26 deletions(-) diff --git a/hotspot/test/runtime/6888954/vmerrors.sh b/hotspot/test/runtime/6888954/vmerrors.sh index 28f6eea0fec..98540c4d235 100644 --- a/hotspot/test/runtime/6888954/vmerrors.sh +++ b/hotspot/test/runtime/6888954/vmerrors.sh @@ -1,4 +1,4 @@ -# Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2013, 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 @@ -71,11 +71,12 @@ bad_data_ptr_re='(SIGILL|SIGSEGV|EXCEPTION_ACCESS_VIOLATION).* at pc=' # EXCEPTION_ACCESS_VIOLATION - Win-* # SIGBUS - Solaris SPARC-64 # SIGSEGV - Linux-*, Solaris SPARC-32, Solaris X86-* +# SIGILL - Aix # # Note: would like to use "pc=0x00*0f," in the pattern, but Solaris SPARC-* # gets its signal at a PC in test_error_handler(). # -bad_func_ptr_re='(SIGBUS|SIGSEGV|EXCEPTION_ACCESS_VIOLATION).* at pc=' +bad_func_ptr_re='(SIGBUS|SIGSEGV|SIGILL|EXCEPTION_ACCESS_VIOLATION).* at pc=' guarantee_re='guarantee[(](str|num).*failed: *' fatal_re='fatal error: *' tail_1='.*expected null' diff --git a/hotspot/test/serviceability/dcmd/DynLibDcmdTest.java b/hotspot/test/serviceability/dcmd/DynLibDcmdTest.java index 069b06f1937..2f6b08ddd94 100644 --- a/hotspot/test/serviceability/dcmd/DynLibDcmdTest.java +++ b/hotspot/test/serviceability/dcmd/DynLibDcmdTest.java @@ -3,7 +3,7 @@ import java.util.Set; import com.oracle.java.testlibrary.Platform; /* - * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2013, 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 @@ -39,14 +39,16 @@ public class DynLibDcmdTest { String result = DcmdUtil.executeDcmd("VM.dynlibs"); String osDependentBaseString = null; - if (Platform.isSolaris()) { + if (Platform.isAix()) { + osDependentBaseString = "lib%s.so"; + } else if (Platform.isLinux()) { + osDependentBaseString = "lib%s.so"; + } else if (Platform.isOSX()) { + osDependentBaseString = "lib%s.dylib"; + } else if (Platform.isSolaris()) { osDependentBaseString = "lib%s.so"; } else if (Platform.isWindows()) { osDependentBaseString = "%s.dll"; - } else if (Platform.isOSX()) { - osDependentBaseString = "lib%s.dylib"; - } else if (Platform.isLinux()) { - osDependentBaseString = "lib%s.so"; } if (osDependentBaseString == null) { diff --git a/hotspot/test/test_env.sh b/hotspot/test/test_env.sh index e9fc3abb0ff..7fcfb93c737 100644 --- a/hotspot/test/test_env.sh +++ b/hotspot/test/test_env.sh @@ -1,6 +1,6 @@ #!/bin/sh # -# Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2013, 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 @@ -56,7 +56,7 @@ echo "TESTOPTS=${TESTOPTS}" # set platform-dependent variables OS=`uname -s` case "$OS" in - SunOS | Linux | Darwin ) + AIX | Darwin | Linux | SunOS ) NULL=/dev/null PS=":" FS="/" @@ -133,26 +133,31 @@ then fi VM_OS="unknown" -grep "solaris" vm_version.out > ${NULL} +grep "aix" vm_version.out > ${NULL} if [ $? = 0 ] then - VM_OS="solaris" + VM_OS="aix" +fi +grep "bsd" vm_version.out > ${NULL} +if [ $? = 0 ] +then + VM_OS="bsd" fi grep "linux" vm_version.out > ${NULL} if [ $? = 0 ] then VM_OS="linux" fi +grep "solaris" vm_version.out > ${NULL} +if [ $? = 0 ] +then + VM_OS="solaris" +fi grep "windows" vm_version.out > ${NULL} if [ $? = 0 ] then VM_OS="windows" fi -grep "bsd" vm_version.out > ${NULL} -if [ $? = 0 ] -then - VM_OS="bsd" -fi VM_CPU="unknown" grep "sparc" vm_version.out > ${NULL} diff --git a/hotspot/test/testlibrary/com/oracle/java/testlibrary/Platform.java b/hotspot/test/testlibrary/com/oracle/java/testlibrary/Platform.java index 8bdb863ed46..d26702deffa 100644 --- a/hotspot/test/testlibrary/com/oracle/java/testlibrary/Platform.java +++ b/hotspot/test/testlibrary/com/oracle/java/testlibrary/Platform.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2013, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2013, 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 @@ -68,20 +68,24 @@ public class Platform { return dataModel.equals("64"); } - public static boolean isSolaris() { - return isOs("sunos"); + public static boolean isAix() { + return isOs("aix"); } - public static boolean isWindows() { - return isOs("win"); + public static boolean isLinux() { + return isOs("linux"); } public static boolean isOSX() { return isOs("mac"); } - public static boolean isLinux() { - return isOs("linux"); + public static boolean isSolaris() { + return isOs("sunos"); + } + + public static boolean isWindows() { + return isOs("win"); } private static boolean isOs(String osname) { @@ -140,7 +144,9 @@ public class Platform { */ public static boolean shouldSAAttach() throws Exception { - if (isLinux()) { + if (isAix()) { + return false; // SA not implemented. + } else if (isLinux()) { return canPtraceAttachLinux(); } else if (isOSX()) { return canAttachOSX(); diff --git a/hotspot/test/testlibrary_tests/TestMutuallyExclusivePlatformPredicates.java b/hotspot/test/testlibrary_tests/TestMutuallyExclusivePlatformPredicates.java index b16d6eb7540..1f022a9d0f8 100644 --- a/hotspot/test/testlibrary_tests/TestMutuallyExclusivePlatformPredicates.java +++ b/hotspot/test/testlibrary_tests/TestMutuallyExclusivePlatformPredicates.java @@ -45,7 +45,7 @@ public class TestMutuallyExclusivePlatformPredicates { private static enum MethodGroup { ARCH("isARM", "isPPC", "isSparc", "isX86", "isX64"), BITNESS("is32bit", "is64bit"), - OS("isLinux", "isSolaris", "isWindows", "isOSX"), + OS("isAix", "isLinux", "isOSX", "isSolaris", "isWindows"), VM_TYPE("isClient", "isServer", "isGraal", "isMinimal"), IGNORED("isEmbedded", "isDebugBuild", "shouldSAAttach", "canPtraceAttachLinux", "canAttachOSX", "isTieredSupported"); From 2a73208887f42a09d59d8c7b9076b72d46ab5028 Mon Sep 17 00:00:00 2001 From: Kim Barrett Date: Fri, 9 Jan 2015 11:33:48 -0500 Subject: [PATCH 12/77] 8067306: Improve STATIC_ASSERT New improved implementation Reviewed-by: ehelin, stefank --- hotspot/src/share/vm/utilities/debug.hpp | 31 ++++++++++++------------ 1 file changed, 16 insertions(+), 15 deletions(-) diff --git a/hotspot/src/share/vm/utilities/debug.hpp b/hotspot/src/share/vm/utilities/debug.hpp index 7995e6c085d..c5ab86d76c5 100644 --- a/hotspot/src/share/vm/utilities/debug.hpp +++ b/hotspot/src/share/vm/utilities/debug.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 @@ -225,21 +225,22 @@ void report_untested(const char* file, int line, const char* message); void warning(const char* format, ...) ATTRIBUTE_PRINTF(1, 2); -#ifdef ASSERT -// Compile-time asserts. -template struct StaticAssert; -template <> struct StaticAssert {}; +// Compile-time asserts. Cond must be a compile-time constant expression that +// is convertible to bool. STATIC_ASSERT() can be used anywhere a declaration +// may appear. +// +// Implementation Note: STATIC_ASSERT_FAILURE provides a value member +// rather than type member that could be used directly in the typedef, because +// a type member would require conditional use of "typename", depending on +// whether Cond is dependent or not. The use of a value member leads to the +// use of an array type. -// Only StaticAssert is defined, so if cond evaluates to false we get -// a compile time exception when trying to use StaticAssert. -#define STATIC_ASSERT(cond) \ - do { \ - StaticAssert<(cond)> DUMMY_STATIC_ASSERT; \ - (void)DUMMY_STATIC_ASSERT; /* ignore */ \ - } while (false) -#else -#define STATIC_ASSERT(cond) -#endif +template struct STATIC_ASSERT_FAILURE; +template<> struct STATIC_ASSERT_FAILURE { enum { value = 1 }; }; + +#define STATIC_ASSERT(Cond) \ + typedef char STATIC_ASSERT_FAILURE_ ## __LINE__ [ \ + STATIC_ASSERT_FAILURE< (Cond) >::value ] // out of shared space reporting enum SharedSpaceType { From 90c58f1cc987e4adfd39e9b34fed8de332e72146 Mon Sep 17 00:00:00 2001 From: Christian Tornqvist Date: Fri, 9 Jan 2015 08:34:32 -0800 Subject: [PATCH 13/77] 8068733: [TESTBUG] runtime/Unsafe/Reallocate.java sometimes fails when running with -Xcomp Reviewed-by: coleenp, gtriantafill --- hotspot/test/runtime/Unsafe/Reallocate.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/hotspot/test/runtime/Unsafe/Reallocate.java b/hotspot/test/runtime/Unsafe/Reallocate.java index e5abe9aa9a6..19442038374 100644 --- a/hotspot/test/runtime/Unsafe/Reallocate.java +++ b/hotspot/test/runtime/Unsafe/Reallocate.java @@ -25,7 +25,7 @@ * @test * @bug 8058897 * @library /testlibrary - * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:MallocMaxTestWords=20m Reallocate + * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:MallocMaxTestWords=100m Reallocate */ import com.oracle.java.testlibrary.*; @@ -57,7 +57,7 @@ public class Reallocate { // Make sure we can throw an OOME when we fail to reallocate due to OOM try { - unsafe.reallocateMemory(address, 20 * 1024 * 1024 * 8); + unsafe.reallocateMemory(address, 100 * 1024 * 1024 * 8); } catch (OutOfMemoryError e) { // Expected return; From c041b882e3f419ad11251635d3bf0585339caef4 Mon Sep 17 00:00:00 2001 From: Dean Long Date: Fri, 9 Jan 2015 17:43:02 -0500 Subject: [PATCH 14/77] 8068746: Exclude hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java from nightly runs Add @ignore to PoolsIndependenceTest.java Reviewed-by: kvn --- hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java | 1 + 1 file changed, 1 insertion(+) diff --git a/hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java b/hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java index a2ea70a1fe2..04551a2d2df 100644 --- a/hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java +++ b/hotspot/test/compiler/codecache/jmx/PoolsIndependenceTest.java @@ -37,6 +37,7 @@ import sun.hotspot.code.BlobType; /* * @test PoolsIndependenceTest + * @ignore 8068385 * @library /testlibrary /../../test/lib * @build PoolsIndependenceTest * @run main ClassFileInstaller sun.hotspot.WhiteBox From 4e84bf1e9332e84d48b292db21d9b79350e04979 Mon Sep 17 00:00:00 2001 From: Mattias Tobiasson Date: Mon, 12 Jan 2015 09:27:52 +0100 Subject: [PATCH 15/77] 8068584: Compiler attach tests should be quarantined Quarantine tests Reviewed-by: ctornqvi --- .../jsr292/RedefineMethodUsedByMultipleMethodHandles.java | 1 + .../test/compiler/profiling/spectrapredefineclass/Launcher.java | 1 + .../profiling/spectrapredefineclass_classloaders/Launcher.java | 1 + 3 files changed, 3 insertions(+) diff --git a/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java b/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java index 1695c35558a..920431525a8 100644 --- a/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java +++ b/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java @@ -26,6 +26,7 @@ * @bug 8042235 * @summary redefining method used by multiple MethodHandles crashes VM * @compile -XDignore.symbol.file RedefineMethodUsedByMultipleMethodHandles.java + * @ignore 7076820 * @run main RedefineMethodUsedByMultipleMethodHandles */ diff --git a/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java b/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java index 7a645ab0b8f..3c4f6b92bb4 100644 --- a/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java +++ b/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java @@ -28,6 +28,7 @@ import com.oracle.java.testlibrary.*; * @bug 8038636 * @library /testlibrary * @build Agent + * @ignore 7076820 * @run main ClassFileInstaller Agent * @run main Launcher * @run main/othervm -XX:-TieredCompilation -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:TypeProfileLevel=222 -XX:ReservedCodeCacheSize=3M Agent diff --git a/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java b/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java index 3c49cd58ea7..443f11e6c16 100644 --- a/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java +++ b/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java @@ -28,6 +28,7 @@ import com.oracle.java.testlibrary.*; * @bug 8040237 * @library /testlibrary * @build Agent Test A B + * @ignore 7076820 * @run main ClassFileInstaller Agent * @run main Launcher * @run main/othervm -XX:-TieredCompilation -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:TypeProfileLevel=222 -XX:ReservedCodeCacheSize=3M Agent From 53821ffdfa02461a0f72eceeffd2486e87a08446 Mon Sep 17 00:00:00 2001 From: Martin Doerr Date: Mon, 12 Jan 2015 11:14:49 +0100 Subject: [PATCH 16/77] 8068724: ppc64: update assembler: SPR access, CR logic, HTM Fix bug in encoding of special purpose registers. Provide more convenient version of condition register logic instructions. Enhance support for hardware transactional memory. Reviewed-by: kvn, goetz --- hotspot/src/cpu/ppc/vm/assembler_ppc.hpp | 70 +++++++++++++++---- .../src/cpu/ppc/vm/assembler_ppc.inline.hpp | 53 +++++++++++++- hotspot/src/cpu/ppc/vm/interp_masm_ppc_64.cpp | 10 +-- hotspot/src/cpu/ppc/vm/stubGenerator_ppc.cpp | 6 +- .../src/cpu/ppc/vm/templateTable_ppc_64.cpp | 12 ++-- 5 files changed, 121 insertions(+), 30 deletions(-) diff --git a/hotspot/src/cpu/ppc/vm/assembler_ppc.hpp b/hotspot/src/cpu/ppc/vm/assembler_ppc.hpp index 7d9aecb310f..f1087dbc02c 100644 --- a/hotspot/src/cpu/ppc/vm/assembler_ppc.hpp +++ b/hotspot/src/cpu/ppc/vm/assembler_ppc.hpp @@ -1,6 +1,6 @@ /* - * Copyright (c) 2002, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2013 SAP AG. All rights reserved. + * Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright 2012, 2015 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 @@ -284,19 +284,20 @@ class Assembler : public AbstractAssembler { MTCTR_OPCODE = (MTSPR_OPCODE | 9 << SPR_0_4_SHIFT), MFCTR_OPCODE = (MFSPR_OPCODE | 9 << SPR_0_4_SHIFT), - MTTFHAR_OPCODE = (MTSPR_OPCODE | 128 << SPR_0_4_SHIFT), - MFTFHAR_OPCODE = (MFSPR_OPCODE | 128 << SPR_0_4_SHIFT), - MTTFIAR_OPCODE = (MTSPR_OPCODE | 129 << SPR_0_4_SHIFT), - MFTFIAR_OPCODE = (MFSPR_OPCODE | 129 << SPR_0_4_SHIFT), - MTTEXASR_OPCODE = (MTSPR_OPCODE | 130 << SPR_0_4_SHIFT), - MFTEXASR_OPCODE = (MFSPR_OPCODE | 130 << SPR_0_4_SHIFT), - MTTEXASRU_OPCODE = (MTSPR_OPCODE | 131 << SPR_0_4_SHIFT), - MFTEXASRU_OPCODE = (MFSPR_OPCODE | 131 << SPR_0_4_SHIFT), + // Attention: Higher and lower half are inserted in reversed order. + MTTFHAR_OPCODE = (MTSPR_OPCODE | 4 << SPR_5_9_SHIFT | 0 << SPR_0_4_SHIFT), + MFTFHAR_OPCODE = (MFSPR_OPCODE | 4 << SPR_5_9_SHIFT | 0 << SPR_0_4_SHIFT), + MTTFIAR_OPCODE = (MTSPR_OPCODE | 4 << SPR_5_9_SHIFT | 1 << SPR_0_4_SHIFT), + MFTFIAR_OPCODE = (MFSPR_OPCODE | 4 << SPR_5_9_SHIFT | 1 << SPR_0_4_SHIFT), + MTTEXASR_OPCODE = (MTSPR_OPCODE | 4 << SPR_5_9_SHIFT | 2 << SPR_0_4_SHIFT), + MFTEXASR_OPCODE = (MFSPR_OPCODE | 4 << SPR_5_9_SHIFT | 2 << SPR_0_4_SHIFT), + MTTEXASRU_OPCODE = (MTSPR_OPCODE | 4 << SPR_5_9_SHIFT | 3 << SPR_0_4_SHIFT), + MFTEXASRU_OPCODE = (MFSPR_OPCODE | 4 << SPR_5_9_SHIFT | 3 << SPR_0_4_SHIFT), - MTVRSAVE_OPCODE = (MTSPR_OPCODE | 256 << SPR_0_4_SHIFT), - MFVRSAVE_OPCODE = (MFSPR_OPCODE | 256 << SPR_0_4_SHIFT), + MTVRSAVE_OPCODE = (MTSPR_OPCODE | 8 << SPR_5_9_SHIFT | 0 << SPR_0_4_SHIFT), + MFVRSAVE_OPCODE = (MFSPR_OPCODE | 8 << SPR_5_9_SHIFT | 0 << SPR_0_4_SHIFT), - MFTB_OPCODE = (MFSPR_OPCODE | 268 << SPR_0_4_SHIFT), + MFTB_OPCODE = (MFSPR_OPCODE | 8 << SPR_5_9_SHIFT | 12 << SPR_0_4_SHIFT), MTCRF_OPCODE = (31u << OPCODE_SHIFT | 144u << 1), MFCR_OPCODE = (31u << OPCODE_SHIFT | 19u << 1), @@ -1494,6 +1495,26 @@ class Assembler : public AbstractAssembler { inline void mftexasr(Register d); inline void mftexasru(Register d); + // TEXASR bit description + enum transaction_failure_reason { + // Upper half (TEXASRU): + tm_failure_persistent = 7, // The failure is likely to recur on each execution. + tm_disallowed = 8, // The instruction is not permitted. + tm_nesting_of = 9, // The maximum transaction level was exceeded. + tm_footprint_of = 10, // The tracking limit for transactional storage accesses was exceeded. + tm_self_induced_cf = 11, // A self-induced conflict occurred in Suspended state. + tm_non_trans_cf = 12, // A conflict occurred with a non-transactional access by another processor. + tm_trans_cf = 13, // A conflict occurred with another transaction. + tm_translation_cf = 14, // A conflict occurred with a TLB invalidation. + tm_inst_fetch_cf = 16, // An instruction fetch was performed from a block that was previously written transactionally. + tm_tabort = 31, // Termination was caused by the execution of an abort instruction. + // Lower half: + tm_suspended = 32, // Failure was recorded in Suspended state. + tm_failure_summary = 36, // Failure has been detected and recorded. + tm_tfiar_exact = 37, // Value in the TFIAR is exact. + tm_rot = 38, // Rollback-only transaction. + }; + // PPC 1, section 2.4.1 Branch Instructions inline void b( address a, relocInfo::relocType rt = relocInfo::none); inline void b( Label& L); @@ -1581,6 +1602,7 @@ class Assembler : public AbstractAssembler { inline void bnectrl(ConditionRegister crx, relocInfo::relocType rt = relocInfo::none); // condition register logic instructions + // NOTE: There's a preferred form: d and s2 should point into the same condition register. inline void crand( int d, int s1, int s2); inline void crnand(int d, int s1, int s2); inline void cror( int d, int s1, int s2); @@ -1590,6 +1612,19 @@ class Assembler : public AbstractAssembler { inline void crandc(int d, int s1, int s2); inline void crorc( int d, int s1, int s2); + // More convenient version. + int condition_register_bit(ConditionRegister cr, Condition c) { + return 4 * (int)(intptr_t)cr + c; + } + void crand( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void crnand(ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void cror( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void crxor( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void crnor( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void creqv( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void crandc(ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + void crorc( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc); + // icache and dcache related instructions inline void icbi( Register s1, Register s2); //inline void dcba(Register s1, Register s2); // Instruction for embedded processor only. @@ -1673,6 +1708,10 @@ class Assembler : public AbstractAssembler { inline void smt_prio_low(); inline void smt_prio_medium_low(); inline void smt_prio_medium(); + // >= Power7 + inline void smt_yield(); + inline void smt_mdoio(); + inline void smt_mdoom(); // trap instructions inline void twi_0(Register a); // for load with acquire semantics use load+twi_0+isync (trap can't occur) @@ -1958,6 +1997,7 @@ class Assembler : public AbstractAssembler { inline void tbeginrot_(); // R=1 Rollback-Only Transaction inline void tend_(); // A=0 inline void tendall_(); // A=1 + inline void tabort_(); inline void tabort_(Register a); inline void tabortwc_(int t, Register a, Register b); inline void tabortwci_(int t, Register a, int si); @@ -1967,6 +2007,10 @@ class Assembler : public AbstractAssembler { inline void tresume_(); // tsr with L=1 inline void tcheck(int f); + static bool is_tbegin(int x) { + return TBEGIN_OPCODE == (x & (0x3f << OPCODE_SHIFT | 0x3ff << 1)); + } + // The following encoders use r0 as second operand. These instructions // read r0 as '0'. inline void lwzx( Register d, Register s2); diff --git a/hotspot/src/cpu/ppc/vm/assembler_ppc.inline.hpp b/hotspot/src/cpu/ppc/vm/assembler_ppc.inline.hpp index b4a7370994d..5493f124371 100644 --- a/hotspot/src/cpu/ppc/vm/assembler_ppc.inline.hpp +++ b/hotspot/src/cpu/ppc/vm/assembler_ppc.inline.hpp @@ -1,6 +1,6 @@ /* - * Copyright (c) 2002, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2014 SAP AG. All rights reserved. + * Copyright (c) 2002, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright 2012, 2015 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 @@ -453,6 +453,48 @@ inline void Assembler::creqv( int d, int s1, int s2) { emit_int32(CREQV_OPCODE inline void Assembler::crandc(int d, int s1, int s2) { emit_int32(CRANDC_OPCODE | bt(d) | ba(s1) | bb(s2)); } inline void Assembler::crorc( int d, int s1, int s2) { emit_int32(CRORC_OPCODE | bt(d) | ba(s1) | bb(s2)); } +// More convenient version. +inline void Assembler::crand( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crand(dst_bit, src_bit, dst_bit); +} +inline void Assembler::crnand(ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crnand(dst_bit, src_bit, dst_bit); +} +inline void Assembler::cror( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + cror(dst_bit, src_bit, dst_bit); +} +inline void Assembler::crxor( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crxor(dst_bit, src_bit, dst_bit); +} +inline void Assembler::crnor( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crnor(dst_bit, src_bit, dst_bit); +} +inline void Assembler::creqv( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + creqv(dst_bit, src_bit, dst_bit); +} +inline void Assembler::crandc(ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crandc(dst_bit, src_bit, dst_bit); +} +inline void Assembler::crorc( ConditionRegister crdst, Condition cdst, ConditionRegister crsrc, Condition csrc) { + int dst_bit = condition_register_bit(crdst, cdst), + src_bit = condition_register_bit(crsrc, csrc); + crorc(dst_bit, src_bit, dst_bit); +} + // Conditional move (>= Power7) inline void Assembler::isel(Register d, ConditionRegister cr, Condition cc, bool inv, Register a, Register b) { if (b == noreg) { @@ -516,6 +558,10 @@ inline void Assembler::smt_prio_medium_low() { Assembler::or_unchecked(R6, R6, inline void Assembler::smt_prio_medium() { Assembler::or_unchecked(R2, R2, R2); } inline void Assembler::smt_prio_medium_high() { Assembler::or_unchecked(R5, R5, R5); } inline void Assembler::smt_prio_high() { Assembler::or_unchecked(R3, R3, R3); } +// >= Power7 +inline void Assembler::smt_yield() { Assembler::or_unchecked(R27, R27, R27); } +inline void Assembler::smt_mdoio() { Assembler::or_unchecked(R29, R29, R29); } +inline void Assembler::smt_mdoom() { Assembler::or_unchecked(R30, R30, R30); } inline void Assembler::twi_0(Register a) { twi_unchecked(0, a, 0);} @@ -778,7 +824,8 @@ inline void Assembler::tbegin_() { emit_int32( TB inline void Assembler::tbeginrot_() { emit_int32( TBEGIN_OPCODE | /*R=1*/ 1u << (31-10) | rc(1)); } inline void Assembler::tend_() { emit_int32( TEND_OPCODE | rc(1)); } inline void Assembler::tendall_() { emit_int32( TEND_OPCODE | /*A=1*/ 1u << (31-6) | rc(1)); } -inline void Assembler::tabort_(Register a) { emit_int32( TABORT_OPCODE | ra(a) | rc(1)); } +inline void Assembler::tabort_() { emit_int32( TABORT_OPCODE | rc(1)); } +inline void Assembler::tabort_(Register a) { assert(a != R0, "r0 not allowed"); emit_int32( TABORT_OPCODE | ra(a) | rc(1)); } inline void Assembler::tabortwc_(int t, Register a, Register b) { emit_int32( TABORTWC_OPCODE | to(t) | ra(a) | rb(b) | rc(1)); } inline void Assembler::tabortwci_(int t, Register a, int si) { emit_int32( TABORTWCI_OPCODE | to(t) | ra(a) | sh1620(si) | rc(1)); } inline void Assembler::tabortdc_(int t, Register a, Register b) { emit_int32( TABORTDC_OPCODE | to(t) | ra(a) | rb(b) | rc(1)); } diff --git a/hotspot/src/cpu/ppc/vm/interp_masm_ppc_64.cpp b/hotspot/src/cpu/ppc/vm/interp_masm_ppc_64.cpp index dec0d9732a8..9cb22c11433 100644 --- a/hotspot/src/cpu/ppc/vm/interp_masm_ppc_64.cpp +++ b/hotspot/src/cpu/ppc/vm/interp_masm_ppc_64.cpp @@ -1,6 +1,6 @@ /* - * Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2014 SAP AG. All rights reserved. + * Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright 2012, 2015 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 @@ -1712,7 +1712,7 @@ void InterpreterMacroAssembler::profile_obj_type(Register obj, Register mdo_addr andi_(R0, klass, TypeEntries::type_unknown); // Already unknown. Nothing to do anymore. //bne(CCR0, do_nothing); - crorc(/*CCR0 eq*/2, /*CCR1 eq*/4+2, /*CCR0 eq*/2); // cr0 eq = cr1 eq or cr0 ne + crorc(CCR0, Assembler::equal, CCR1, Assembler::equal); // cr0 eq = cr1 eq or cr0 ne beq(CCR0, do_nothing); clrrdi_(R0, tmp, exact_log2(-TypeEntries::type_mask)); @@ -1826,9 +1826,9 @@ void InterpreterMacroAssembler::profile_return_type(Register ret, Register tmp1, lbz(tmp2, Method::intrinsic_id_offset_in_bytes(), R19_method); cmpwi(CCR0, tmp1, Bytecodes::_invokedynamic); cmpwi(CCR1, tmp1, Bytecodes::_invokehandle); - cror(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); + cror(CCR0, Assembler::equal, CCR1, Assembler::equal); cmpwi(CCR1, tmp2, vmIntrinsics::_compiledLambdaForm); - cror(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); + cror(CCR0, Assembler::equal, CCR1, Assembler::equal); bne(CCR0, profile_continue); } diff --git a/hotspot/src/cpu/ppc/vm/stubGenerator_ppc.cpp b/hotspot/src/cpu/ppc/vm/stubGenerator_ppc.cpp index 0f8c752cacc..43258ce2a9a 100644 --- a/hotspot/src/cpu/ppc/vm/stubGenerator_ppc.cpp +++ b/hotspot/src/cpu/ppc/vm/stubGenerator_ppc.cpp @@ -1,6 +1,6 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. - * Copyright 2012, 2014 SAP AG. All rights reserved. + * Copyright (c) 1997, 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright 2012, 2015 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 @@ -1079,7 +1079,7 @@ class StubGenerator: public StubCodeGenerator { __ sldi(tmp2, R5_ARG3, log2_elem_size); // size in bytes __ cmpld(CCR0, R3_ARG1, R4_ARG2); // Use unsigned comparison! __ cmpld(CCR1, tmp1, tmp2); - __ crand(/*CCR0 lt*/0, /*CCR1 lt*/4+0, /*CCR0 lt*/0); + __ crand(CCR0, Assembler::less, CCR1, Assembler::less); __ blt(CCR0, l_overlap); // Src before dst and distance smaller than size. // need to copy forwards diff --git a/hotspot/src/cpu/ppc/vm/templateTable_ppc_64.cpp b/hotspot/src/cpu/ppc/vm/templateTable_ppc_64.cpp index 40cd339c2c0..69c1e0d1be1 100644 --- a/hotspot/src/cpu/ppc/vm/templateTable_ppc_64.cpp +++ b/hotspot/src/cpu/ppc/vm/templateTable_ppc_64.cpp @@ -1,6 +1,6 @@ /* - * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. - * Copyright 2013, 2014 SAP AG. All rights reserved. + * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. + * Copyright 2013, 2015 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 @@ -335,11 +335,11 @@ void TemplateTable::ldc(bool wide) { __ cmpwi(CCR0, Rscratch2, JVM_CONSTANT_UnresolvedClass); // Unresolved class? __ cmpwi(CCR1, Rscratch2, JVM_CONSTANT_UnresolvedClassInError); // Unresolved class in error state? - __ cror(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); + __ cror(CCR0, Assembler::equal, CCR1, Assembler::equal); // Resolved class - need to call vm to get java mirror of the class. __ cmpwi(CCR1, Rscratch2, JVM_CONSTANT_Class); - __ crnor(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); // Neither resolved class nor unresolved case from above? + __ crnor(CCR0, Assembler::equal, CCR1, Assembler::equal); // Neither resolved class nor unresolved case from above? __ beq(CCR0, notClass); __ li(R4, wide ? 1 : 0); @@ -2611,7 +2611,7 @@ void TemplateTable::jvmti_post_field_mod(Register Rcache, Register Rscratch, boo __ cmpwi(CCR0, Rflags, ltos); __ cmpwi(CCR1, Rflags, dtos); __ addi(base, R15_esp, Interpreter::expr_offset_in_bytes(1)); - __ crnor(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); + __ crnor(CCR0, Assembler::equal, CCR1, Assembler::equal); __ beq(CCR0, is_one_slot); __ addi(base, R15_esp, Interpreter::expr_offset_in_bytes(2)); __ bind(is_one_slot); @@ -3563,7 +3563,7 @@ void TemplateTable::_new() { // Make sure klass does not have has_finalizer, or is abstract, or interface or java/lang/Class. __ andi_(R0, Rinstance_size, Klass::_lh_instance_slow_path_bit); // slow path bit equals 0? - __ crnand(/*CR0 eq*/2, /*CR1 eq*/4+2, /*CR0 eq*/2); // slow path bit set or not fully initialized? + __ crnand(CCR0, Assembler::equal, CCR1, Assembler::equal); // slow path bit set or not fully initialized? __ beq(CCR0, Lslow_case); // -------------------------------------------------------------------------- From b5909f6d03d5eb6af1de21dedb67ddfbac0ce29e Mon Sep 17 00:00:00 2001 From: Zoltan Majo Date: Mon, 12 Jan 2015 09:55:20 +0100 Subject: [PATCH 17/77] 8065894: CodeHeap::next_free should be renamed Rename next_free() to next_used() Reviewed-by: thartmann, iveresov --- hotspot/src/share/vm/memory/heap.cpp | 10 ++++++---- hotspot/src/share/vm/memory/heap.hpp | 6 +++--- 2 files changed, 9 insertions(+), 7 deletions(-) diff --git a/hotspot/src/share/vm/memory/heap.cpp b/hotspot/src/share/vm/memory/heap.cpp index 5ba099e78e8..5501700312e 100644 --- a/hotspot/src/share/vm/memory/heap.cpp +++ b/hotspot/src/share/vm/memory/heap.cpp @@ -279,10 +279,12 @@ size_t CodeHeap::alignment_offset() const { return sizeof(HeapBlock) & (_segment_size - 1); } -// Finds the next free heapblock. If the current one is free, that it returned -void* CodeHeap::next_free(HeapBlock* b) const { - // Since free blocks are merged, there is max. on free block - // between two used ones +// Returns the current block if available and used. +// If not, it returns the subsequent block (if available), NULL otherwise. +// Free blocks are merged, therefore there is at most one free block +// between two used ones. As a result, the subsequent block (if available) is +// guaranteed to be used. +void* CodeHeap::next_used(HeapBlock* b) const { if (b != NULL && b->free()) b = next_block(b); assert(b == NULL || !b->free(), "must be in use or at end of heap"); return (b == NULL) ? NULL : b->allocated_space(); diff --git a/hotspot/src/share/vm/memory/heap.hpp b/hotspot/src/share/vm/memory/heap.hpp index 4fdbeabbfc3..45b43ce06c1 100644 --- a/hotspot/src/share/vm/memory/heap.hpp +++ b/hotspot/src/share/vm/memory/heap.hpp @@ -123,7 +123,7 @@ class CodeHeap : public CHeapObj { FreeBlock* search_freelist(size_t length); // Iteration helpers - void* next_free(HeapBlock* b) const; + void* next_used(HeapBlock* b) const; HeapBlock* first_block() const; HeapBlock* next_block(HeapBlock* b) const; HeapBlock* block_start(void* p) const; @@ -158,9 +158,9 @@ class CodeHeap : public CHeapObj { int freelist_length() const { return _freelist_length; } // number of elements in the freelist // returns the first block or NULL - void* first() const { return next_free(first_block()); } + void* first() const { return next_used(first_block()); } // returns the next block given a block p or NULL - void* next(void* p) const { return next_free(next_block(block_start(p))); } + void* next(void* p) const { return next_used(next_block(block_start(p))); } // Statistics size_t capacity() const; From 7a8a19d1d9bd2046d4b6bee20d74fe9ca3885ffc Mon Sep 17 00:00:00 2001 From: Stefan Johansson Date: Mon, 12 Jan 2015 15:24:29 +0100 Subject: [PATCH 18/77] 8062063: Usage of UseHugeTLBFS, UseLargePagesInMetaspace and huge SurvivorAlignmentInBytes cause crashes in CMBitMapClosure::do_bit Making sure committed memory is cleared when re-committed, even if using large pages. Reviewed-by: kbarrett, tschatzl --- .../g1/g1PageBasedVirtualSpace.cpp | 40 +++++++++++++------ .../g1/g1PageBasedVirtualSpace.hpp | 13 ++++-- .../g1/g1RegionToSpaceMapper.cpp | 7 ++-- 3 files changed, 39 insertions(+), 21 deletions(-) diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.cpp b/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.cpp index d972b5cc71c..008002566da 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.cpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.cpp @@ -45,7 +45,8 @@ #include "utilities/bitMap.inline.hpp" G1PageBasedVirtualSpace::G1PageBasedVirtualSpace() : _low_boundary(NULL), - _high_boundary(NULL), _committed(), _page_size(0), _special(false), _executable(false) { + _high_boundary(NULL), _committed(), _page_size(0), _special(false), + _dirty(), _executable(false) { } bool G1PageBasedVirtualSpace::initialize_with_granularity(ReservedSpace rs, size_t page_size) { @@ -66,6 +67,9 @@ bool G1PageBasedVirtualSpace::initialize_with_granularity(ReservedSpace rs, size assert(_committed.size() == 0, "virtual space initialized more than once"); uintx size_in_bits = rs.size() / page_size; _committed.resize(size_in_bits, /* in_resource_area */ false); + if (_special) { + _dirty.resize(size_in_bits, /* in_resource_area */ false); + } return true; } @@ -84,6 +88,7 @@ void G1PageBasedVirtualSpace::release() { _executable = false; _page_size = 0; _committed.resize(0, false); + _dirty.resize(0, false); } size_t G1PageBasedVirtualSpace::committed_size() const { @@ -120,34 +125,43 @@ size_t G1PageBasedVirtualSpace::byte_size_for_pages(size_t num) { return num * _page_size; } -MemRegion G1PageBasedVirtualSpace::commit(uintptr_t start, size_t size_in_pages) { +bool G1PageBasedVirtualSpace::commit(uintptr_t start, size_t size_in_pages) { // We need to make sure to commit all pages covered by the given area. guarantee(is_area_uncommitted(start, size_in_pages), "Specified area is not uncommitted"); - if (!_special) { + bool zero_filled = true; + uintptr_t end = start + size_in_pages; + + if (_special) { + // Check for dirty pages and update zero_filled if any found. + if (_dirty.get_next_one_offset(start,end) < end) { + zero_filled = false; + _dirty.clear_range(start, end); + } + } else { os::commit_memory_or_exit(page_start(start), byte_size_for_pages(size_in_pages), _executable, err_msg("Failed to commit pages from "SIZE_FORMAT" of length "SIZE_FORMAT, start, size_in_pages)); } - _committed.set_range(start, start + size_in_pages); + _committed.set_range(start, end); - MemRegion result((HeapWord*)page_start(start), byte_size_for_pages(size_in_pages) / HeapWordSize); if (AlwaysPreTouch) { - os::pretouch_memory((char*)result.start(), (char*)result.end()); + os::pretouch_memory(page_start(start), page_start(end)); } - return result; + return zero_filled; } -MemRegion G1PageBasedVirtualSpace::uncommit(uintptr_t start, size_t size_in_pages) { +void G1PageBasedVirtualSpace::uncommit(uintptr_t start, size_t size_in_pages) { guarantee(is_area_committed(start, size_in_pages), "checking"); - if (!_special) { + if (_special) { + // Mark that memory is dirty. If committed again the memory might + // need to be cleared explicitly. + _dirty.set_range(start, start + size_in_pages); + } else { os::uncommit_memory(page_start(start), byte_size_for_pages(size_in_pages)); } _committed.clear_range(start, start + size_in_pages); - - MemRegion result((HeapWord*)page_start(start), byte_size_for_pages(size_in_pages) / HeapWordSize); - return result; } bool G1PageBasedVirtualSpace::contains(const void* p) const { @@ -157,7 +171,7 @@ bool G1PageBasedVirtualSpace::contains(const void* p) const { #ifndef PRODUCT void G1PageBasedVirtualSpace::print_on(outputStream* out) { out->print ("Virtual space:"); - if (special()) out->print(" (pinned in memory)"); + if (_special) out->print(" (pinned in memory)"); out->cr(); out->print_cr(" - committed: " SIZE_FORMAT, committed_size()); out->print_cr(" - reserved: " SIZE_FORMAT, reserved_size()); diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.hpp b/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.hpp index 972a69b2e8f..fb2c78415f9 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.hpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1PageBasedVirtualSpace.hpp @@ -49,6 +49,12 @@ class G1PageBasedVirtualSpace VALUE_OBJ_CLASS_SPEC { // Bitmap used for verification of commit/uncommit operations. BitMap _committed; + // Bitmap used to keep track of which pages are dirty or not for _special + // spaces. This is needed because for those spaces the underlying memory + // will only be zero filled the first time it is committed. Calls to commit + // will use this bitmap and return whether or not the memory is zero filled. + BitMap _dirty; + // Indicates that the entire space has been committed and pinned in memory, // os::commit_memory() or os::uncommit_memory() have no function. bool _special; @@ -71,12 +77,11 @@ class G1PageBasedVirtualSpace VALUE_OBJ_CLASS_SPEC { public: // Commit the given area of pages starting at start being size_in_pages large. - MemRegion commit(uintptr_t start, size_t size_in_pages); + // Returns true if the given area is zero filled upon completion. + bool commit(uintptr_t start, size_t size_in_pages); // Uncommit the given area of pages starting at start being size_in_pages large. - MemRegion uncommit(uintptr_t start, size_t size_in_pages); - - bool special() const { return _special; } + void uncommit(uintptr_t start, size_t size_in_pages); // Initialization G1PageBasedVirtualSpace(); diff --git a/hotspot/src/share/vm/gc_implementation/g1/g1RegionToSpaceMapper.cpp b/hotspot/src/share/vm/gc_implementation/g1/g1RegionToSpaceMapper.cpp index 41eb0265dbf..68d967c764a 100644 --- a/hotspot/src/share/vm/gc_implementation/g1/g1RegionToSpaceMapper.cpp +++ b/hotspot/src/share/vm/gc_implementation/g1/g1RegionToSpaceMapper.cpp @@ -67,9 +67,9 @@ class G1RegionsLargerThanCommitSizeMapper : public G1RegionToSpaceMapper { } virtual void commit_regions(uintptr_t start_idx, size_t num_regions) { - _storage.commit(start_idx * _pages_per_region, num_regions * _pages_per_region); + bool zero_filled = _storage.commit(start_idx * _pages_per_region, num_regions * _pages_per_region); _commit_map.set_range(start_idx, start_idx + num_regions); - fire_on_commit(start_idx, num_regions, true); + fire_on_commit(start_idx, num_regions, zero_filled); } virtual void uncommit_regions(uintptr_t start_idx, size_t num_regions) { @@ -117,8 +117,7 @@ class G1RegionsSmallerThanCommitSizeMapper : public G1RegionToSpaceMapper { uint old_refcount = _refcounts.get_by_index(idx); bool zero_filled = false; if (old_refcount == 0) { - _storage.commit(idx, 1); - zero_filled = true; + zero_filled = _storage.commit(idx, 1); } _refcounts.set_by_index(idx, old_refcount + 1); _commit_map.set_bit(i); From fc66edfd511419380931bf3d6e7db22fd12c5481 Mon Sep 17 00:00:00 2001 From: Filipp Zhinkin Date: Wed, 14 Jan 2015 09:53:36 +0300 Subject: [PATCH 19/77] 8068269: RTM tests that assert on non-zero lock statistics are too strict in RTMTotalCountIncrRate > 1 cases Reviewed-by: kvn, iignatyev --- .../locking/TestRTMTotalCountIncrRate.java | 22 ++++++++++++++----- .../TestPrintPreciseRTMLockingStatistics.java | 3 --- 2 files changed, 17 insertions(+), 8 deletions(-) diff --git a/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java b/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java index 7a3327606fd..d27a87d8c2e 100644 --- a/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java +++ b/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java @@ -35,6 +35,7 @@ * -XX:+WhiteBoxAPI TestRTMTotalCountIncrRate */ +import sun.misc.Unsafe; import java.util.List; import com.oracle.java.testlibrary.*; @@ -97,14 +98,12 @@ public class TestRTMTotalCountIncrRate extends CommandLineOptionTest { Asserts.assertEQ(lock.getTotalLocks(), Test.TOTAL_ITERATIONS, "Total locks should be exactly the same as amount of " + "iterations."); - } else { - Asserts.assertGT(lock.getTotalLocks(), 0L, "RTM statistics " - + "should contain information for at least on lock."); } } public static class Test implements CompilableTest { private static final long TOTAL_ITERATIONS = 10000L; + private static final Unsafe UNSAFE = Utils.getUnsafe(); private final Object monitor = new Object(); // Following field have to be static in order to avoid escape analysis. @SuppressWarnings("UnsuedDeclaration") @@ -120,8 +119,17 @@ public class TestRTMTotalCountIncrRate extends CommandLineOptionTest { return new String[] { getMethodWithLockName() }; } - public void lock() { + public void lock(booleab forceAbort) { synchronized(monitor) { + if (forceAbort) { + // We're calling native method in order to force + // abort. It's done by explicit xabort call emitted + // in SharedRuntime::generate_native_wrapper. + // If an actuall JNI call will be replaced by + // intrinsic - we'll be in trouble, since xabort + // will be no longer called and test may fail. + UNSAFE.addressSize(); + } Test.field++; } } @@ -140,7 +148,11 @@ public class TestRTMTotalCountIncrRate extends CommandLineOptionTest { for (long i = 0L; i < Test.TOTAL_ITERATIONS; i++) { AbortProvoker.verifyMonitorState(test.monitor, shouldBeInflated); - test.lock(); + // Force abort on first iteration to avoid rare case when + // there were no aborts and locks count was not incremented + // with RTMTotalCountIncrRate > 1 (in such case JVM won't + // print JVM locking statistics). + test.lock(i == 0); } } } diff --git a/hotspot/test/compiler/rtm/print/TestPrintPreciseRTMLockingStatistics.java b/hotspot/test/compiler/rtm/print/TestPrintPreciseRTMLockingStatistics.java index 80b412272b0..291d4cabef5 100644 --- a/hotspot/test/compiler/rtm/print/TestPrintPreciseRTMLockingStatistics.java +++ b/hotspot/test/compiler/rtm/print/TestPrintPreciseRTMLockingStatistics.java @@ -125,9 +125,6 @@ public class TestPrintPreciseRTMLockingStatistics RTMLockingStatistics lock = statistics.get(0); - Asserts.assertGT(lock.getTotalLocks(), 0L, "RTM locking statistics " - + "should contain non zero total locks count"); - Asserts.assertGT(lock.getTotalAborts(), 0L, "RTM locking statistics should contain non zero total aborts " + "count"); From 251b4df76459b71dc3adc249be0bb24ba4f057ec Mon Sep 17 00:00:00 2001 From: Filipp Zhinkin Date: Wed, 14 Jan 2015 09:54:45 +0300 Subject: [PATCH 20/77] 8059342: Add test to cover JDK-8030976 Reviewed-by: kvn, iignatyev --- .../testlibrary/uncommontrap/Verifier.java | 145 ++++++++++ .../uncommontrap/TestUnstableIfTrap.java | 257 ++++++++++++++++++ 2 files changed, 402 insertions(+) create mode 100644 hotspot/test/compiler/testlibrary/uncommontrap/Verifier.java create mode 100644 hotspot/test/compiler/uncommontrap/TestUnstableIfTrap.java diff --git a/hotspot/test/compiler/testlibrary/uncommontrap/Verifier.java b/hotspot/test/compiler/testlibrary/uncommontrap/Verifier.java new file mode 100644 index 00000000000..5ec38bf254a --- /dev/null +++ b/hotspot/test/compiler/testlibrary/uncommontrap/Verifier.java @@ -0,0 +1,145 @@ +/* + * Copyright (c) 2014, 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. + */ + +package uncommontrap; + +import java.io.FileReader; +import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.util.Iterator; +import java.util.List; +import java.util.Properties; +import java.util.regex.Pattern; + +import com.oracle.java.testlibrary.Asserts; +/** + * Utility tool aimed to verify presence or absence of specified uncommon trap + * in compilation log. + */ +public class Verifier { + public static final String PROPERTIES_FILE_SUFFIX = ".verify.properties"; + public static final String VERIFICATION_SHOULD_BE_SKIPPED + = "uncommon.trap.verification.skipped"; + public static final String UNCOMMON_TRAP_NAME = "uncommon.trap.name"; + public static final String UNCOMMON_TRAP_BCI = "uncommon.trap.bci"; + public static final String UNCOMMON_TRAP_COMMENT = "uncommon.trap.comment"; + public static final String UNCOMMON_TRAP_ACTION = "uncommon.trap.action"; + public static final String UNCOMMON_TRAP_SHOULD_EMITTED + = "uncommon.trap.emitted"; + public static final String UNCOMMON_TRAP_SHOULD_FIRED + = "uncommon.trap.fired"; + + private static final String EMITTED_TRAP_PATTERN + = " compLogContent = Files.readAllLines(compLogFile); + verifyUncommonTrapEmitted(properties, compLogContent); + verifyUncommonTrapFired(properties, compLogContent); + } + + private static void verifyUncommonTrapEmitted(Properties properties, + List compLogContent) { + String emittedTrapRE = String.format(EMITTED_TRAP_PATTERN, + properties.getProperty(UNCOMMON_TRAP_BCI, ".*"), + properties.getProperty(UNCOMMON_TRAP_NAME, ".*"), + properties.getProperty(UNCOMMON_TRAP_ACTION, ".*"), + properties.getProperty(UNCOMMON_TRAP_COMMENT, ".*")); + Pattern pattern = Pattern.compile(emittedTrapRE); + + long trapsCount = compLogContent.stream() + .filter(line -> pattern.matcher(line).find()) + .count(); + + boolean shouldBeEmitted = Boolean.valueOf( + properties.getProperty(UNCOMMON_TRAP_SHOULD_EMITTED)); + + Asserts.assertEQ(shouldBeEmitted, trapsCount > 0, String.format( + "Uncommon trap that matches following string in compilation log" + + " should %sbe emitted: %s.", + (shouldBeEmitted ? " " : "not "), emittedTrapRE)); + } + + private static void verifyUncommonTrapFired(Properties properties, + List compLogContent) { + String firedTrapRE = String.format(FIRED_TRAP_PATTERN, + properties.getProperty(UNCOMMON_TRAP_NAME, ".*"), + properties.getProperty(UNCOMMON_TRAP_ACTION, ".*")); + String jvmsRE = String.format(JVMS_PATTERN, + properties.getProperty(UNCOMMON_TRAP_BCI, ".*")); + + boolean trapFired = false; + Pattern firedTrapPattern = Pattern.compile(firedTrapRE); + Pattern jvmsPattern = Pattern.compile(jvmsRE); + + Iterator iterator = compLogContent.iterator(); + while (iterator.hasNext() && !trapFired) { + trapFired = firedTrapPattern.matcher(iterator.next()).find() + && jvmsPattern.matcher(iterator.next()).find(); + } + + boolean shouldBeFired = Boolean.valueOf( + properties.getProperty(UNCOMMON_TRAP_SHOULD_FIRED)); + Asserts.assertEQ(shouldBeFired, trapFired, String.format( + "Uncommon trap that matches following string in compilation log" + + " should %sbe fired: %s.", + (shouldBeFired ? "" : "not "), firedTrapRE)); + } +} + diff --git a/hotspot/test/compiler/uncommontrap/TestUnstableIfTrap.java b/hotspot/test/compiler/uncommontrap/TestUnstableIfTrap.java new file mode 100644 index 00000000000..87aaf66d247 --- /dev/null +++ b/hotspot/test/compiler/uncommontrap/TestUnstableIfTrap.java @@ -0,0 +1,257 @@ +/* + * Copyright (c) 2014, 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.io.File; +import java.io.FileWriter; +import java.io.IOException; +import java.lang.reflect.Method; +import java.util.Properties; + +import com.oracle.java.testlibrary.ByteCodeLoader; +import com.oracle.java.testlibrary.Platform; +import jdk.internal.org.objectweb.asm.ClassVisitor; +import jdk.internal.org.objectweb.asm.ClassWriter; +import jdk.internal.org.objectweb.asm.Label; +import jdk.internal.org.objectweb.asm.MethodVisitor; +import static jdk.internal.org.objectweb.asm.Opcodes.*; + +import sun.hotspot.WhiteBox; +import uncommontrap.Verifier; + +/* + * @test + * @bug 8030976 8059226 + * @library /testlibrary /compiler/testlibrary /../../test/lib + * @build TestUnstableIfTrap com.oracle.java.testlibrary.* uncommontrap.Verifier + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbatch -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI -XX:+LogCompilation + * -XX:CompileCommand=compileonly,UnstableIfExecutable.test + * -XX:LogFile=always_taken_not_fired.xml + * TestUnstableIfTrap ALWAYS_TAKEN false + * @run main/othervm -Xbatch -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI -XX:+LogCompilation + * -XX:CompileCommand=compileonly,UnstableIfExecutable.test + * -XX:LogFile=always_taken_fired.xml + * TestUnstableIfTrap ALWAYS_TAKEN true + * @run main/othervm -Xbatch -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI -XX:+LogCompilation + * -XX:CompileCommand=compileonly,UnstableIfExecutable.test + * -XX:LogFile=never_taken_not_fired.xml + * TestUnstableIfTrap NEVER_TAKEN false + * @run main/othervm -Xbatch -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:+WhiteBoxAPI -XX:+LogCompilation + * -XX:CompileCommand=compileonly,UnstableIfExecutable.test + * -XX:LogFile=never_taken_fired.xml + * TestUnstableIfTrap NEVER_TAKEN true + * @run main uncommontrap.Verifier always_taken_not_fired.xml + * always_taken_fired.xml + * never_taken_not_fired.xml + * never_taken_fired.xml + */ +public class TestUnstableIfTrap { + private static final WhiteBox WB = WhiteBox.getWhiteBox(); + private static final String CLASS_NAME = "UnstableIfExecutable"; + private static final String METHOD_NAME = "test"; + private static final String FIELD_NAME = "field"; + private static final int ITERATIONS = 1_000_000; + // There is no dependency on particular class file version, so it could be + // set to any version (if you're updating this test for Java 42). + private static final int CLASS_FILE_VERSION = 49; + private static final int MAX_TIER = 4; + // This test aimed to verify that uncommon trap with reason "unstable_if" + // is emitted when method that contain control-flow divergence such that + // one of two branches is never taken (and other one is taken always). + // C2 will made a decision whether or not the branch was ever taken + // depending on method's profile. + // If profile was collected for a few method's invocations, then C2 will not + // trust in branches' probabilities and the tested trap won't be emitted. + // In fact, a method has to be invoked at least 40 time at the day when this + // comment was written (see Parse::dynamic_branch_prediction for an actual + // value). It would be to implementation dependent to use "40" as + // a threshold value in the test, so in order to improve test's robustness + // the threshold value is 1000: if the tested method was compiled by C2 + // before it was invoked 1000 times, then we won't verify that trap was + // emitted and fired. + private static final int MIN_INVOCATIONS_BEFORE_C2_COMPILATION = 1000; + /** + * Description of test case parameters and uncommon trap that will + * be emitted during tested method compilation. + */ + private static enum TestCaseName { + ALWAYS_TAKEN(false, "taken always"), + NEVER_TAKEN(true, "taken never"); + TestCaseName(boolean predicate, String comment) { + this.predicate = predicate; + this.comment = comment; + } + + public final boolean predicate; + public final String name = "unstable_if"; + public final String comment; + } + + public static void main(String args[]) { + if (args.length != 2) { + throw new Error("Expected two arguments: test case name and a " + + "boolean determining if uncommon trap should be fired."); + } + test(TestCaseName.valueOf(args[0]), Boolean.valueOf(args[1])); + } + + private static void test(TestCaseName testCase, boolean shouldBeFired) { + Method testMethod; + Label unstableIfLocation = new Label(); + boolean shouldBeEmitted; + boolean compiledToEarly = false; + + try { + Class testClass = ByteCodeLoader.load(CLASS_NAME, + generateTest(unstableIfLocation)); + testMethod = testClass.getDeclaredMethod(METHOD_NAME, + boolean.class); + for (int i = 0; i < ITERATIONS; i++) { + testMethod.invoke(null, testCase.predicate); + if (i < MIN_INVOCATIONS_BEFORE_C2_COMPILATION + && isMethodCompiledByC2(testMethod)) { + compiledToEarly = true; + // There is no sense in further invocations: we already + // decided to avoid verification. + break; + } + } + // We're checking that trap should be emitted (i.e. it was compiled + // by C2) before the trap is fired, because otherwise the nmethod + // will be deoptimized and isMethodCompiledByC2 will return false. + shouldBeEmitted = isMethodCompiledByC2(testMethod) + && !compiledToEarly; + if (shouldBeFired) { + testMethod.invoke(null, !testCase.predicate); + } + } catch (ReflectiveOperationException e) { + throw new Error("Test case should be generated, loaded and executed" + + " without any issues.", e); + } + + shouldBeFired &= shouldBeEmitted; + + Properties properties = new Properties(); + properties.setProperty(Verifier.VERIFICATION_SHOULD_BE_SKIPPED, + Boolean.toString(compiledToEarly)); + properties.setProperty(Verifier.UNCOMMON_TRAP_SHOULD_EMITTED, + Boolean.toString(shouldBeEmitted)); + properties.setProperty(Verifier.UNCOMMON_TRAP_SHOULD_FIRED, + Boolean.toString(shouldBeFired)); + properties.setProperty(Verifier.UNCOMMON_TRAP_NAME, testCase.name); + properties.setProperty(Verifier.UNCOMMON_TRAP_COMMENT, + testCase.comment); + properties.setProperty(Verifier.UNCOMMON_TRAP_BCI, + Integer.toString(unstableIfLocation.getOffset())); + + properties.list(System.out); + + File f = new File(WB.getStringVMFlag("LogFile") + + Verifier.PROPERTIES_FILE_SUFFIX); + try (FileWriter wr = new FileWriter(f)) { + properties.store(wr, ""); + } catch (IOException e) { + throw new Error("Unable to store test properties.", e); + } + } + + private static boolean isMethodCompiledByC2(Method m) { + boolean isTiered = WB.getBooleanVMFlag("TieredCompilation"); + boolean isMethodCompiled = WB.isMethodCompiled(m); + boolean isMethodCompiledAtMaxTier + = WB.getMethodCompilationLevel(m) == MAX_TIER; + + return Platform.isServer() && isMethodCompiled + && (!isTiered || isMethodCompiledAtMaxTier); + } + + /** + * Generates class with name {@code CLASS_NAME}, which will contain a + * static method {@code METHOD_NAME}: + * + *
{@code
+     * public abstract class UnstableIfExecutable {
+     *   private static int field = 0;
+     *
+     *   public static void test(boolean alwaysTrue) {
+     *     if (alwaysTrue) {
+     *       field++;
+     *     } else {
+     *       field--;
+     *     }
+     *   }
+     * }
+     * }
+ * + * @return generated bytecode. + */ + private static byte[] generateTest(Label unstableIfLocation) { + ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES); + + cw.visit(CLASS_FILE_VERSION, ACC_PUBLIC | ACC_ABSTRACT, CLASS_NAME, + null, "java/lang/Object", null); + + cw.visitField(ACC_PUBLIC | ACC_STATIC | ACC_VOLATILE, FIELD_NAME, + "I", null, Integer.valueOf(0)); + + generateTestMethod(cw, unstableIfLocation); + + return cw.toByteArray(); + } + + private static void generateTestMethod(ClassVisitor cv, + Label unstableIfLocation) { + MethodVisitor mv = cv.visitMethod(ACC_PUBLIC | ACC_STATIC, METHOD_NAME, + "(Z)V", null, null); + mv.visitCode(); + + Label end = new Label(); + Label falseBranch = new Label(); + + // push "field" field's value and 1 to stack + mv.visitFieldInsn(GETSTATIC, CLASS_NAME, FIELD_NAME, "I"); + mv.visitInsn(ICONST_1); + // load argument's value + mv.visitVarInsn(ILOAD, 0); // alwaysTrue + // here is our unstable if + mv.visitLabel(unstableIfLocation); + mv.visitJumpInsn(IFEQ, falseBranch); + // increment on "true" + mv.visitInsn(IADD); + mv.visitJumpInsn(GOTO, end); + // decrement on "false" + mv.visitLabel(falseBranch); + mv.visitInsn(ISUB); + mv.visitLabel(end); + // bye bye + mv.visitInsn(RETURN); + + mv.visitMaxs(0, 0); + mv.visitEnd(); + } +} + From b8506172439246fda42e994f81772e0ededc1325 Mon Sep 17 00:00:00 2001 From: Filipp Zhinkin Date: Wed, 14 Jan 2015 09:54:08 +0300 Subject: [PATCH 21/77] 8049355: compiler/rtm/locking/TestRTMLockingThreshold test may fail if transaction was aborted by interrupt Reviewed-by: kvn, iignatyev --- .../locking/TestRTMDeoptOnLowAbortRatio.java | 9 ++++--- .../rtm/locking/TestRTMLockingThreshold.java | 26 +++++++++---------- 2 files changed, 18 insertions(+), 17 deletions(-) diff --git a/hotspot/test/compiler/rtm/locking/TestRTMDeoptOnLowAbortRatio.java b/hotspot/test/compiler/rtm/locking/TestRTMDeoptOnLowAbortRatio.java index 128ec7bd403..ae39ffb25c0 100644 --- a/hotspot/test/compiler/rtm/locking/TestRTMDeoptOnLowAbortRatio.java +++ b/hotspot/test/compiler/rtm/locking/TestRTMDeoptOnLowAbortRatio.java @@ -53,6 +53,7 @@ import sun.misc.Unsafe; */ public class TestRTMDeoptOnLowAbortRatio extends CommandLineOptionTest { private static final long LOCKING_THRESHOLD = 100L; + private static final long ABORT_THRESHOLD = LOCKING_THRESHOLD / 2L; private TestRTMDeoptOnLowAbortRatio() { super(new AndPredicate(new SupportedCPU(), new SupportedVM())); @@ -77,7 +78,8 @@ public class TestRTMDeoptOnLowAbortRatio extends CommandLineOptionTest { useStackLock), CommandLineOptionTest.prepareNumericFlag("RTMLockingThreshold", TestRTMDeoptOnLowAbortRatio.LOCKING_THRESHOLD), - "-XX:RTMAbortThreshold=1", + CommandLineOptionTest.prepareNumericFlag("RTMAbortThreshold", + TestRTMDeoptOnLowAbortRatio.ABORT_THRESHOLD), "-XX:RTMAbortRatio=100", "-XX:CompileThreshold=1", "-XX:RTMRetryCount=0", @@ -107,7 +109,7 @@ public class TestRTMDeoptOnLowAbortRatio extends CommandLineOptionTest { for (RTMLockingStatistics s : statistics) { if (s.getTotalLocks() - == TestRTMDeoptOnLowAbortRatio.LOCKING_THRESHOLD + 1L) { + == TestRTMDeoptOnLowAbortRatio.LOCKING_THRESHOLD) { Asserts.assertNull(statisticsBeforeDeopt, "Only one abort was expected during test run"); statisticsBeforeDeopt = s; @@ -154,8 +156,7 @@ public class TestRTMDeoptOnLowAbortRatio extends CommandLineOptionTest { } for (int i = 0; i < AbortProvoker.DEFAULT_ITERATIONS; i++) { AbortProvoker.verifyMonitorState(t.monitor, shouldBeInflated); - t.forceAbort( - i == TestRTMDeoptOnLowAbortRatio.LOCKING_THRESHOLD); + t.forceAbort(i >= TestRTMDeoptOnLowAbortRatio.ABORT_THRESHOLD); } } } diff --git a/hotspot/test/compiler/rtm/locking/TestRTMLockingThreshold.java b/hotspot/test/compiler/rtm/locking/TestRTMLockingThreshold.java index 75a94c6d989..66daef96ff5 100644 --- a/hotspot/test/compiler/rtm/locking/TestRTMLockingThreshold.java +++ b/hotspot/test/compiler/rtm/locking/TestRTMLockingThreshold.java @@ -58,7 +58,7 @@ public class TestRTMLockingThreshold extends CommandLineOptionTest { * interrupts, VMM calls, etc. during first lock attempt. * */ - private static final int ABORT_THRESHOLD = 10; + private static final int MIN_ABORT_THRESHOLD = 10; @Override protected void runTestCases() throws Throwable { @@ -75,6 +75,9 @@ public class TestRTMLockingThreshold extends CommandLineOptionTest { boolean useStackLock) throws Throwable { CompilableTest test = new Test(); + int abortThreshold = Math.max(lockingThreshold / 2, + TestRTMLockingThreshold.MIN_ABORT_THRESHOLD); + OutputAnalyzer outputAnalyzer = RTMTestBase.executeRTMTest( test, "-XX:CompileThreshold=1", @@ -84,7 +87,7 @@ public class TestRTMLockingThreshold extends CommandLineOptionTest { "-XX:RTMTotalCountIncrRate=1", "-XX:RTMRetryCount=0", CommandLineOptionTest.prepareNumericFlag("RTMAbortThreshold", - TestRTMLockingThreshold.ABORT_THRESHOLD), + abortThreshold), CommandLineOptionTest.prepareNumericFlag("RTMLockingThreshold", lockingThreshold), "-XX:RTMAbortRatio=100", @@ -103,16 +106,12 @@ public class TestRTMLockingThreshold extends CommandLineOptionTest { + "RTM locking statistics entries."); /** - * We force abort on each odd iteration, so if RTMLockingThreshold==0, - * then we have to make 1 call without abort to avoid rtm state - * transition to NoRTM (otherwise actual abort ratio will be 100%), - * and after that make 1 call with abort to force deoptimization. - * This leads us to two locks for threshold 0. - * For other threshold values we have to make RTMLockingThreshold + 1 - * locks if locking threshold is even, or + 0 if odd. + * If RTMLockingThreshold==0, then we have to make at least 1 call. */ - long expectedValue = lockingThreshold + - (lockingThreshold == 0L ? 2L : lockingThreshold % 2L); + long expectedValue = lockingThreshold; + if (expectedValue == 0) { + expectedValue++; + } RTMLockingStatistics statBeforeDeopt = null; for (RTMLockingStatistics s : statistics) { @@ -159,15 +158,16 @@ public class TestRTMLockingThreshold extends CommandLineOptionTest { * Test <inflate monitor> */ public static void main(String args[]) throws Throwable { - Asserts.assertGTE(args.length, 1, "One argument required."); + Asserts.assertGTE(args.length, 2, "Two arguments required."); Test t = new Test(); boolean shouldBeInflated = Boolean.valueOf(args[0]); + int lockingThreshold = Integer.valueOf(args[1]); if (shouldBeInflated) { AbortProvoker.inflateMonitor(t.monitor); } for (int i = 0; i < Test.TOTAL_ITERATIONS; i++) { AbortProvoker.verifyMonitorState(t.monitor, shouldBeInflated); - t.lock(i % 2 == 1); + t.lock(i >= lockingThreshold / 2); } } } From 5bd69d7ad42607440327b788d64033bbd7f49f49 Mon Sep 17 00:00:00 2001 From: Pavel Chistyakov Date: Mon, 12 Jan 2015 18:58:49 +0300 Subject: [PATCH 22/77] 8059551: JEP-JDK-8043304: Test task: stress tests Reviewed-by: drchase, kvn, iignatyev --- .../src/share/vm/compiler/compileBroker.cpp | 2 +- .../stress/CodeCacheStressRunner.java | 52 +++++++ .../compiler/codecache/stress/Helper.java | 141 ++++++++++++++++++ .../stress/OverloadCompileQueueTest.java | 111 ++++++++++++++ .../stress/RandomAllocationTest.java | 71 +++++++++ .../stress/UnexpectedDeoptimizationTest.java | 52 +++++++ 6 files changed, 428 insertions(+), 1 deletion(-) create mode 100644 hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java create mode 100644 hotspot/test/compiler/codecache/stress/Helper.java create mode 100644 hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java create mode 100644 hotspot/test/compiler/codecache/stress/RandomAllocationTest.java create mode 100644 hotspot/test/compiler/codecache/stress/UnexpectedDeoptimizationTest.java diff --git a/hotspot/src/share/vm/compiler/compileBroker.cpp b/hotspot/src/share/vm/compiler/compileBroker.cpp index 2b22da1f705..4f0def89a20 100644 --- a/hotspot/src/share/vm/compiler/compileBroker.cpp +++ b/hotspot/src/share/vm/compiler/compileBroker.cpp @@ -252,7 +252,7 @@ CompileTask* CompileTask::allocate() { } else { task = new CompileTask(); DEBUG_ONLY(_num_allocated_tasks++;) - assert (_num_allocated_tasks < 10000, "Leaking compilation tasks?"); + assert (WhiteBoxAPI || _num_allocated_tasks < 10000, "Leaking compilation tasks?"); task->set_next(NULL); task->set_is_free(true); } diff --git a/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java b/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java new file mode 100644 index 00000000000..1a7ae636097 --- /dev/null +++ b/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2014, 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 com.oracle.java.testlibrary.TimeLimitedRunner; +import com.oracle.java.testlibrary.Utils; + +public class CodeCacheStressRunner { + private final Runnable action; + public CodeCacheStressRunner(Runnable action) { + this.action = action; + } + + protected final void runTest() { + Helper.startInfiniteLoopThread(action); + try { + // adjust timeout and substract vm init and exit time + long timeout = Utils.adjustTimeout(Utils.DEFAULT_TEST_TIMEOUT); + timeout *= 0.9; + new TimeLimitedRunner(timeout, 2.0d, this::test).call(); + } catch (Exception e) { + throw new Error("Exception occurred during test execution", e); + } + } + + private boolean test() { + Helper.TestCase obj = Helper.TestCase.get(); + Helper.callMethod(obj.getCallable(), obj.expectedValue()); + return true; + } + +} diff --git a/hotspot/test/compiler/codecache/stress/Helper.java b/hotspot/test/compiler/codecache/stress/Helper.java new file mode 100644 index 00000000000..0a4a3791b6c --- /dev/null +++ b/hotspot/test/compiler/codecache/stress/Helper.java @@ -0,0 +1,141 @@ +/* + * Copyright (c) 2014, 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.io.BufferedInputStream; +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.util.concurrent.Callable; +import java.util.Random; + +import com.oracle.java.testlibrary.Asserts; +import com.oracle.java.testlibrary.ByteCodeLoader; +import com.oracle.java.testlibrary.InfiniteLoop; +import com.oracle.java.testlibrary.Utils; +import sun.hotspot.WhiteBox; + +public final class Helper { + public static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox(); + public static final Random RNG = Utils.getRandomInstance(); + + private static final long THRESHOLD = WHITE_BOX.getIntxVMFlag("CompileThreshold"); + private static final String TEST_CASE_IMPL_CLASS_NAME = "Helper$TestCaseImpl"; + private static byte[] CLASS_DATA; + static { + try { + CLASS_DATA = loadClassData(TEST_CASE_IMPL_CLASS_NAME); + } catch (IOException e) { + throw new Error("TESTBUG: cannot load class byte code", e); + } + } + + private Helper() { + } + + public static void startInfiniteLoopThread(Runnable action) { + startInfiniteLoopThread(action, 0L); + } + + public static void startInfiniteLoopThread(Runnable action, long millis) { + Thread t = new Thread(new InfiniteLoop(action, millis)); + t.setDaemon(true); + t.start(); + } + + public static int callMethod(Callable callable, int expected) { + int result = 0; + for (int i = 0; i < THRESHOLD; ++i) { + try { + result = callable.call(); + } catch (Exception e) { + throw new AssertionError( + "Exception occurred during test method execution", e); + } + Asserts.assertEQ(result, expected, "Method returns unexpected value"); + } + return result; + } + + private static byte[] loadClassData(String name) throws IOException { + try (BufferedInputStream in = new BufferedInputStream( + ClassLoader.getSystemResourceAsStream(name.replace(".", "/") + + ".class"))) { + ByteArrayOutputStream result = new ByteArrayOutputStream(); + byte[] buffer = new byte[1024]; + int read; + while ((read = in.read(buffer)) != -1) { + result.write(buffer, 0, read); + } + return result.toByteArray(); + } + } + + public interface TestCase { + + public static TestCase get() { + try { + Class clazz = ByteCodeLoader.load( + TEST_CASE_IMPL_CLASS_NAME, CLASS_DATA); + return (TestCase) clazz.newInstance(); + } catch (ReflectiveOperationException e) { + throw new Error(String.format( + "TESTBUG: error while creating %s instance from reloaded class", + TEST_CASE_IMPL_CLASS_NAME), e); + } + } + + Callable getCallable(); + int method(); + int expectedValue(); + } + + public static class TestCaseImpl implements TestCase { + private static final int RETURN_VALUE = 42; + private static final int RECURSION_DEPTH = 10; + private volatile int i; + + @Override + public Callable getCallable() { + return () -> { + i = 0; + return method(); + }; + } + + @Override + public int method() { + ++i; + int result = RETURN_VALUE; + if (i < RECURSION_DEPTH) { + return result + method(); + } + return result; + } + + @Override + public int expectedValue() { + return RETURN_VALUE * RECURSION_DEPTH; + } + } + +} diff --git a/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java b/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java new file mode 100644 index 00000000000..5f52ef679ea --- /dev/null +++ b/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java @@ -0,0 +1,111 @@ +/* + * Copyright (c) 2014, 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.reflect.Method; +import java.util.stream.IntStream; + +import com.oracle.java.testlibrary.Platform; + +/* + * @test OverloadCompileQueueTest + * @library /testlibrary /../../test/lib + * @build OverloadCompileQueueTest + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:-SegmentedCodeCache OverloadCompileQueueTest + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:+SegmentedCodeCache OverloadCompileQueueTest + * @summary stressing code cache by overloading compile queues + */ +public class OverloadCompileQueueTest implements Runnable { + private static final int MAX_SLEEP = 10000; + private static final String METHOD_TO_ENQUEUE = "method"; + private static final int LEVEL_SIMPLE = 1; + private static final int LEVEL_FULL_OPTIMIZATION = 4; + private static final boolean INTERPRETED + = System.getProperty("java.vm.info").startsWith("interpreted "); + private static final boolean TIERED_COMPILATION + = Helper.WHITE_BOX.getBooleanVMFlag("TieredCompilation"); + private static final int TIERED_STOP_AT_LEVEL + = Helper.WHITE_BOX.getIntxVMFlag("TieredStopAtLevel").intValue(); + private static final int[] AVAILABLE_LEVELS; + static { + if (TIERED_COMPILATION) { + AVAILABLE_LEVELS = IntStream + .rangeClosed(LEVEL_SIMPLE, TIERED_STOP_AT_LEVEL) + .toArray(); + } else if (Platform.isServer()) { + AVAILABLE_LEVELS = new int[] { LEVEL_FULL_OPTIMIZATION }; + } else if (Platform.isClient() || Platform.isMinimal()) { + AVAILABLE_LEVELS = new int[] { LEVEL_SIMPLE }; + } else { + throw new Error(String.format( + "TESTBUG: unknown VM: %s", System.getProperty("java.vm.name"))); + } + } + + public static void main(String[] args) { + if (INTERPRETED) { + System.err.println("Test isn't applicable for interpreter. Skip test."); + return; + } + new CodeCacheStressRunner(new OverloadCompileQueueTest()).runTest(); + } + + public OverloadCompileQueueTest() { + Helper.startInfiniteLoopThread(this::lockUnlock); + } + + @Override + public void run() { + Helper.TestCase obj = Helper.TestCase.get(); + Class clazz = obj.getClass(); + Method mEnqueue; + try { + mEnqueue = clazz.getMethod(METHOD_TO_ENQUEUE); + } catch (NoSuchMethodException | SecurityException e) { + throw new Error(String.format( + "TESTBUG: cannot get method '%s' of class %s", + METHOD_TO_ENQUEUE, clazz.getName()), e); + } + for (int compLevel : AVAILABLE_LEVELS) { + Helper.WHITE_BOX.enqueueMethodForCompilation(mEnqueue, compLevel); + } + } + + private void lockUnlock() { + try { + Helper.WHITE_BOX.lockCompilation(); + Thread.sleep(Helper.RNG.nextInt(MAX_SLEEP)); + } catch (InterruptedException e) { + throw new Error("TESTBUG: lockUnlocker thread was unexpectedly interrupted", e); + } finally { + Helper.WHITE_BOX.unlockCompilation(); + } + } + +} diff --git a/hotspot/test/compiler/codecache/stress/RandomAllocationTest.java b/hotspot/test/compiler/codecache/stress/RandomAllocationTest.java new file mode 100644 index 00000000000..466bcc4b895 --- /dev/null +++ b/hotspot/test/compiler/codecache/stress/RandomAllocationTest.java @@ -0,0 +1,71 @@ +/* + * Copyright (c) 2014, 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.util.ArrayList; + +import sun.hotspot.code.BlobType; + +/* + * @test RandomAllocationTest + * @library /testlibrary /../../test/lib + * @build RandomAllocationTest + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:-SegmentedCodeCache RandomAllocationTest + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:+SegmentedCodeCache RandomAllocationTest + * @summary stressing code cache by allocating randomly sized "dummy" code blobs + */ +public class RandomAllocationTest implements Runnable { + private static final long CODE_CACHE_SIZE + = Helper.WHITE_BOX.getUintxVMFlag("ReservedCodeCacheSize"); + private static final int MAX_BLOB_SIZE = (int) (CODE_CACHE_SIZE >> 7); + private static final BlobType[] BLOB_TYPES + = BlobType.getAvailable().toArray(new BlobType[0]); + + public static void main(String[] args) { + new CodeCacheStressRunner(new RandomAllocationTest()).runTest(); + } + + private final ArrayList blobs = new ArrayList<>(); + @Override + public void run() { + boolean allocate = blobs.isEmpty() || Helper.RNG.nextBoolean(); + if (allocate) { + int type = Helper.RNG.nextInt(BLOB_TYPES.length); + long addr = Helper.WHITE_BOX.allocateCodeBlob( + Helper.RNG.nextInt(MAX_BLOB_SIZE), BLOB_TYPES[type].id); + if (addr != 0) { + blobs.add(addr); + } + } else { + int index = Helper.RNG.nextInt(blobs.size()); + Helper.WHITE_BOX.freeCodeBlob(blobs.remove(index)); + } + } + +} diff --git a/hotspot/test/compiler/codecache/stress/UnexpectedDeoptimizationTest.java b/hotspot/test/compiler/codecache/stress/UnexpectedDeoptimizationTest.java new file mode 100644 index 00000000000..f8ff8e5deb4 --- /dev/null +++ b/hotspot/test/compiler/codecache/stress/UnexpectedDeoptimizationTest.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2014, 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 UnexpectedDeoptimizationTest + * @library /testlibrary /../../test/lib + * @build UnexpectedDeoptimizationTest + * @run main ClassFileInstaller sun.hotspot.WhiteBox + * sun.hotspot.WhiteBox$WhiteBoxPermission + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:-SegmentedCodeCache -XX:-DeoptimizeRandom + * UnexpectedDeoptimizationTest + * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions + * -XX:CompileCommand=dontinline,Helper$TestCase::method + * -XX:+WhiteBoxAPI -XX:+SegmentedCodeCache -XX:-DeoptimizeRandom + * UnexpectedDeoptimizationTest + * @summary stressing code cache by forcing unexpected deoptimizations + */ +public class UnexpectedDeoptimizationTest implements Runnable { + + public static void main(String[] args) { + new CodeCacheStressRunner(new UnexpectedDeoptimizationTest()).runTest(); + } + + @Override + public void run() { + Helper.WHITE_BOX.deoptimizeFrames(Helper.RNG.nextBoolean()); + } + +} From 9000f8c3c7e1d1206dea3facdbf66ff2a57d3ffa Mon Sep 17 00:00:00 2001 From: Calvin Cheung Date: Mon, 12 Jan 2015 11:37:58 -0800 Subject: [PATCH 23/77] 8067187: -XX:MaxMetaspaceSize=20m -Xshare:dump caused JVM to crash Added check for the MaxMetaspaceSize. If it is less than the estimated required size, print an error and exit. Reviewed-by: jiangli, mseledtsov, stefank --- hotspot/src/share/vm/memory/metaspace.cpp | 13 +++++- hotspot/src/share/vm/utilities/debug.cpp | 10 ++++- hotspot/src/share/vm/utilities/debug.hpp | 4 +- .../SharedArchiveFile/MaxMetaspaceSize.java | 41 +++++++++++++++++++ 4 files changed, 65 insertions(+), 3 deletions(-) create mode 100644 hotspot/test/runtime/SharedArchiveFile/MaxMetaspaceSize.java diff --git a/hotspot/src/share/vm/memory/metaspace.cpp b/hotspot/src/share/vm/memory/metaspace.cpp index de7ea1abee1..88d49fc78bf 100644 --- a/hotspot/src/share/vm/memory/metaspace.cpp +++ b/hotspot/src/share/vm/memory/metaspace.cpp @@ -3170,7 +3170,9 @@ void Metaspace::global_initialize() { } // the min_misc_data_size and min_misc_code_size estimates are based on - // MetaspaceShared::generate_vtable_methods() + // MetaspaceShared::generate_vtable_methods(). + // The minimum size only accounts for the vtable methods. Any size less than the + // minimum required size would cause vm crash when allocating the vtable methods. uint min_misc_data_size = align_size_up( MetaspaceShared::num_virtuals * MetaspaceShared::vtbl_list_size * sizeof(void*), max_alignment); @@ -3336,6 +3338,10 @@ void Metaspace::initialize(Mutex* lock, MetaspaceType type) { Metachunk* new_chunk = get_initialization_chunk(NonClassType, word_size, vsm()->medium_chunk_bunch()); + // For dumping shared archive, report error if allocation has failed. + if (DumpSharedSpaces && new_chunk == NULL) { + report_insufficient_metaspace(MetaspaceAux::committed_bytes() + word_size * BytesPerWord); + } assert(!DumpSharedSpaces || new_chunk != NULL, "should have enough space for both chunks"); if (new_chunk != NULL) { // Add to this manager's list of chunks in use and current_chunk(). @@ -3349,6 +3355,11 @@ void Metaspace::initialize(Mutex* lock, MetaspaceType type) { class_vsm()->medium_chunk_bunch()); if (class_chunk != NULL) { class_vsm()->add_chunk(class_chunk, true); + } else { + // For dumping shared archive, report error if allocation has failed. + if (DumpSharedSpaces) { + report_insufficient_metaspace(MetaspaceAux::committed_bytes() + class_word_size * BytesPerWord); + } } } diff --git a/hotspot/src/share/vm/utilities/debug.cpp b/hotspot/src/share/vm/utilities/debug.cpp index adacdc4233f..74643724632 100644 --- a/hotspot/src/share/vm/utilities/debug.cpp +++ b/hotspot/src/share/vm/utilities/debug.cpp @@ -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 @@ -273,6 +273,14 @@ void report_out_of_shared_space(SharedSpaceType shared_space) { exit(2); } +void report_insufficient_metaspace(size_t required_size) { + warning("\nThe MaxMetaspaceSize of " UINTX_FORMAT " bytes is not large enough.\n" + "Either don't specify the -XX:MaxMetaspaceSize=\n" + "or increase the size to at least " SIZE_FORMAT ".\n", + MaxMetaspaceSize, required_size); + exit(2); +} + void report_java_out_of_memory(const char* message) { static jint out_of_memory_reported = 0; diff --git a/hotspot/src/share/vm/utilities/debug.hpp b/hotspot/src/share/vm/utilities/debug.hpp index 7995e6c085d..9d980d59ca0 100644 --- a/hotspot/src/share/vm/utilities/debug.hpp +++ b/hotspot/src/share/vm/utilities/debug.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 @@ -251,6 +251,8 @@ enum SharedSpaceType { void report_out_of_shared_space(SharedSpaceType space_type); +void report_insufficient_metaspace(size_t required_size); + // out of memory reporting void report_java_out_of_memory(const char* message); diff --git a/hotspot/test/runtime/SharedArchiveFile/MaxMetaspaceSize.java b/hotspot/test/runtime/SharedArchiveFile/MaxMetaspaceSize.java new file mode 100644 index 00000000000..9d8c4984377 --- /dev/null +++ b/hotspot/test/runtime/SharedArchiveFile/MaxMetaspaceSize.java @@ -0,0 +1,41 @@ +/* + * 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 + * @bug 8067187 + * @summary Testing CDS dumping with the -XX:MaxMetaspaceSize= option + * @library /testlibrary + */ + +import com.oracle.java.testlibrary.*; + +public class MaxMetaspaceSize { + public static void main(String[] args) throws Exception { + ProcessBuilder pb = ProcessTools.createJavaProcessBuilder( + "-XX:MaxMetaspaceSize=20m", "-Xshare:dump"); + OutputAnalyzer output = new OutputAnalyzer(pb.start()); + output.shouldContain("is not large enough.\nEither don't specify the -XX:MaxMetaspaceSize=\nor increase the size to at least"); + output.shouldHaveExitValue(2); + } +} From 4742c185a7650ece8d148079292b43da3f6edf45 Mon Sep 17 00:00:00 2001 From: Nils Eliasson Date: Tue, 13 Jan 2015 12:30:26 +0100 Subject: [PATCH 24/77] 8027829: CompileCommand does not accept all JLS-conformant class/method names Fix parsing and updated comments Reviewed-by: kvn, vlivanov --- .../src/share/vm/compiler/compilerOracle.cpp | 138 +++++++++++++----- .../src/share/vm/compiler/compilerOracle.hpp | 1 + .../oracle/CheckCompileCommandOption.java | 4 +- .../CompilerConfigFileWarning.java | 2 +- 4 files changed, 103 insertions(+), 42 deletions(-) diff --git a/hotspot/src/share/vm/compiler/compilerOracle.cpp b/hotspot/src/share/vm/compiler/compilerOracle.cpp index b733305666b..a6351be1783 100644 --- a/hotspot/src/share/vm/compiler/compilerOracle.cpp +++ b/hotspot/src/share/vm/compiler/compilerOracle.cpp @@ -105,7 +105,6 @@ class MethodMatcher : public CHeapObj { tty->print("."); print_symbol(method_name(), _method_mode); if (signature() != NULL) { - tty->print(" "); signature()->print_symbol_on(tty); } } @@ -467,43 +466,85 @@ static OracleCommand parse_command_name(const char * line, int* bytes_read) { return UnknownCommand; } - static void usage() { - tty->print_cr(" CompileCommand and the CompilerOracle allows simple control over"); - tty->print_cr(" what's allowed to be compiled. The standard supported directives"); - tty->print_cr(" are exclude and compileonly. The exclude directive stops a method"); - tty->print_cr(" from being compiled and compileonly excludes all methods except for"); - tty->print_cr(" the ones mentioned by compileonly directives. The basic form of"); - tty->print_cr(" all commands is a command name followed by the name of the method"); - tty->print_cr(" in one of two forms: the standard class file format as in"); - tty->print_cr(" class/name.methodName or the PrintCompilation format"); - tty->print_cr(" class.name::methodName. The method name can optionally be followed"); - tty->print_cr(" by a space then the signature of the method in the class file"); - tty->print_cr(" format. Otherwise the directive applies to all methods with the"); - tty->print_cr(" same name and class regardless of signature. Leading and trailing"); - tty->print_cr(" *'s in the class and/or method name allows a small amount of"); - tty->print_cr(" wildcarding. "); tty->cr(); - tty->print_cr(" Examples:"); + tty->print_cr("The CompileCommand option enables the user of the JVM to control specific"); + tty->print_cr("behavior of the dynamic compilers. Many commands require a pattern that defines"); + tty->print_cr("the set of methods the command shall be applied to. The CompileCommand"); + tty->print_cr("option provides the following commands:"); tty->cr(); - tty->print_cr(" exclude java/lang/StringBuffer.append"); - tty->print_cr(" compileonly java/lang/StringBuffer.toString ()Ljava/lang/String;"); - tty->print_cr(" exclude java/lang/String*.*"); - tty->print_cr(" exclude *.toString"); -} + tty->print_cr(" break, - debug breakpoint in compiler and in generated code"); + tty->print_cr(" print, - print assembly"); + tty->print_cr(" exclude, - don't compile or inline"); + tty->print_cr(" inline, - always inline"); + tty->print_cr(" dontinline, - don't inline"); + tty->print_cr(" compileonly, - compile only"); + tty->print_cr(" log, - log compilation"); + tty->print_cr(" option,,