From e13c212339d6af9f5698efdc090a1e9690c94bf7 Mon Sep 17 00:00:00 2001 From: Severin Gehwolf Date: Wed, 7 Jan 2015 16:43:04 +0100 Subject: [PATCH 01/94] 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/94] 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 f3c99841f9afa3b2cc43012e200791d42ef73f41 Mon Sep 17 00:00:00 2001 From: Jiangli Zhou Date: Wed, 14 Jan 2015 16:35:00 -0500 Subject: [PATCH 03/94] 8067982: Some jcmd /gc/heap_dump tests failed: hprof output contains warning or error Include shared symbols in SymbolTable::symbols_do(SymbolClosure). Reviewed-by: minqi, farvidsson, coleenp --- .../share/vm/classfile/compactHashtable.cpp | 26 ++++++++++++++++++- .../share/vm/classfile/compactHashtable.hpp | 5 +++- .../src/share/vm/classfile/symbolTable.cpp | 6 ++++- 3 files changed, 34 insertions(+), 3 deletions(-) diff --git a/hotspot/src/share/vm/classfile/compactHashtable.cpp b/hotspot/src/share/vm/classfile/compactHashtable.cpp index 157b1681425..bd197554293 100644 --- a/hotspot/src/share/vm/classfile/compactHashtable.cpp +++ b/hotspot/src/share/vm/classfile/compactHashtable.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 @@ -221,6 +221,30 @@ template const char* CompactHashtable::init(const char* return (const char*)end; } +template void CompactHashtable::symbols_do(SymbolClosure *cl) { + assert(!DumpSharedSpaces, "run-time only"); + for (juint i = 0; i < _bucket_count; i ++) { + juint bucket_info = _buckets[i]; + juint bucket_offset = BUCKET_OFFSET(bucket_info); + int bucket_type = BUCKET_TYPE(bucket_info); + juint* bucket = _buckets + bucket_offset; + juint* bucket_end = _buckets; + + Symbol* sym; + if (bucket_type == COMPACT_BUCKET_TYPE) { + sym = (Symbol*)((void*)(_base_address + bucket[0])); + cl->do_symbol(&sym); + } else { + bucket_end += BUCKET_OFFSET(_buckets[i + 1]); + while (bucket < bucket_end) { + sym = (Symbol*)((void*)(_base_address + bucket[1])); + cl->do_symbol(&sym); + bucket += 2; + } + } + } +} + // Explicitly instantiate these types template class CompactHashtable; diff --git a/hotspot/src/share/vm/classfile/compactHashtable.hpp b/hotspot/src/share/vm/classfile/compactHashtable.hpp index ac01fa7114b..3e32fc07df9 100644 --- a/hotspot/src/share/vm/classfile/compactHashtable.hpp +++ b/hotspot/src/share/vm/classfile/compactHashtable.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 @@ -249,6 +249,9 @@ public: } return NULL; } + + // iterate over symbols + void symbols_do(SymbolClosure *cl); }; //////////////////////////////////////////////////////////////////////// diff --git a/hotspot/src/share/vm/classfile/symbolTable.cpp b/hotspot/src/share/vm/classfile/symbolTable.cpp index 4df696f74c7..3d08364be82 100644 --- a/hotspot/src/share/vm/classfile/symbolTable.cpp +++ b/hotspot/src/share/vm/classfile/symbolTable.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 @@ -82,6 +82,10 @@ void SymbolTable::initialize_symbols(int arena_alloc_size) { // Call function for all symbols in the symbol table. void SymbolTable::symbols_do(SymbolClosure *cl) { + // all symbols from shared table + _shared_table.symbols_do(cl); + + // all symbols from the dynamic table const int n = the_table()->table_size(); for (int i = 0; i < n; i++) { for (HashtableEntry* p = the_table()->bucket(i); From dd5f249290582bed676ec952907726e65562449e Mon Sep 17 00:00:00 2001 From: Goetz Lindenmaier Date: Wed, 7 Jan 2015 08:37:49 +0100 Subject: [PATCH 04/94] 8068503: ppc64: Encode/Decode nodes for disjoint cOops mode Reviewed-by: simonis --- hotspot/src/cpu/ppc/vm/macroAssembler_ppc.hpp | 11 +- .../cpu/ppc/vm/macroAssembler_ppc.inline.hpp | 41 +++-- hotspot/src/cpu/ppc/vm/methodHandles_ppc.cpp | 38 ++-- hotspot/src/cpu/ppc/vm/ppc.ad | 167 +++++++++++++++--- 4 files changed, 192 insertions(+), 65 deletions(-) diff --git a/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.hpp b/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.hpp index 5c118981415..46216782c97 100644 --- a/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.hpp +++ b/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.hpp @@ -567,16 +567,21 @@ class MacroAssembler: public Assembler { inline void load_with_trap_null_check(Register d, int si16, Register s1); // Load heap oop and decompress. Loaded oop may not be null. - inline void load_heap_oop_not_null(Register d, RegisterOrConstant offs, Register s1 = noreg); + // Specify tmp to save one cycle. + inline void load_heap_oop_not_null(Register d, RegisterOrConstant offs, Register s1 = noreg, + Register tmp = noreg); + // Store heap oop and decompress. Decompressed oop may not be null. + // Specify tmp register if d should not be changed. inline void store_heap_oop_not_null(Register d, RegisterOrConstant offs, Register s1, - /*specify if d must stay uncompressed*/ Register tmp = noreg); + Register tmp = noreg); // Null allowed. inline void load_heap_oop(Register d, RegisterOrConstant offs, Register s1 = noreg); // Encode/decode heap oop. Oop may not be null, else en/decoding goes wrong. + // src == d allowed. inline Register encode_heap_oop_not_null(Register d, Register src = noreg); - inline void decode_heap_oop_not_null(Register d); + inline Register decode_heap_oop_not_null(Register d, Register src = noreg); // Null allowed. inline void decode_heap_oop(Register d); diff --git a/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.inline.hpp b/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.inline.hpp index 84485d4f6af..f5d19dff066 100644 --- a/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.inline.hpp +++ b/hotspot/src/cpu/ppc/vm/macroAssembler_ppc.inline.hpp @@ -311,11 +311,14 @@ inline void MacroAssembler::load_with_trap_null_check(Register d, int si16, Regi ld(d, si16, s1); } -inline void MacroAssembler::load_heap_oop_not_null(Register d, RegisterOrConstant offs, Register s1) { +inline void MacroAssembler::load_heap_oop_not_null(Register d, RegisterOrConstant offs, Register s1, Register tmp) { if (UseCompressedOops) { - lwz(d, offs, s1); + // In disjoint mode decoding can save a cycle if src != dst. + Register narrowOop = (tmp != noreg && Universe::narrow_oop_base_disjoint()) ? tmp : d; + lwz(narrowOop, offs, s1); // Attention: no null check here! - decode_heap_oop_not_null(d); + Register res = decode_heap_oop_not_null(d, narrowOop); + assert(res == d, "caller will not consume loaded value"); } else { ld(d, offs, s1); } @@ -340,26 +343,36 @@ inline void MacroAssembler::load_heap_oop(Register d, RegisterOrConstant offs, R } inline Register MacroAssembler::encode_heap_oop_not_null(Register d, Register src) { - Register current = (src!=noreg) ? src : d; // Compressed oop is in d if no src provided. - if (Universe::narrow_oop_base() != NULL) { + Register current = (src != noreg) ? src : d; // Oop to be compressed is in d if no src provided. + if (Universe::narrow_oop_base_overlaps()) { sub(d, current, R30); current = d; } if (Universe::narrow_oop_shift() != 0) { - srdi(d, current, LogMinObjAlignmentInBytes); + rldicl(d, current, 64-Universe::narrow_oop_shift(), 32); // Clears the upper bits. current = d; } return current; // Encoded oop is in this register. } -inline void MacroAssembler::decode_heap_oop_not_null(Register d) { +inline Register MacroAssembler::decode_heap_oop_not_null(Register d, Register src) { + if (Universe::narrow_oop_base_disjoint() && src != noreg && src != d && + Universe::narrow_oop_shift() != 0) { + mr(d, R30); + rldimi(d, src, Universe::narrow_oop_shift(), 32-Universe::narrow_oop_shift()); + return d; + } + + Register current = (src != noreg) ? src : d; // Compressed oop is in d if no src provided. if (Universe::narrow_oop_shift() != 0) { - assert (LogMinObjAlignmentInBytes == Universe::narrow_oop_shift(), "decode alg wrong"); - sldi(d, d, LogMinObjAlignmentInBytes); + sldi(d, current, Universe::narrow_oop_shift()); + current = d; } if (Universe::narrow_oop_base() != NULL) { - add(d, d, R30); + add(d, current, R30); + current = d; } + return current; // Decoded oop is in this register. } inline void MacroAssembler::decode_heap_oop(Register d) { @@ -368,13 +381,7 @@ inline void MacroAssembler::decode_heap_oop(Register d) { cmpwi(CCR0, d, 0); beq(CCR0, isNull); } - if (Universe::narrow_oop_shift() != 0) { - assert (LogMinObjAlignmentInBytes == Universe::narrow_oop_shift(), "decode alg wrong"); - sldi(d, d, LogMinObjAlignmentInBytes); - } - if (Universe::narrow_oop_base() != NULL) { - add(d, d, R30); - } + decode_heap_oop_not_null(d); bind(isNull); } diff --git a/hotspot/src/cpu/ppc/vm/methodHandles_ppc.cpp b/hotspot/src/cpu/ppc/vm/methodHandles_ppc.cpp index 89c0344a5fe..88e0dec0709 100644 --- a/hotspot/src/cpu/ppc/vm/methodHandles_ppc.cpp +++ b/hotspot/src/cpu/ppc/vm/methodHandles_ppc.cpp @@ -172,15 +172,15 @@ void MethodHandles::jump_to_lambda_form(MacroAssembler* _masm, // Load the invoker, as MH -> MH.form -> LF.vmentry __ verify_oop(recv); - __ load_heap_oop_not_null(method_temp, NONZERO(java_lang_invoke_MethodHandle::form_offset_in_bytes()), recv); + __ load_heap_oop_not_null(method_temp, NONZERO(java_lang_invoke_MethodHandle::form_offset_in_bytes()), recv, temp2); __ verify_oop(method_temp); - __ load_heap_oop_not_null(method_temp, NONZERO(java_lang_invoke_LambdaForm::vmentry_offset_in_bytes()), method_temp); + __ load_heap_oop_not_null(method_temp, NONZERO(java_lang_invoke_LambdaForm::vmentry_offset_in_bytes()), method_temp, temp2); __ verify_oop(method_temp); - // the following assumes that a Method* is normally compressed in the vmtarget field: + // The following assumes that a Method* is normally compressed in the vmtarget field: __ ld(method_temp, NONZERO(java_lang_invoke_MemberName::vmtarget_offset_in_bytes()), method_temp); if (VerifyMethodHandles && !for_compiler_entry) { - // make sure recv is already on stack + // Make sure recv is already on stack. __ ld(temp2, in_bytes(Method::const_offset()), method_temp); __ load_sized_value(temp2, in_bytes(ConstMethod::size_of_parameters_offset()), temp2, sizeof(u2), /*is_signed*/ false); @@ -259,8 +259,9 @@ address MethodHandles::generate_method_handle_interpreter_entry(MacroAssembler* } if (TraceMethodHandles) { - if (tmp_mh != noreg) + if (tmp_mh != noreg) { __ mr(R23_method_handle, tmp_mh); // make stub happy + } trace_method_handle_interpreter_entry(_masm, iid); } @@ -332,7 +333,7 @@ void MethodHandles::generate_method_handle_dispatch(MacroAssembler* _masm, if (VerifyMethodHandles && iid != vmIntrinsics::_linkToInterface) { Label L_ok; Register temp2_defc = temp2; - __ load_heap_oop_not_null(temp2_defc, NONZERO(java_lang_invoke_MemberName::clazz_offset_in_bytes()), member_reg); + __ load_heap_oop_not_null(temp2_defc, NONZERO(java_lang_invoke_MemberName::clazz_offset_in_bytes()), member_reg, temp3); load_klass_from_Class(_masm, temp2_defc, temp3, temp4); __ verify_klass_ptr(temp2_defc); __ check_klass_subtype(temp1_recv_klass, temp2_defc, temp3, temp4, L_ok); @@ -407,7 +408,7 @@ void MethodHandles::generate_method_handle_dispatch(MacroAssembler* _masm, } Register temp2_intf = temp2; - __ load_heap_oop_not_null(temp2_intf, NONZERO(java_lang_invoke_MemberName::clazz_offset_in_bytes()), member_reg); + __ load_heap_oop_not_null(temp2_intf, NONZERO(java_lang_invoke_MemberName::clazz_offset_in_bytes()), member_reg, temp3); load_klass_from_Class(_masm, temp2_intf, temp3, temp4); __ verify_klass_ptr(temp2_intf); @@ -464,7 +465,7 @@ void trace_method_handle_stub(const char* adaptername, strstr(adaptername, "linkTo") == NULL); // static linkers don't have MH const char* mh_reg_name = has_mh ? "R23_method_handle" : "G23"; tty->print_cr("MH %s %s="INTPTR_FORMAT " sp=" INTPTR_FORMAT, - adaptername, mh_reg_name, (intptr_t) mh, (intptr_t) entry_sp); + adaptername, mh_reg_name, (intptr_t) mh, entry_sp); if (Verbose) { tty->print_cr("Registers:"); @@ -535,23 +536,22 @@ void MethodHandles::trace_method_handle(MacroAssembler* _masm, const char* adapt BLOCK_COMMENT("trace_method_handle {"); - int nbytes_save = 10 * 8; // 10 volatile gprs - __ save_LR_CR(R0); - __ mr(R0, R1_SP); // saved_sp - assert(Assembler::is_simm(-nbytes_save, 16), "Overwriting R0"); - // Push_frame_reg_args only uses R0 if nbytes_save is wider than 16 bit. - __ push_frame_reg_args(nbytes_save, R0); - __ save_volatile_gprs(R1_SP, frame::abi_reg_args_size); // Except R0. + const Register tmp = R11; // Will be preserved. + const int nbytes_save = 11*8; // volatile gprs except R0 + __ save_volatile_gprs(R1_SP, -nbytes_save); // except R0 + __ save_LR_CR(tmp); // save in old frame - __ load_const(R3_ARG1, (address)adaptername); + __ mr(R5_ARG3, R1_SP); // saved_sp + __ push_frame_reg_args(nbytes_save, tmp); + + __ load_const_optimized(R3_ARG1, (address)adaptername, tmp); __ mr(R4_ARG2, R23_method_handle); - __ mr(R5_ARG3, R0); // saved_sp __ mr(R6_ARG4, R1_SP); __ call_VM_leaf(CAST_FROM_FN_PTR(address, trace_method_handle_stub)); - __ restore_volatile_gprs(R1_SP, 112); // Except R0. __ pop_frame(); - __ restore_LR_CR(R0); + __ restore_LR_CR(tmp); + __ restore_volatile_gprs(R1_SP, -nbytes_save); // except R0 BLOCK_COMMENT("} trace_method_handle"); } diff --git a/hotspot/src/cpu/ppc/vm/ppc.ad b/hotspot/src/cpu/ppc/vm/ppc.ad index 51a9762e983..a9d576c5014 100644 --- a/hotspot/src/cpu/ppc/vm/ppc.ad +++ b/hotspot/src/cpu/ppc/vm/ppc.ad @@ -1,6 +1,6 @@ // -// Copyright (c) 2011, 2014, Oracle and/or its affiliates. All rights reserved. -// Copyright 2012, 2014 SAP AG. All rights reserved. +// Copyright (c) 2011, 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 @@ -2698,7 +2698,7 @@ encode %{ const_toc_addr = __ address_constant((address)a.value(), RelocationHolder::none); __ relocate(a.rspec()); } else if (constant_reloc == relocInfo::metadata_type) { - AddressLiteral a = __ allocate_metadata_address((Metadata *)val); + AddressLiteral a = __ constant_metadata_address((Metadata *)val); const_toc_addr = __ address_constant((address)a.value(), RelocationHolder::none); __ relocate(a.rspec()); } else { @@ -2727,7 +2727,7 @@ encode %{ const_toc_addr = __ address_constant((address)a.value(), RelocationHolder::none); __ relocate(a.rspec()); } else if (constant_reloc == relocInfo::metadata_type) { - AddressLiteral a = __ allocate_metadata_address((Metadata *)val); + AddressLiteral a = __ constant_metadata_address((Metadata *)val); const_toc_addr = __ address_constant((address)a.value(), RelocationHolder::none); __ relocate(a.rspec()); } else { // non-oop pointers, e.g. card mark base, heap top @@ -6029,6 +6029,20 @@ instruct clearMs32b(iRegNdst dst, iRegNsrc src) %{ ins_pipe(pipe_class_default); %} +// Optimize DecodeN for disjoint base. +// Load base of compressed oops into a register +instruct loadBase(iRegLdst dst) %{ + effect(DEF dst); + + format %{ "MR $dst, r30_heapbase" %} + size(4); + ins_encode %{ + // TODO: PPC port $archOpcode(ppc64Opcode_or); + __ mr($dst$$Register, R30); + %} + ins_pipe(pipe_class_default); +%} + // Loading ConN must be postalloc expanded so that edges between // the nodes are safe. They may not interfere with a safepoint. // GL TODO: This needs three instructions: better put this into the constant pool. @@ -6724,13 +6738,12 @@ instruct cond_set_0_oop(iRegNdst dst, flagsReg crx, iRegPsrc src1) %{ ins_pipe(pipe_class_default); %} -// base != 0 -// 32G aligned narrow oop base. -instruct encodeP_32GAligned(iRegNdst dst, iRegPsrc src) %{ +// Disjoint narrow oop base. +instruct encodeP_Disjoint(iRegNdst dst, iRegPsrc src) %{ match(Set dst (EncodeP src)); - predicate(false /* TODO: PPC port Universe::narrow_oop_base_disjoint()*/); + predicate(Universe::narrow_oop_base_disjoint()); - format %{ "EXTRDI $dst, $src, #32, #3 \t// encode with 32G aligned base" %} + format %{ "EXTRDI $dst, $src, #32, #3 \t// encode with disjoint base" %} size(4); ins_encode %{ // TODO: PPC port $archOpcode(ppc64Opcode_rldicl); @@ -6745,7 +6758,7 @@ instruct encodeP_Ex(iRegNdst dst, flagsReg crx, iRegPsrc src) %{ effect(TEMP crx); predicate(n->bottom_type()->make_ptr()->ptr() != TypePtr::NotNull && Universe::narrow_oop_shift() != 0 && - true /* TODO: PPC port Universe::narrow_oop_base_overlaps()*/); + Universe::narrow_oop_base_overlaps()); format %{ "EncodeP $dst, $crx, $src \t// postalloc expanded" %} postalloc_expand( postalloc_expand_encode_oop(dst, src, crx)); @@ -6756,7 +6769,7 @@ instruct encodeP_not_null_Ex(iRegNdst dst, iRegPsrc src) %{ match(Set dst (EncodeP src)); predicate(n->bottom_type()->make_ptr()->ptr() == TypePtr::NotNull && Universe::narrow_oop_shift() != 0 && - true /* TODO: PPC port Universe::narrow_oop_base_overlaps()*/); + Universe::narrow_oop_base_overlaps()); format %{ "EncodeP $dst, $src\t// $src != Null, postalloc expanded" %} postalloc_expand( postalloc_expand_encode_oop_not_null(dst, src) ); @@ -6876,6 +6889,7 @@ instruct decodeN_Ex(iRegPdst dst, iRegNsrc src, flagsReg crx) %{ n->bottom_type()->is_oopptr()->ptr() != TypePtr::Constant) && Universe::narrow_oop_shift() != 0 && Universe::narrow_oop_base() != 0); + ins_cost(4 * DEFAULT_COST); // Should be more expensive than decodeN_Disjoint_isel_Ex. effect(TEMP crx); format %{ "DecodeN $dst, $src \t// Kills $crx, postalloc expanded" %} @@ -6897,6 +6911,106 @@ instruct decodeN_nullBase(iRegPdst dst, iRegNsrc src) %{ ins_pipe(pipe_class_default); %} +// Optimize DecodeN for disjoint base. +// Shift narrow oop and or it into register that already contains the heap base. +// Base == dst must hold, and is assured by construction in postaloc_expand. +instruct decodeN_mergeDisjoint(iRegPdst dst, iRegNsrc src, iRegLsrc base) %{ + match(Set dst (DecodeN src)); + effect(TEMP base); + predicate(false); + + format %{ "RLDIMI $dst, $src, shift, 32-shift \t// DecodeN (disjoint base)" %} + size(4); + ins_encode %{ + // TODO: PPC port $archOpcode(ppc64Opcode_rldimi); + __ rldimi($dst$$Register, $src$$Register, Universe::narrow_oop_shift(), 32-Universe::narrow_oop_shift()); + %} + ins_pipe(pipe_class_default); +%} + +// Optimize DecodeN for disjoint base. +// This node requires only one cycle on the critical path. +// We must postalloc_expand as we can not express use_def effects where +// the used register is L and the def'ed register P. +instruct decodeN_Disjoint_notNull_Ex(iRegPdst dst, iRegNsrc src) %{ + match(Set dst (DecodeN src)); + effect(TEMP_DEF dst); + predicate((n->bottom_type()->is_oopptr()->ptr() == TypePtr::NotNull || + n->bottom_type()->is_oopptr()->ptr() == TypePtr::Constant) && + Universe::narrow_oop_base_disjoint()); + ins_cost(DEFAULT_COST); + + format %{ "MOV $dst, R30 \t\n" + "RLDIMI $dst, $src, shift, 32-shift \t// decode with disjoint base" %} + postalloc_expand %{ + loadBaseNode *n1 = new loadBaseNode(); + n1->add_req(NULL); + n1->_opnds[0] = op_dst; + + decodeN_mergeDisjointNode *n2 = new decodeN_mergeDisjointNode(); + n2->add_req(n_region, n_src, n1); + n2->_opnds[0] = op_dst; + n2->_opnds[1] = op_src; + n2->_opnds[2] = op_dst; + n2->_bottom_type = _bottom_type; + + ra_->set_pair(n1->_idx, ra_->get_reg_second(this), ra_->get_reg_first(this)); + ra_->set_pair(n2->_idx, ra_->get_reg_second(this), ra_->get_reg_first(this)); + + nodes->push(n1); + nodes->push(n2); + %} +%} + +instruct decodeN_Disjoint_isel_Ex(iRegPdst dst, iRegNsrc src, flagsReg crx) %{ + match(Set dst (DecodeN src)); + effect(TEMP_DEF dst, TEMP crx); + predicate((n->bottom_type()->is_oopptr()->ptr() != TypePtr::NotNull && + n->bottom_type()->is_oopptr()->ptr() != TypePtr::Constant) && + Universe::narrow_oop_base_disjoint() && VM_Version::has_isel()); + ins_cost(3 * DEFAULT_COST); + + format %{ "DecodeN $dst, $src \t// decode with disjoint base using isel" %} + postalloc_expand %{ + loadBaseNode *n1 = new loadBaseNode(); + n1->add_req(NULL); + n1->_opnds[0] = op_dst; + + cmpN_reg_imm0Node *n_compare = new cmpN_reg_imm0Node(); + n_compare->add_req(n_region, n_src); + n_compare->_opnds[0] = op_crx; + n_compare->_opnds[1] = op_src; + n_compare->_opnds[2] = new immN_0Oper(TypeNarrowOop::NULL_PTR); + + decodeN_mergeDisjointNode *n2 = new decodeN_mergeDisjointNode(); + n2->add_req(n_region, n_src, n1); + n2->_opnds[0] = op_dst; + n2->_opnds[1] = op_src; + n2->_opnds[2] = op_dst; + n2->_bottom_type = _bottom_type; + + cond_set_0_ptrNode *n_cond_set = new cond_set_0_ptrNode(); + n_cond_set->add_req(n_region, n_compare, n2); + n_cond_set->_opnds[0] = op_dst; + n_cond_set->_opnds[1] = op_crx; + n_cond_set->_opnds[2] = op_dst; + n_cond_set->_bottom_type = _bottom_type; + + assert(ra_->is_oop(this) == true, "A decodeN node must produce an oop!"); + ra_->set_oop(n_cond_set, true); + + ra_->set_pair(n1->_idx, ra_->get_reg_second(this), ra_->get_reg_first(this)); + ra_->set_pair(n_compare->_idx, ra_->get_reg_second(n_crx), ra_->get_reg_first(n_crx)); + ra_->set_pair(n2->_idx, ra_->get_reg_second(this), ra_->get_reg_first(this)); + ra_->set_pair(n_cond_set->_idx, ra_->get_reg_second(this), ra_->get_reg_first(this)); + + nodes->push(n1); + nodes->push(n_compare); + nodes->push(n2); + nodes->push(n_cond_set); + %} +%} + // src != 0, shift != 0, base != 0 instruct decodeN_notNull_addBase_Ex(iRegPdst dst, iRegNsrc src) %{ match(Set dst (DecodeN src)); @@ -6904,6 +7018,7 @@ instruct decodeN_notNull_addBase_Ex(iRegPdst dst, iRegNsrc src) %{ n->bottom_type()->is_oopptr()->ptr() == TypePtr::Constant) && Universe::narrow_oop_shift() != 0 && Universe::narrow_oop_base() != 0); + ins_cost(2 * DEFAULT_COST); format %{ "DecodeN $dst, $src \t// $src != NULL, postalloc expanded" %} postalloc_expand( postalloc_expand_decode_oop_not_null(dst, src)); @@ -6973,13 +7088,12 @@ instruct encodePKlass_sub_base(iRegPdst dst, iRegLsrc base, iRegPdst src) %{ ins_pipe(pipe_class_default); %} -// base != 0 -// 32G aligned narrow oop base. -instruct encodePKlass_32GAligned(iRegNdst dst, iRegPsrc src) %{ +// Disjoint narrow oop base. +instruct encodePKlass_Disjoint(iRegNdst dst, iRegPsrc src) %{ match(Set dst (EncodePKlass src)); predicate(false /* TODO: PPC port Universe::narrow_klass_base_disjoint()*/); - format %{ "EXTRDI $dst, $src, #32, #3 \t// encode with 32G aligned base" %} + format %{ "EXTRDI $dst, $src, #32, #3 \t// encode with disjoint base" %} size(4); ins_encode %{ // TODO: PPC port $archOpcode(ppc64Opcode_rldicl); @@ -7486,7 +7600,7 @@ instruct storeLConditional_regP_regL_regL(flagsReg crx, indirect mem_ptr, iRegLs ins_encode %{ // TODO: PPC port $archOpcode(ppc64Opcode_compound); __ cmpxchgd($crx$$CondRegister, R0, $oldVal$$Register, $newVal$$Register, $mem_ptr$$Register, - MacroAssembler::MemBarNone, MacroAssembler::cmpxchgx_hint_atomic_update(), + MacroAssembler::MemBarAcq, MacroAssembler::cmpxchgx_hint_atomic_update(), noreg, NULL, true); %} ins_pipe(pipe_class_default); @@ -10476,7 +10590,7 @@ instruct cmpN_reg_reg(flagsReg crx, iRegNsrc src1, iRegNsrc src2) %{ match(Set crx (CmpN src1 src2)); size(4); - ins_cost(DEFAULT_COST); + ins_cost(2); format %{ "CMPLW $crx, $src1, $src2 \t// compressed ptr" %} ins_encode %{ // TODO: PPC port $archOpcode(ppc64Opcode_cmpl); @@ -10488,7 +10602,7 @@ instruct cmpN_reg_reg(flagsReg crx, iRegNsrc src1, iRegNsrc src2) %{ instruct cmpN_reg_imm0(flagsReg crx, iRegNsrc src1, immN_0 src2) %{ match(Set crx (CmpN src1 src2)); // Make this more expensive than zeroCheckN_iReg_imm0. - ins_cost(DEFAULT_COST); + ins_cost(2); format %{ "CMPLWI $crx, $src1, $src2 \t// compressed ptr" %} size(4); @@ -10508,6 +10622,7 @@ instruct zeroCheckP_reg_imm0(cmpOp cmp, iRegP_N2P value, immP_0 zero, label labl _kids[0]->_leaf->as_Bool()->_test._test == BoolTest::ne && _leaf->as_If()->_prob >= PROB_LIKELY_MAG(4) && Matcher::branches_to_uncommon_trap(_leaf)); + ins_cost(1); // Should not be cheaper than zeroCheckN. ins_is_TrapBasedCheckNode(true); @@ -10889,7 +11004,7 @@ instruct branchLoopEndSched(cmpOp cmp, flagsReg crx, label labl) %{ instruct partialSubtypeCheck(iRegPdst result, iRegP_N2P subklass, iRegP_N2P superklass, iRegPdst tmp_klass, iRegPdst tmp_arrayptr) %{ match(Set result (PartialSubtypeCheck subklass superklass)); - effect(TEMP result, TEMP tmp_klass, TEMP tmp_arrayptr); + effect(TEMP_DEF result, TEMP tmp_klass, TEMP tmp_arrayptr); ins_cost(DEFAULT_COST*10); format %{ "PartialSubtypeCheck $result = ($subklass instanceOf $superklass) tmp: $tmp_klass, $tmp_arrayptr" %} @@ -11000,7 +11115,7 @@ instruct string_indexOf_imm1_char(iRegIdst result, iRegPsrc haystack, iRegIsrc h predicate(SpecialStringIndexOf); // type check implicit by parameter type, See Matcher::match_rule_supported match(Set result (StrIndexOf (Binary haystack haycnt) (Binary (AddP needleImm offsetImm) needlecntImm))); - effect(TEMP result, TEMP tmp1, TEMP tmp2, KILL cr0, KILL cr1); + effect(TEMP_DEF result, TEMP tmp1, TEMP tmp2, KILL cr0, KILL cr1); ins_cost(150); format %{ "String IndexOf CSCL1 $haystack[0..$haycnt], $needleImm+$offsetImm[0..$needlecntImm]" @@ -11037,7 +11152,7 @@ instruct string_indexOf_imm1(iRegIdst result, iRegPsrc haystack, iRegIsrc haycnt iRegIdst tmp1, iRegIdst tmp2, flagsRegCR0 cr0, flagsRegCR1 cr1) %{ match(Set result (StrIndexOf (Binary haystack haycnt) (Binary needle needlecntImm))); - effect(USE_KILL needle, /* TDEF needle, */ TEMP result, + effect(USE_KILL needle, /* TDEF needle, */ TEMP_DEF result, TEMP tmp1, TEMP tmp2); // Required for EA: check if it is still a type_array. predicate(SpecialStringIndexOf && n->in(3)->in(1)->bottom_type()->is_aryptr()->const_oop() && @@ -11084,7 +11199,7 @@ instruct string_indexOf_imm(iRegIdst result, iRegPsrc haystack, rscratch1RegI ha iRegIdst tmp1, iRegIdst tmp2, iRegIdst tmp3, iRegIdst tmp4, iRegIdst tmp5, flagsRegCR0 cr0, flagsRegCR1 cr1, flagsRegCR6 cr6) %{ match(Set result (StrIndexOf (Binary haystack haycnt) (Binary needle needlecntImm))); - effect(USE_KILL haycnt, /* better: TDEF haycnt, */ TEMP result, + effect(USE_KILL haycnt, /* better: TDEF haycnt, */ TEMP_DEF result, TEMP tmp1, TEMP tmp2, TEMP tmp3, TEMP tmp4, TEMP tmp5, KILL cr0, KILL cr1, KILL cr6); // Required for EA: check if it is still a type_array. predicate(SpecialStringIndexOf && n->in(3)->in(1)->bottom_type()->is_aryptr()->const_oop() && @@ -11118,7 +11233,7 @@ instruct string_indexOf(iRegIdst result, iRegPsrc haystack, rscratch1RegI haycnt flagsRegCR0 cr0, flagsRegCR1 cr1, flagsRegCR6 cr6) %{ match(Set result (StrIndexOf (Binary haystack haycnt) (Binary needle needlecnt))); effect(USE_KILL haycnt, USE_KILL needlecnt, /*better: TDEF haycnt, TDEF needlecnt,*/ - TEMP result, + TEMP_DEF result, TEMP tmp1, TEMP tmp2, TEMP tmp3, TEMP tmp4, KILL cr0, KILL cr1, KILL cr6); predicate(SpecialStringIndexOf); // See Matcher::match_rule_supported. ins_cost(300); @@ -11142,7 +11257,7 @@ instruct string_equals_imm(iRegPsrc str1, iRegPsrc str2, uimmI15 cntImm, iRegIds iRegPdst tmp1, iRegPdst tmp2, flagsRegCR0 cr0, flagsRegCR6 cr6, regCTR ctr) %{ match(Set result (StrEquals (Binary str1 str2) cntImm)); - effect(TEMP result, TEMP tmp1, TEMP tmp2, + effect(TEMP_DEF result, TEMP tmp1, TEMP tmp2, KILL cr0, KILL cr6, KILL ctr); predicate(SpecialStringEquals); // See Matcher::match_rule_supported. ins_cost(250); @@ -11165,7 +11280,7 @@ instruct string_equals(iRegPsrc str1, iRegPsrc str2, iRegIsrc cnt, iRegIdst resu iRegPdst tmp1, iRegPdst tmp2, iRegPdst tmp3, iRegPdst tmp4, iRegPdst tmp5, flagsRegCR0 cr0, flagsRegCR1 cr1, flagsRegCR6 cr6, regCTR ctr) %{ match(Set result (StrEquals (Binary str1 str2) cnt)); - effect(TEMP result, TEMP tmp1, TEMP tmp2, TEMP tmp3, TEMP tmp4, TEMP tmp5, + effect(TEMP_DEF result, TEMP tmp1, TEMP tmp2, TEMP tmp3, TEMP tmp4, TEMP tmp5, KILL cr0, KILL cr1, KILL cr6, KILL ctr); predicate(SpecialStringEquals); // See Matcher::match_rule_supported. ins_cost(300); @@ -11188,7 +11303,7 @@ instruct string_equals(iRegPsrc str1, iRegPsrc str2, iRegIsrc cnt, iRegIdst resu instruct string_compare(rarg1RegP str1, rarg2RegP str2, rarg3RegI cnt1, rarg4RegI cnt2, iRegIdst result, iRegPdst tmp, flagsRegCR0 cr0, regCTR ctr) %{ match(Set result (StrComp (Binary str1 cnt1) (Binary str2 cnt2))); - effect(USE_KILL cnt1, USE_KILL cnt2, USE_KILL str1, USE_KILL str2, TEMP result, TEMP tmp, KILL cr0, KILL ctr); + effect(USE_KILL cnt1, USE_KILL cnt2, USE_KILL str1, USE_KILL str2, TEMP_DEF result, TEMP tmp, KILL cr0, KILL ctr); ins_cost(300); ins_alignment(8); // 'compute_padding()' gets called, up to this number-1 nops will get inserted. From 3b3dd985de385a8dc6aca09abf8dd8f3bdf537cb Mon Sep 17 00:00:00 2001 From: Roland Westrelin Date: Wed, 7 Jan 2015 16:40:02 +0100 Subject: [PATCH 05/94] 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 06/94] 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 07/94] 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 08/94] 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 09/94] 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 10/94] 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 11/94] 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 12/94] 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 13/94] 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 14/94] 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 15/94] 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 16/94] 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 17/94] 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 18/94] 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 19/94] 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 20/94] 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 21/94] 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 22/94] 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 23/94] 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 24/94] 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 25/94] 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 26/94] 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,,