This commit is contained in:
Alejandro Murillo 2016-06-23 17:07:27 -07:00
commit f7095d870c
232 changed files with 2800 additions and 1172 deletions

View File

@ -153,6 +153,13 @@ else ifeq ($(OPENJDK_TARGET_OS), aix)
# mode, so don't optimize sharedRuntimeTrig.cpp at all. # mode, so don't optimize sharedRuntimeTrig.cpp at all.
BUILD_LIBJVM_sharedRuntimeTrig.cpp_CXXFLAGS := $(CXX_O_FLAG_NONE) BUILD_LIBJVM_sharedRuntimeTrig.cpp_CXXFLAGS := $(CXX_O_FLAG_NONE)
ifneq ($(DEBUG_LEVEL),slowdebug)
# Compiling jvmtiEnterTrace.cpp with full optimization needs more than 30min
# (mostly because of '-qhot=level=1' and the more than 1300 'log_trace' calls
# which cause a lot of template expansion).
BUILD_LIBJVM_jvmtiEnterTrace.cpp_OPTIMIZATION := LOW
endif
# Disable ELF decoder on AIX (AIX uses XCOFF). # Disable ELF decoder on AIX (AIX uses XCOFF).
JVM_EXCLUDE_PATTERNS += elf JVM_EXCLUDE_PATTERNS += elf

View File

@ -12179,21 +12179,21 @@ instruct rolL_rReg_Var_C0(iRegLNoSp dst, iRegL src, iRegI shift, immI0 c0, rFlag
%} %}
%} %}
instruct rolI_rReg_Var_C_32(iRegLNoSp dst, iRegL src, iRegI shift, immI_32 c_32, rFlagsReg cr) instruct rolI_rReg_Var_C_32(iRegINoSp dst, iRegI src, iRegI shift, immI_32 c_32, rFlagsReg cr)
%{ %{
match(Set dst (OrI (LShiftI src shift) (URShiftI src (SubI c_32 shift)))); match(Set dst (OrI (LShiftI src shift) (URShiftI src (SubI c_32 shift))));
expand %{ expand %{
rolL_rReg(dst, src, shift, cr); rolI_rReg(dst, src, shift, cr);
%} %}
%} %}
instruct rolI_rReg_Var_C0(iRegLNoSp dst, iRegL src, iRegI shift, immI0 c0, rFlagsReg cr) instruct rolI_rReg_Var_C0(iRegINoSp dst, iRegI src, iRegI shift, immI0 c0, rFlagsReg cr)
%{ %{
match(Set dst (OrI (LShiftI src shift) (URShiftI src (SubI c0 shift)))); match(Set dst (OrI (LShiftI src shift) (URShiftI src (SubI c0 shift))));
expand %{ expand %{
rolL_rReg(dst, src, shift, cr); rolI_rReg(dst, src, shift, cr);
%} %}
%} %}

View File

@ -944,8 +944,7 @@ OopMapSet* Runtime1::generate_code_for(StubID id, StubAssembler* sasm) {
Register t = r5; Register t = r5;
__ load_klass(t, r0); __ load_klass(t, r0);
__ ldrw(t, Address(t, Klass::access_flags_offset())); __ ldrw(t, Address(t, Klass::access_flags_offset()));
__ tst(t, JVM_ACC_HAS_FINALIZER); __ tbnz(t, exact_log2(JVM_ACC_HAS_FINALIZER), register_finalizer);
__ br(Assembler::NE, register_finalizer);
__ ret(lr); __ ret(lr);
__ bind(register_finalizer); __ bind(register_finalizer);

View File

@ -93,10 +93,8 @@ void InterpreterMacroAssembler::check_and_handle_popframe(Register java_thread)
// This method is only called just after the call into the vm in // This method is only called just after the call into the vm in
// call_VM_base, so the arg registers are available. // call_VM_base, so the arg registers are available.
ldrw(rscratch1, Address(rthread, JavaThread::popframe_condition_offset())); ldrw(rscratch1, Address(rthread, JavaThread::popframe_condition_offset()));
tstw(rscratch1, JavaThread::popframe_pending_bit); tbz(rscratch1, exact_log2(JavaThread::popframe_pending_bit), L);
br(Assembler::EQ, L); tbnz(rscratch1, exact_log2(JavaThread::popframe_processing_bit), L);
tstw(rscratch1, JavaThread::popframe_processing_bit);
br(Assembler::NE, L);
// Call Interpreter::remove_activation_preserving_args_entry() to get the // Call Interpreter::remove_activation_preserving_args_entry() to get the
// address of the same-named entrypoint in the generated interpreter code. // address of the same-named entrypoint in the generated interpreter code.
call_VM_leaf(CAST_FROM_FN_PTR(address, Interpreter::remove_activation_preserving_args_entry)); call_VM_leaf(CAST_FROM_FN_PTR(address, Interpreter::remove_activation_preserving_args_entry));
@ -505,8 +503,7 @@ void InterpreterMacroAssembler::remove_activation(
// get method access flags // get method access flags
ldr(r1, Address(rfp, frame::interpreter_frame_method_offset * wordSize)); ldr(r1, Address(rfp, frame::interpreter_frame_method_offset * wordSize));
ldr(r2, Address(r1, Method::access_flags_offset())); ldr(r2, Address(r1, Method::access_flags_offset()));
tst(r2, JVM_ACC_SYNCHRONIZED); tbz(r2, exact_log2(JVM_ACC_SYNCHRONIZED), unlocked);
br(Assembler::EQ, unlocked);
// Don't unlock anything if the _do_not_unlock_if_synchronized flag // Don't unlock anything if the _do_not_unlock_if_synchronized flag
// is set. // is set.
@ -1582,8 +1579,8 @@ void InterpreterMacroAssembler::profile_obj_type(Register obj, const Address& md
// do. The unknown bit may have been // do. The unknown bit may have been
// set already but no need to check. // set already but no need to check.
tst(obj, TypeEntries::type_unknown); tbnz(obj, exact_log2(TypeEntries::type_unknown), next);
br(Assembler::NE, next); // already unknown. Nothing to do anymore. // already unknown. Nothing to do anymore.
ldr(rscratch1, mdo_addr); ldr(rscratch1, mdo_addr);
cbz(rscratch1, none); cbz(rscratch1, none);

View File

@ -710,7 +710,7 @@ class StubGenerator: public StubCodeGenerator {
__ BIND(L_loop); __ BIND(L_loop);
__ strb(zr, Address(start, count)); __ strb(zr, Address(start, count));
__ subs(count, count, 1); __ subs(count, count, 1);
__ br(Assembler::HS, L_loop); __ br(Assembler::GE, L_loop);
} }
break; break;
default: default:
@ -1299,7 +1299,7 @@ class StubGenerator: public StubCodeGenerator {
if (VerifyOops) if (VerifyOops)
verify_oop_array(size, d, count, r16); verify_oop_array(size, d, count, r16);
__ sub(count, count, 1); // make an inclusive end pointer __ sub(count, count, 1); // make an inclusive end pointer
__ lea(count, Address(d, count, Address::uxtw(exact_log2(size)))); __ lea(count, Address(d, count, Address::lsl(exact_log2(size))));
gen_write_ref_array_post_barrier(d, count, rscratch1); gen_write_ref_array_post_barrier(d, count, rscratch1);
} }
__ leave(); __ leave();
@ -2002,9 +2002,9 @@ class StubGenerator: public StubCodeGenerator {
arraycopy_range_checks(src, src_pos, dst, dst_pos, scratch_length, arraycopy_range_checks(src, src_pos, dst, dst_pos, scratch_length,
rscratch2, L_failed); rscratch2, L_failed);
__ lea(from, Address(src, src_pos, Address::lsl(3))); __ lea(from, Address(src, src_pos, Address::lsl(LogBytesPerHeapOop)));
__ add(from, from, arrayOopDesc::base_offset_in_bytes(T_OBJECT)); __ add(from, from, arrayOopDesc::base_offset_in_bytes(T_OBJECT));
__ lea(to, Address(dst, dst_pos, Address::lsl(3))); __ lea(to, Address(dst, dst_pos, Address::lsl(LogBytesPerHeapOop)));
__ add(to, to, arrayOopDesc::base_offset_in_bytes(T_OBJECT)); __ add(to, to, arrayOopDesc::base_offset_in_bytes(T_OBJECT));
__ movw(count, scratch_length); // length __ movw(count, scratch_length); // length
__ BIND(L_plain_copy); __ BIND(L_plain_copy);
@ -2027,9 +2027,9 @@ class StubGenerator: public StubCodeGenerator {
__ load_klass(rscratch2_dst_klass, dst); // reload __ load_klass(rscratch2_dst_klass, dst); // reload
// Marshal the base address arguments now, freeing registers. // Marshal the base address arguments now, freeing registers.
__ lea(from, Address(src, src_pos, Address::lsl(3))); __ lea(from, Address(src, src_pos, Address::lsl(LogBytesPerHeapOop)));
__ add(from, from, arrayOopDesc::base_offset_in_bytes(T_OBJECT)); __ add(from, from, arrayOopDesc::base_offset_in_bytes(T_OBJECT));
__ lea(to, Address(dst, dst_pos, Address::lsl(3))); __ lea(to, Address(dst, dst_pos, Address::lsl(LogBytesPerHeapOop)));
__ add(to, to, arrayOopDesc::base_offset_in_bytes(T_OBJECT)); __ add(to, to, arrayOopDesc::base_offset_in_bytes(T_OBJECT));
__ movw(count, length); // length (reloaded) __ movw(count, length); // length (reloaded)
Register sco_temp = c_rarg3; // this register is free now Register sco_temp = c_rarg3; // this register is free now

View File

@ -1242,8 +1242,7 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
{ {
Label L; Label L;
__ ldrw(t, Address(rmethod, Method::access_flags_offset())); __ ldrw(t, Address(rmethod, Method::access_flags_offset()));
__ tst(t, JVM_ACC_STATIC); __ tbz(t, exact_log2(JVM_ACC_STATIC), L);
__ br(Assembler::EQ, L);
// get mirror // get mirror
__ load_mirror(t, rmethod); __ load_mirror(t, rmethod);
// copy mirror into activation frame // copy mirror into activation frame
@ -1435,8 +1434,7 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
{ {
Label L; Label L;
__ ldrw(t, Address(rmethod, Method::access_flags_offset())); __ ldrw(t, Address(rmethod, Method::access_flags_offset()));
__ tst(t, JVM_ACC_SYNCHRONIZED); __ tbz(t, exact_log2(JVM_ACC_SYNCHRONIZED), L);
__ br(Assembler::EQ, L);
// the code below should be shared with interpreter macro // the code below should be shared with interpreter macro
// assembler implementation // assembler implementation
{ {

View File

@ -2190,9 +2190,8 @@ void TemplateTable::_return(TosState state)
__ ldr(c_rarg1, aaddress(0)); __ ldr(c_rarg1, aaddress(0));
__ load_klass(r3, c_rarg1); __ load_klass(r3, c_rarg1);
__ ldrw(r3, Address(r3, Klass::access_flags_offset())); __ ldrw(r3, Address(r3, Klass::access_flags_offset()));
__ tst(r3, JVM_ACC_HAS_FINALIZER);
Label skip_register_finalizer; Label skip_register_finalizer;
__ br(Assembler::EQ, skip_register_finalizer); __ tbz(r3, exact_log2(JVM_ACC_HAS_FINALIZER), skip_register_finalizer);
__ call_VM(noreg, CAST_FROM_FN_PTR(address, InterpreterRuntime::register_finalizer), c_rarg1); __ call_VM(noreg, CAST_FROM_FN_PTR(address, InterpreterRuntime::register_finalizer), c_rarg1);

View File

@ -503,6 +503,10 @@ class Assembler : public AbstractAssembler {
LVSL_OPCODE = (31u << OPCODE_SHIFT | 6u << 1), LVSL_OPCODE = (31u << OPCODE_SHIFT | 6u << 1),
LVSR_OPCODE = (31u << OPCODE_SHIFT | 38u << 1), LVSR_OPCODE = (31u << OPCODE_SHIFT | 38u << 1),
// Vector-Scalar (VSX) instruction support.
LXVD2X_OPCODE = (31u << OPCODE_SHIFT | 844u << 1),
STXVD2X_OPCODE = (31u << OPCODE_SHIFT | 972u << 1),
// Vector Permute and Formatting // Vector Permute and Formatting
VPKPX_OPCODE = (4u << OPCODE_SHIFT | 782u ), VPKPX_OPCODE = (4u << OPCODE_SHIFT | 782u ),
VPKSHSS_OPCODE = (4u << OPCODE_SHIFT | 398u ), VPKSHSS_OPCODE = (4u << OPCODE_SHIFT | 398u ),
@ -1085,6 +1089,19 @@ class Assembler : public AbstractAssembler {
static int vrs( VectorRegister r) { return vrs(r->encoding());} static int vrs( VectorRegister r) { return vrs(r->encoding());}
static int vrt( VectorRegister r) { return vrt(r->encoding());} static int vrt( VectorRegister r) { return vrt(r->encoding());}
// Support Vector-Scalar (VSX) instructions.
static int vsra( int x) { return opp_u_field(x, 15, 11); }
static int vsrb( int x) { return opp_u_field(x, 20, 16); }
static int vsrc( int x) { return opp_u_field(x, 25, 21); }
static int vsrs( int x) { return opp_u_field(x, 10, 6); }
static int vsrt( int x) { return opp_u_field(x, 10, 6); }
static int vsra( VectorSRegister r) { return vsra(r->encoding());}
static int vsrb( VectorSRegister r) { return vsrb(r->encoding());}
static int vsrc( VectorSRegister r) { return vsrc(r->encoding());}
static int vsrs( VectorSRegister r) { return vsrs(r->encoding());}
static int vsrt( VectorSRegister r) { return vsrt(r->encoding());}
static int vsplt_uim( int x) { return opp_u_field(x, 15, 12); } // for vsplt* instructions static int vsplt_uim( int x) { return opp_u_field(x, 15, 12); } // for vsplt* instructions
static int vsplti_sim(int x) { return opp_u_field(x, 15, 11); } // for vsplti* instructions static int vsplti_sim(int x) { return opp_u_field(x, 15, 11); } // for vsplti* instructions
static int vsldoi_shb(int x) { return opp_u_field(x, 25, 22); } // for vsldoi instruction static int vsldoi_shb(int x) { return opp_u_field(x, 25, 22); } // for vsldoi instruction
@ -2069,6 +2086,10 @@ class Assembler : public AbstractAssembler {
inline void mtvscr( VectorRegister b); inline void mtvscr( VectorRegister b);
inline void mfvscr( VectorRegister d); inline void mfvscr( VectorRegister d);
// Vector-Scalar (VSX) instructions.
inline void lxvd2x( VectorSRegister d, Register a, Register b);
inline void stxvd2x( VectorSRegister d, Register a, Register b);
// AES (introduced with Power 8) // AES (introduced with Power 8)
inline void vcipher( VectorRegister d, VectorRegister a, VectorRegister b); inline void vcipher( VectorRegister d, VectorRegister a, VectorRegister b);
inline void vcipherlast( VectorRegister d, VectorRegister a, VectorRegister b); inline void vcipherlast( VectorRegister d, VectorRegister a, VectorRegister b);

View File

@ -724,6 +724,10 @@ inline void Assembler::stvxl( VectorRegister d, Register s1, Register s2) { emit
inline void Assembler::lvsl( VectorRegister d, Register s1, Register s2) { emit_int32( LVSL_OPCODE | vrt(d) | ra0mem(s1) | rb(s2)); } inline void Assembler::lvsl( VectorRegister d, Register s1, Register s2) { emit_int32( LVSL_OPCODE | vrt(d) | ra0mem(s1) | rb(s2)); }
inline void Assembler::lvsr( VectorRegister d, Register s1, Register s2) { emit_int32( LVSR_OPCODE | vrt(d) | ra0mem(s1) | rb(s2)); } inline void Assembler::lvsr( VectorRegister d, Register s1, Register s2) { emit_int32( LVSR_OPCODE | vrt(d) | ra0mem(s1) | rb(s2)); }
// Vector-Scalar (VSX) instructions.
inline void Assembler::lxvd2x (VectorSRegister d, Register s1, Register s2) { emit_int32( LXVD2X_OPCODE | vsrt(d) | ra(s1) | rb(s2)); }
inline void Assembler::stxvd2x(VectorSRegister d, Register s1, Register s2) { emit_int32( STXVD2X_OPCODE | vsrt(d) | ra(s1) | rb(s2)); }
inline void Assembler::vpkpx( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKPX_OPCODE | vrt(d) | vra(a) | vrb(b)); } inline void Assembler::vpkpx( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKPX_OPCODE | vrt(d) | vra(a) | vrb(b)); }
inline void Assembler::vpkshss( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKSHSS_OPCODE | vrt(d) | vra(a) | vrb(b)); } inline void Assembler::vpkshss( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKSHSS_OPCODE | vrt(d) | vra(a) | vrb(b)); }
inline void Assembler::vpkswss( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKSWSS_OPCODE | vrt(d) | vra(a) | vrb(b)); } inline void Assembler::vpkswss( VectorRegister d, VectorRegister a, VectorRegister b) { emit_int32( VPKSWSS_OPCODE | vrt(d) | vra(a) | vrb(b)); }

View File

@ -75,3 +75,14 @@ const char* VectorRegisterImpl::name() const {
}; };
return is_valid() ? names[encoding()] : "vnoreg"; return is_valid() ? names[encoding()] : "vnoreg";
} }
const char* VectorSRegisterImpl::name() const {
const char* names[number_of_registers] = {
"VSR0", "VSR1", "VSR2", "VSR3", "VSR4", "VSR5", "VSR6", "VSR7",
"VSR8", "VSR9", "VSR10", "VSR11", "VSR12", "VSR13", "VSR14", "VSR15",
"VSR16", "VSR17", "VSR18", "VSR19", "VSR20", "VSR21", "VSR22", "VSR23",
"VSR24", "VSR25", "VSR26", "VSR27", "VSR28", "VSR29", "VSR30", "VSR31"
};
return is_valid() ? names[encoding()] : "vsnoreg";
}

View File

@ -491,6 +491,106 @@ CONSTANT_REGISTER_DECLARATION(VectorRegister, VR31, (31));
#endif // DONT_USE_REGISTER_DEFINES #endif // DONT_USE_REGISTER_DEFINES
// Use VectorSRegister as a shortcut.
class VectorSRegisterImpl;
typedef VectorSRegisterImpl* VectorSRegister;
inline VectorSRegister as_VectorSRegister(int encoding) {
return (VectorSRegister)(intptr_t)encoding;
}
// The implementation of Vector-Scalar (VSX) registers on POWER architecture.
class VectorSRegisterImpl: public AbstractRegisterImpl {
public:
enum {
number_of_registers = 32
};
// construction
inline friend VectorSRegister as_VectorSRegister(int encoding);
// accessors
int encoding() const { assert(is_valid(), "invalid register"); return value(); }
// testers
bool is_valid() const { return 0 <= value() && value() < number_of_registers; }
const char* name() const;
};
// The Vector-Scalar (VSX) registers of the POWER architecture.
CONSTANT_REGISTER_DECLARATION(VectorSRegister, vsnoreg, (-1));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR0, ( 0));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR1, ( 1));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR2, ( 2));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR3, ( 3));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR4, ( 4));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR5, ( 5));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR6, ( 6));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR7, ( 7));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR8, ( 8));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR9, ( 9));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR10, (10));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR11, (11));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR12, (12));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR13, (13));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR14, (14));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR15, (15));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR16, (16));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR17, (17));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR18, (18));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR19, (19));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR20, (20));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR21, (21));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR22, (22));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR23, (23));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR24, (24));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR25, (25));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR26, (26));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR27, (27));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR28, (28));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR29, (29));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR30, (30));
CONSTANT_REGISTER_DECLARATION(VectorSRegister, VSR31, (31));
#ifndef DONT_USE_REGISTER_DEFINES
#define vsnoregi ((VectorSRegister)(vsnoreg_VectorSRegisterEnumValue))
#define VSR0 ((VectorSRegister)( VSR0_VectorSRegisterEnumValue))
#define VSR1 ((VectorSRegister)( VSR1_VectorSRegisterEnumValue))
#define VSR2 ((VectorSRegister)( VSR2_VectorSRegisterEnumValue))
#define VSR3 ((VectorSRegister)( VSR3_VectorSRegisterEnumValue))
#define VSR4 ((VectorSRegister)( VSR4_VectorSRegisterEnumValue))
#define VSR5 ((VectorSRegister)( VSR5_VectorSRegisterEnumValue))
#define VSR6 ((VectorSRegister)( VSR6_VectorSRegisterEnumValue))
#define VSR7 ((VectorSRegister)( VSR7_VectorSRegisterEnumValue))
#define VSR8 ((VectorSRegister)( VSR8_VectorSRegisterEnumValue))
#define VSR9 ((VectorSRegister)( VSR9_VectorSRegisterEnumValue))
#define VSR10 ((VectorSRegister)( VSR10_VectorSRegisterEnumValue))
#define VSR11 ((VectorSRegister)( VSR11_VectorSRegisterEnumValue))
#define VSR12 ((VectorSRegister)( VSR12_VectorSRegisterEnumValue))
#define VSR13 ((VectorSRegister)( VSR13_VectorSRegisterEnumValue))
#define VSR14 ((VectorSRegister)( VSR14_VectorSRegisterEnumValue))
#define VSR15 ((VectorSRegister)( VSR15_VectorSRegisterEnumValue))
#define VSR16 ((VectorSRegister)( VSR16_VectorSRegisterEnumValue))
#define VSR17 ((VectorSRegister)( VSR17_VectorSRegisterEnumValue))
#define VSR18 ((VectorSRegister)( VSR18_VectorSRegisterEnumValue))
#define VSR19 ((VectorSRegister)( VSR19_VectorSRegisterEnumValue))
#define VSR20 ((VectorSRegister)( VSR20_VectorSRegisterEnumValue))
#define VSR21 ((VectorSRegister)( VSR21_VectorSRegisterEnumValue))
#define VSR22 ((VectorSRegister)( VSR22_VectorSRegisterEnumValue))
#define VSR23 ((VectorSRegister)( VSR23_VectorSRegisterEnumValue))
#define VSR24 ((VectorSRegister)( VSR24_VectorSRegisterEnumValue))
#define VSR25 ((VectorSRegister)( VSR25_VectorSRegisterEnumValue))
#define VSR26 ((VectorSRegister)( VSR26_VectorSRegisterEnumValue))
#define VSR27 ((VectorSRegister)( VSR27_VectorSRegisterEnumValue))
#define VSR28 ((VectorSRegister)( VSR28_VectorSRegisterEnumValue))
#define VSR29 ((VectorSRegister)( VSR29_VectorSRegisterEnumValue))
#define VSR30 ((VectorSRegister)( VSR30_VectorSRegisterEnumValue))
#define VSR31 ((VectorSRegister)( VSR31_VectorSRegisterEnumValue))
#endif // DONT_USE_REGISTER_DEFINES
// Maximum number of incoming arguments that can be passed in i registers. // Maximum number of incoming arguments that can be passed in i registers.
const int PPC_ARGS_IN_REGS_NUM = 8; const int PPC_ARGS_IN_REGS_NUM = 8;

View File

@ -1341,10 +1341,13 @@ class StubGenerator: public StubCodeGenerator {
Register tmp3 = R8_ARG6; Register tmp3 = R8_ARG6;
Register tmp4 = R9_ARG7; Register tmp4 = R9_ARG7;
VectorSRegister tmp_vsr1 = VSR1;
VectorSRegister tmp_vsr2 = VSR2;
address start = __ function_entry(); address start = __ function_entry();
assert_positive_int(R5_ARG3); assert_positive_int(R5_ARG3);
Label l_1, l_2, l_3, l_4, l_5, l_6, l_7, l_8; Label l_1, l_2, l_3, l_4, l_5, l_6, l_7, l_8, l_9;
// don't try anything fancy if arrays don't have many elements // don't try anything fancy if arrays don't have many elements
__ li(tmp3, 0); __ li(tmp3, 0);
@ -1403,6 +1406,8 @@ class StubGenerator: public StubCodeGenerator {
__ andi_(R5_ARG3, R5_ARG3, 15); __ andi_(R5_ARG3, R5_ARG3, 15);
__ mtctr(tmp1); __ mtctr(tmp1);
if (!VM_Version::has_vsx()) {
__ bind(l_8); __ bind(l_8);
// Use unrolled version for mass copying (copy 16 elements a time). // Use unrolled version for mass copying (copy 16 elements a time).
// Load feeding store gets zero latency on Power6, however not on Power5. // Load feeding store gets zero latency on Power6, however not on Power5.
@ -1418,7 +1423,43 @@ class StubGenerator: public StubCodeGenerator {
__ addi(R3_ARG1, R3_ARG1, 32); __ addi(R3_ARG1, R3_ARG1, 32);
__ addi(R4_ARG2, R4_ARG2, 32); __ addi(R4_ARG2, R4_ARG2, 32);
__ bdnz(l_8); __ bdnz(l_8);
} else { // Processor supports VSX, so use it to mass copy.
// Prefetch src data into L2 cache.
__ dcbt(R3_ARG1, 0);
// If supported set DSCR pre-fetch to deepest.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val | 7);
__ mtdscr(tmp2);
} }
__ li(tmp1, 16);
// Backbranch target aligned to 32-byte. It's not aligned 16-byte
// as loop contains < 8 instructions that fit inside a single
// i-cache sector.
__ align(32);
__ bind(l_9);
// Use loop with VSX load/store instructions to
// copy 16 elements a time.
__ lxvd2x(tmp_vsr1, 0, R3_ARG1); // Load from src.
__ stxvd2x(tmp_vsr1, 0, R4_ARG2); // Store to dst.
__ lxvd2x(tmp_vsr2, R3_ARG1, tmp1); // Load from src + 16.
__ stxvd2x(tmp_vsr2, R4_ARG2, tmp1); // Store to dst + 16.
__ addi(R3_ARG1, R3_ARG1, 32); // Update src+=32.
__ addi(R4_ARG2, R4_ARG2, 32); // Update dsc+=32.
__ bdnz(l_9); // Dec CTR and loop if not zero.
// Restore DSCR pre-fetch value.
if (VM_Version::has_mfdscr()) {
__ load_const_optimized(tmp2, VM_Version::_dscr_val);
__ mtdscr(tmp2);
}
}
} // FasterArrayCopy
__ bind(l_6); __ bind(l_6);
// copy 2 elements at a time // copy 2 elements at a time

View File

@ -38,7 +38,7 @@
# include <sys/sysinfo.h> # include <sys/sysinfo.h>
bool VM_Version::_is_determine_features_test_running = false; bool VM_Version::_is_determine_features_test_running = false;
uint64_t VM_Version::_dscr_val = 0;
#define MSG(flag) \ #define MSG(flag) \
if (flag && !FLAG_IS_DEFAULT(flag)) \ if (flag && !FLAG_IS_DEFAULT(flag)) \
@ -111,7 +111,7 @@ void VM_Version::initialize() {
// Create and print feature-string. // Create and print feature-string.
char buf[(num_features+1) * 16]; // Max 16 chars per feature. char buf[(num_features+1) * 16]; // Max 16 chars per feature.
jio_snprintf(buf, sizeof(buf), jio_snprintf(buf, sizeof(buf),
"ppc64%s%s%s%s%s%s%s%s%s%s%s%s%s", "ppc64%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
(has_fsqrt() ? " fsqrt" : ""), (has_fsqrt() ? " fsqrt" : ""),
(has_isel() ? " isel" : ""), (has_isel() ? " isel" : ""),
(has_lxarxeh() ? " lxarxeh" : ""), (has_lxarxeh() ? " lxarxeh" : ""),
@ -125,7 +125,8 @@ void VM_Version::initialize() {
(has_vcipher() ? " aes" : ""), (has_vcipher() ? " aes" : ""),
(has_vpmsumb() ? " vpmsumb" : ""), (has_vpmsumb() ? " vpmsumb" : ""),
(has_tcheck() ? " tcheck" : ""), (has_tcheck() ? " tcheck" : ""),
(has_mfdscr() ? " mfdscr" : "") (has_mfdscr() ? " mfdscr" : ""),
(has_vsx() ? " vsx" : "")
// Make sure number of %s matches num_features! // Make sure number of %s matches num_features!
); );
_features_string = os::strdup(buf); _features_string = os::strdup(buf);
@ -643,6 +644,7 @@ void VM_Version::determine_features() {
a->vpmsumb(VR0, VR1, VR2); // code[11] -> vpmsumb a->vpmsumb(VR0, VR1, VR2); // code[11] -> vpmsumb
a->tcheck(0); // code[12] -> tcheck a->tcheck(0); // code[12] -> tcheck
a->mfdscr(R0); // code[13] -> mfdscr a->mfdscr(R0); // code[13] -> mfdscr
a->lxvd2x(VSR0, 0, R3_ARG1); // code[14] -> vsx
a->blr(); a->blr();
// Emit function to set one cache line to zero. Emit function descriptor and get pointer to it. // Emit function to set one cache line to zero. Emit function descriptor and get pointer to it.
@ -691,6 +693,7 @@ void VM_Version::determine_features() {
if (code[feature_cntr++]) features |= vpmsumb_m; if (code[feature_cntr++]) features |= vpmsumb_m;
if (code[feature_cntr++]) features |= tcheck_m; if (code[feature_cntr++]) features |= tcheck_m;
if (code[feature_cntr++]) features |= mfdscr_m; if (code[feature_cntr++]) features |= mfdscr_m;
if (code[feature_cntr++]) features |= vsx_m;
// Print the detection code. // Print the detection code.
if (PrintAssembly) { if (PrintAssembly) {
@ -733,31 +736,31 @@ void VM_Version::config_dscr() {
} }
// Apply the configuration if needed. // Apply the configuration if needed.
uint64_t dscr_val = (*get_dscr)(); _dscr_val = (*get_dscr)();
if (Verbose) { if (Verbose) {
tty->print_cr("dscr value was 0x%lx" , dscr_val); tty->print_cr("dscr value was 0x%lx" , _dscr_val);
} }
bool change_requested = false; bool change_requested = false;
if (DSCR_PPC64 != (uintx)-1) { if (DSCR_PPC64 != (uintx)-1) {
dscr_val = DSCR_PPC64; _dscr_val = DSCR_PPC64;
change_requested = true; change_requested = true;
} }
if (DSCR_DPFD_PPC64 <= 7) { if (DSCR_DPFD_PPC64 <= 7) {
uint64_t mask = 0x7; uint64_t mask = 0x7;
if ((dscr_val & mask) != DSCR_DPFD_PPC64) { if ((_dscr_val & mask) != DSCR_DPFD_PPC64) {
dscr_val = (dscr_val & ~mask) | (DSCR_DPFD_PPC64); _dscr_val = (_dscr_val & ~mask) | (DSCR_DPFD_PPC64);
change_requested = true; change_requested = true;
} }
} }
if (DSCR_URG_PPC64 <= 7) { if (DSCR_URG_PPC64 <= 7) {
uint64_t mask = 0x7 << 6; uint64_t mask = 0x7 << 6;
if ((dscr_val & mask) != DSCR_DPFD_PPC64 << 6) { if ((_dscr_val & mask) != DSCR_DPFD_PPC64 << 6) {
dscr_val = (dscr_val & ~mask) | (DSCR_URG_PPC64 << 6); _dscr_val = (_dscr_val & ~mask) | (DSCR_URG_PPC64 << 6);
change_requested = true; change_requested = true;
} }
} }
if (change_requested) { if (change_requested) {
(*set_dscr)(dscr_val); (*set_dscr)(_dscr_val);
if (Verbose) { if (Verbose) {
tty->print_cr("dscr was set to 0x%lx" , (*get_dscr)()); tty->print_cr("dscr was set to 0x%lx" , (*get_dscr)());
} }

View File

@ -46,6 +46,7 @@ protected:
vpmsumb, vpmsumb,
tcheck, tcheck,
mfdscr, mfdscr,
vsx,
num_features // last entry to count features num_features // last entry to count features
}; };
enum Feature_Flag_Set { enum Feature_Flag_Set {
@ -64,6 +65,7 @@ protected:
vpmsumb_m = (1 << vpmsumb), vpmsumb_m = (1 << vpmsumb),
tcheck_m = (1 << tcheck ), tcheck_m = (1 << tcheck ),
mfdscr_m = (1 << mfdscr ), mfdscr_m = (1 << mfdscr ),
vsx_m = (1 << vsx ),
all_features_m = (unsigned long)-1 all_features_m = (unsigned long)-1
}; };
@ -97,10 +99,14 @@ public:
static bool has_vpmsumb() { return (_features & vpmsumb_m) != 0; } static bool has_vpmsumb() { return (_features & vpmsumb_m) != 0; }
static bool has_tcheck() { return (_features & tcheck_m) != 0; } static bool has_tcheck() { return (_features & tcheck_m) != 0; }
static bool has_mfdscr() { return (_features & mfdscr_m) != 0; } static bool has_mfdscr() { return (_features & mfdscr_m) != 0; }
static bool has_vsx() { return (_features & vsx_m) != 0; }
// Assembler testing // Assembler testing
static void allow_all(); static void allow_all();
static void revert(); static void revert();
// POWER 8: DSCR current value.
static uint64_t _dscr_val;
}; };
#endif // CPU_PPC_VM_VM_VERSION_PPC_HPP #endif // CPU_PPC_VM_VM_VERSION_PPC_HPP

View File

@ -68,7 +68,6 @@ void LinearScan::allocate_fpu_stack() {
if (b->number_of_preds() > 1) { if (b->number_of_preds() > 1) {
int id = b->first_lir_instruction_id(); int id = b->first_lir_instruction_id();
ResourceBitMap regs(FrameMap::nof_fpu_regs); ResourceBitMap regs(FrameMap::nof_fpu_regs);
regs.clear();
iw.walk_to(id); // walk after the first instruction (always a label) of the block iw.walk_to(id); // walk after the first instruction (always a label) of the block
assert(iw.current_position() == id, "did not walk completely to id"); assert(iw.current_position() == id, "did not walk completely to id");

View File

@ -7046,7 +7046,6 @@ void MacroAssembler::string_indexofC8(Register str1, Register str2,
int ae) { int ae) {
ShortBranchVerifier sbv(this); ShortBranchVerifier sbv(this);
assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required"); assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required");
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
assert(ae != StrIntrinsicNode::LU, "Invalid encoding"); assert(ae != StrIntrinsicNode::LU, "Invalid encoding");
// This method uses the pcmpestri instruction with bound registers // This method uses the pcmpestri instruction with bound registers
@ -7225,7 +7224,6 @@ void MacroAssembler::string_indexof(Register str1, Register str2,
int ae) { int ae) {
ShortBranchVerifier sbv(this); ShortBranchVerifier sbv(this);
assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required"); assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required");
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
assert(ae != StrIntrinsicNode::LU, "Invalid encoding"); assert(ae != StrIntrinsicNode::LU, "Invalid encoding");
// //
@ -7543,7 +7541,6 @@ void MacroAssembler::string_indexof_char(Register str1, Register cnt1, Register
XMMRegister vec1, XMMRegister vec2, XMMRegister vec3, Register tmp) { XMMRegister vec1, XMMRegister vec2, XMMRegister vec3, Register tmp) {
ShortBranchVerifier sbv(this); ShortBranchVerifier sbv(this);
assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required"); assert(UseSSE42Intrinsics, "SSE4.2 intrinsics are required");
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
int stride = 8; int stride = 8;
@ -7723,7 +7720,6 @@ void MacroAssembler::string_compare(Register str1, Register str2,
} }
if (UseAVX >= 2 && UseSSE42Intrinsics) { if (UseAVX >= 2 && UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
Label COMPARE_WIDE_VECTORS, VECTOR_NOT_EQUAL, COMPARE_WIDE_TAIL, COMPARE_SMALL_STR; Label COMPARE_WIDE_VECTORS, VECTOR_NOT_EQUAL, COMPARE_WIDE_TAIL, COMPARE_SMALL_STR;
Label COMPARE_WIDE_VECTORS_LOOP, COMPARE_16_CHARS, COMPARE_INDEX_CHAR; Label COMPARE_WIDE_VECTORS_LOOP, COMPARE_16_CHARS, COMPARE_INDEX_CHAR;
Label COMPARE_WIDE_VECTORS_LOOP_AVX2; Label COMPARE_WIDE_VECTORS_LOOP_AVX2;
@ -7891,7 +7887,6 @@ void MacroAssembler::string_compare(Register str1, Register str2,
bind(COMPARE_SMALL_STR); bind(COMPARE_SMALL_STR);
} else if (UseSSE42Intrinsics) { } else if (UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
Label COMPARE_WIDE_VECTORS, VECTOR_NOT_EQUAL, COMPARE_TAIL; Label COMPARE_WIDE_VECTORS, VECTOR_NOT_EQUAL, COMPARE_TAIL;
int pcmpmask = 0x19; int pcmpmask = 0x19;
// Setup to compare 8-char (16-byte) vectors, // Setup to compare 8-char (16-byte) vectors,
@ -8179,7 +8174,6 @@ void MacroAssembler::has_negatives(Register ary1, Register len,
// Fallthru to tail compare // Fallthru to tail compare
} }
else if (UseSSE42Intrinsics) { else if (UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be for SSE4.2 intrinsics to be available");
// With SSE4.2, use double quad vector compare // With SSE4.2, use double quad vector compare
Label COMPARE_WIDE_VECTORS, COMPARE_TAIL; Label COMPARE_WIDE_VECTORS, COMPARE_TAIL;
@ -8383,7 +8377,6 @@ void MacroAssembler::arrays_equals(bool is_array_equ, Register ary1, Register ar
movl(limit, result); movl(limit, result);
// Fallthru to tail compare // Fallthru to tail compare
} else if (UseSSE42Intrinsics) { } else if (UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
// With SSE4.2, use double quad vector compare // With SSE4.2, use double quad vector compare
Label COMPARE_WIDE_VECTORS, COMPARE_TAIL; Label COMPARE_WIDE_VECTORS, COMPARE_TAIL;
@ -8747,7 +8740,6 @@ void MacroAssembler::encode_iso_array(Register src, Register dst, Register len,
negptr(len); negptr(len);
if (UseSSE42Intrinsics || UseAVX >= 2) { if (UseSSE42Intrinsics || UseAVX >= 2) {
assert(UseSSE42Intrinsics ? UseSSE >= 4 : true, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
Label L_chars_8_check, L_copy_8_chars, L_copy_8_chars_exit; Label L_chars_8_check, L_copy_8_chars, L_copy_8_chars_exit;
Label L_chars_16_check, L_copy_16_chars, L_copy_16_chars_exit; Label L_chars_16_check, L_copy_16_chars, L_copy_16_chars_exit;
@ -10159,7 +10151,13 @@ void MacroAssembler::kernel_crc32(Register crc, Register buf, Register len, Regi
movdqa(xmm1, Address(buf, 0)); movdqa(xmm1, Address(buf, 0));
movdl(rax, xmm1); movdl(rax, xmm1);
xorl(crc, rax); xorl(crc, rax);
if (VM_Version::supports_sse4_1()) {
pinsrd(xmm1, crc, 0); pinsrd(xmm1, crc, 0);
} else {
pinsrw(xmm1, crc, 0);
shrl(crc, 16);
pinsrw(xmm1, crc, 1);
}
addptr(buf, 16); addptr(buf, 16);
subl(len, 4); // len > 0 subl(len, 4); // len > 0
jcc(Assembler::less, L_fold_tail); jcc(Assembler::less, L_fold_tail);
@ -10881,7 +10879,6 @@ void MacroAssembler::char_array_compress(Register src, Register dst, Register le
clear_vector_masking(); // closing of the stub context for programming mask registers clear_vector_masking(); // closing of the stub context for programming mask registers
} }
if (UseSSE42Intrinsics) { if (UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
Label copy_32_loop, copy_16, copy_tail; Label copy_32_loop, copy_16, copy_tail;
bind(below_threshold); bind(below_threshold);
@ -11045,7 +11042,6 @@ void MacroAssembler::byte_array_inflate(Register src, Register dst, Register len
clear_vector_masking(); // closing of the stub context for programming mask registers clear_vector_masking(); // closing of the stub context for programming mask registers
} }
if (UseSSE42Intrinsics) { if (UseSSE42Intrinsics) {
assert(UseSSE >= 4, "SSE4 must be enabled for SSE4.2 intrinsics to be available");
Label copy_16_loop, copy_8_loop, copy_bytes, copy_new_tail, copy_tail; Label copy_16_loop, copy_8_loop, copy_bytes, copy_new_tail, copy_tail;
movl(tmp2, len); movl(tmp2, len);

View File

@ -658,7 +658,7 @@ void VM_Version::get_processor_features() {
FLAG_SET_DEFAULT(UseAESCTRIntrinsics, false); FLAG_SET_DEFAULT(UseAESCTRIntrinsics, false);
} }
} else { } else {
if(supports_sse4_1() && UseSSE >= 4) { if(supports_sse4_1()) {
if (FLAG_IS_DEFAULT(UseAESCTRIntrinsics)) { if (FLAG_IS_DEFAULT(UseAESCTRIntrinsics)) {
FLAG_SET_DEFAULT(UseAESCTRIntrinsics, true); FLAG_SET_DEFAULT(UseAESCTRIntrinsics, true);
} }
@ -970,7 +970,7 @@ void VM_Version::get_processor_features() {
UseXmmI2D = false; UseXmmI2D = false;
} }
} }
if (supports_sse4_2() && UseSSE >= 4) { if (supports_sse4_2()) {
if (FLAG_IS_DEFAULT(UseSSE42Intrinsics)) { if (FLAG_IS_DEFAULT(UseSSE42Intrinsics)) {
FLAG_SET_DEFAULT(UseSSE42Intrinsics, true); FLAG_SET_DEFAULT(UseSSE42Intrinsics, true);
} }
@ -1050,7 +1050,7 @@ void VM_Version::get_processor_features() {
UseUnalignedLoadStores = true; // use movdqu on newest Intel cpus UseUnalignedLoadStores = true; // use movdqu on newest Intel cpus
} }
} }
if (supports_sse4_2() && UseSSE >= 4) { if (supports_sse4_2()) {
if (FLAG_IS_DEFAULT(UseSSE42Intrinsics)) { if (FLAG_IS_DEFAULT(UseSSE42Intrinsics)) {
FLAG_SET_DEFAULT(UseSSE42Intrinsics, true); FLAG_SET_DEFAULT(UseSSE42Intrinsics, true);
} }

View File

@ -1718,7 +1718,7 @@ const bool Matcher::match_rule_supported(int opcode) {
ret_value = false; ret_value = false;
break; break;
case Op_StrIndexOfChar: case Op_StrIndexOfChar:
if (!(UseSSE > 4)) if (!UseSSE42Intrinsics)
ret_value = false; ret_value = false;
break; break;
case Op_OnSpinWait: case Op_OnSpinWait:

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2000, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2000, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -96,6 +96,8 @@ public class Universe {
narrowOopShiftField = type.getCIntegerField("_narrow_oop._shift"); narrowOopShiftField = type.getCIntegerField("_narrow_oop._shift");
narrowKlassBaseField = type.getAddressField("_narrow_klass._base"); narrowKlassBaseField = type.getAddressField("_narrow_klass._base");
narrowKlassShiftField = type.getCIntegerField("_narrow_klass._shift"); narrowKlassShiftField = type.getCIntegerField("_narrow_klass._shift");
UniverseExt.initialize(heapConstructor);
} }
public Universe() { public Universe() {

View File

@ -0,0 +1,31 @@
/*
* Copyright (c) 2016, 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 sun.jvm.hotspot.memory;
import sun.jvm.hotspot.runtime.*;
public class UniverseExt {
public static void initialize(VirtualConstructor heapConstructor) { }
}

View File

@ -31,10 +31,13 @@ import java.lang.reflect.Array;
import java.lang.reflect.Field; import java.lang.reflect.Field;
import java.lang.reflect.Method; import java.lang.reflect.Method;
import java.lang.reflect.Modifier; import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.ServiceLoader;
import java.util.TreeMap; import java.util.TreeMap;
import jdk.internal.misc.VM; import jdk.internal.misc.VM;
@ -213,7 +216,22 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
private final Map<Class<? extends Architecture>, JVMCIBackend> backends = new HashMap<>(); private final Map<Class<? extends Architecture>, JVMCIBackend> backends = new HashMap<>();
private final Iterable<HotSpotVMEventListener> vmEventListeners; private volatile List<HotSpotVMEventListener> vmEventListeners;
private Iterable<HotSpotVMEventListener> getVmEventListeners() {
if (vmEventListeners == null) {
synchronized (this) {
if (vmEventListeners == null) {
List<HotSpotVMEventListener> listeners = new ArrayList<>();
for (HotSpotVMEventListener vmEventListener : ServiceLoader.load(HotSpotVMEventListener.class)) {
listeners.add(vmEventListener);
}
vmEventListeners = listeners;
}
}
}
return vmEventListeners;
}
/** /**
* Stores the result of {@link HotSpotJVMCICompilerFactory#getTrivialPrefixes()} so that it can * Stores the result of {@link HotSpotJVMCICompilerFactory#getTrivialPrefixes()} so that it can
@ -240,8 +258,6 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
hostBackend = registerBackend(factory.createJVMCIBackend(this, null)); hostBackend = registerBackend(factory.createJVMCIBackend(this, null));
} }
vmEventListeners = Services.load(HotSpotVMEventListener.class);
metaAccessContext = new HotSpotJVMCIMetaAccessContext(); metaAccessContext = new HotSpotJVMCIMetaAccessContext();
boolean printFlags = Option.PrintFlags.getBoolean(); boolean printFlags = Option.PrintFlags.getBoolean();
@ -370,7 +386,7 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
*/ */
@SuppressWarnings({"unused"}) @SuppressWarnings({"unused"})
private void shutdown() throws Exception { private void shutdown() throws Exception {
for (HotSpotVMEventListener vmEventListener : vmEventListeners) { for (HotSpotVMEventListener vmEventListener : getVmEventListeners()) {
vmEventListener.notifyShutdown(); vmEventListener.notifyShutdown();
} }
} }
@ -382,7 +398,7 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
*/ */
@SuppressWarnings({"unused"}) @SuppressWarnings({"unused"})
private void bootstrapFinished() throws Exception { private void bootstrapFinished() throws Exception {
for (HotSpotVMEventListener vmEventListener : vmEventListeners) { for (HotSpotVMEventListener vmEventListener : getVmEventListeners()) {
vmEventListener.notifyBootstrapFinished(); vmEventListener.notifyBootstrapFinished();
} }
} }
@ -395,7 +411,7 @@ public final class HotSpotJVMCIRuntime implements HotSpotJVMCIRuntimeProvider {
* @param compiledCode * @param compiledCode
*/ */
void notifyInstall(HotSpotCodeCacheProvider hotSpotCodeCacheProvider, InstalledCode installedCode, CompiledCode compiledCode) { void notifyInstall(HotSpotCodeCacheProvider hotSpotCodeCacheProvider, InstalledCode installedCode, CompiledCode compiledCode) {
for (HotSpotVMEventListener vmEventListener : vmEventListeners) { for (HotSpotVMEventListener vmEventListener : getVmEventListeners()) {
vmEventListener.notifyInstall(hotSpotCodeCacheProvider, installedCode, compiledCode); vmEventListener.notifyInstall(hotSpotCodeCacheProvider, installedCode, compiledCode);
} }
} }

View File

@ -144,8 +144,18 @@ class HotSpotMemoryAccessProviderImpl implements HotSpotMemoryAccessProvider {
return ret; return ret;
} }
@Override /**
public JavaConstant readUnsafeConstant(JavaKind kind, JavaConstant baseConstant, long displacement) { * Reads a value of this kind using a base address and a displacement. No bounds checking or
* type checking is performed. Returns {@code null} if the value is not available at this point.
*
* @param baseConstant the base address from which the value is read.
* @param displacement the displacement within the object in bytes
* @return the read value encapsulated in a {@link JavaConstant} object, or {@code null} if the
* value cannot be read.
* @throws IllegalArgumentException if {@code kind} is {@code null}, {@link JavaKind#Void}, not
* {@link JavaKind#Object} or not {@linkplain JavaKind#isPrimitive() primitive} kind
*/
JavaConstant readUnsafeConstant(JavaKind kind, JavaConstant baseConstant, long displacement) {
if (kind == null) { if (kind == null) {
throw new IllegalArgumentException("null JavaKind"); throw new IllegalArgumentException("null JavaKind");
} }

View File

@ -27,19 +27,6 @@ package jdk.vm.ci.meta;
*/ */
public interface MemoryAccessProvider { public interface MemoryAccessProvider {
/**
* Reads a value of this kind using a base address and a displacement. No bounds checking or
* type checking is performed. Returns {@code null} if the value is not available at this point.
*
* @param base the base address from which the value is read.
* @param displacement the displacement within the object in bytes
* @return the read value encapsulated in a {@link JavaConstant} object, or {@code null} if the
* value cannot be read.
* @throws IllegalArgumentException if {@code kind} is {@code null}, {@link JavaKind#Void}, not
* {@link JavaKind#Object} or not {@linkplain JavaKind#isPrimitive() primitive} kind
*/
JavaConstant readUnsafeConstant(JavaKind kind, JavaConstant base, long displacement) throws IllegalArgumentException;
/** /**
* Reads a primitive value using a base address and a displacement. * Reads a primitive value using a base address and a displacement.
* *

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1999, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1999, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -264,7 +264,7 @@ void Canonicalizer::do_LoadIndexed (LoadIndexed* x) {
assert(array == NULL || FoldStableValues, "not enabled"); assert(array == NULL || FoldStableValues, "not enabled");
// Constant fold loads from stable arrays. // Constant fold loads from stable arrays.
if (array != NULL && index != NULL) { if (!x->mismatched() && array != NULL && index != NULL) {
jint idx = index->value(); jint idx = index->value();
if (idx < 0 || idx >= array->value()->length()) { if (idx < 0 || idx >= array->value()->length()) {
// Leave the load as is. The range check will handle it. // Leave the load as is. The range check will handle it.
@ -310,8 +310,6 @@ void Canonicalizer::do_StoreIndexed (StoreIndexed* x) {
return; return;
} }
} }
} }

View File

@ -4227,11 +4227,11 @@ void GraphBuilder::append_char_access(ciMethod* callee, bool is_store) {
Value index = args->at(1); Value index = args->at(1);
if (is_store) { if (is_store) {
Value value = args->at(2); Value value = args->at(2);
Instruction* store = append(new StoreIndexed(array, index, NULL, T_CHAR, value, state_before, false)); Instruction* store = append(new StoreIndexed(array, index, NULL, T_CHAR, value, state_before, false, true));
store->set_flag(Instruction::NeedsRangeCheckFlag, false); store->set_flag(Instruction::NeedsRangeCheckFlag, false);
_memory->store_value(value); _memory->store_value(value);
} else { } else {
Instruction* load = append(new LoadIndexed(array, index, NULL, T_CHAR, state_before)); Instruction* load = append(new LoadIndexed(array, index, NULL, T_CHAR, state_before, true));
load->set_flag(Instruction::NeedsRangeCheckFlag, false); load->set_flag(Instruction::NeedsRangeCheckFlag, false);
push(load->type(), load); push(load->type(), load);
} }

View File

@ -147,9 +147,7 @@ IRScope::IRScope(Compilation* compilation, IRScope* caller, int caller_bci, ciMe
_wrote_volatile = false; _wrote_volatile = false;
_start = NULL; _start = NULL;
if (osr_bci == -1) { if (osr_bci != -1) {
_requires_phi_function.clear();
} else {
// selective creation of phi functions is not possibel in osr-methods // selective creation of phi functions is not possibel in osr-methods
_requires_phi_function.set_range(0, method->max_locals()); _requires_phi_function.set_range(0, method->max_locals());
} }
@ -540,7 +538,6 @@ ComputeLinearScanOrder::ComputeLinearScanOrder(Compilation* c, BlockBegin* start
{ {
TRACE_LINEAR_SCAN(2, tty->print_cr("***** computing linear-scan block order")); TRACE_LINEAR_SCAN(2, tty->print_cr("***** computing linear-scan block order"));
init_visited();
count_edges(start_block, NULL); count_edges(start_block, NULL);
if (compilation()->is_profiling()) { if (compilation()->is_profiling()) {
@ -646,7 +643,6 @@ void ComputeLinearScanOrder::mark_loops() {
TRACE_LINEAR_SCAN(3, tty->print_cr("----- marking loops")); TRACE_LINEAR_SCAN(3, tty->print_cr("----- marking loops"));
_loop_map = BitMap2D(_num_loops, _max_block_id); _loop_map = BitMap2D(_num_loops, _max_block_id);
_loop_map.clear();
for (int i = _loop_end_blocks.length() - 1; i >= 0; i--) { for (int i = _loop_end_blocks.length() - 1; i >= 0; i--) {
BlockBegin* loop_end = _loop_end_blocks.at(i); BlockBegin* loop_end = _loop_end_blocks.at(i);

View File

@ -912,14 +912,16 @@ BASE(AccessIndexed, AccessArray)
Value _index; Value _index;
Value _length; Value _length;
BasicType _elt_type; BasicType _elt_type;
bool _mismatched;
public: public:
// creation // creation
AccessIndexed(Value array, Value index, Value length, BasicType elt_type, ValueStack* state_before) AccessIndexed(Value array, Value index, Value length, BasicType elt_type, ValueStack* state_before, bool mismatched)
: AccessArray(as_ValueType(elt_type), array, state_before) : AccessArray(as_ValueType(elt_type), array, state_before)
, _index(index) , _index(index)
, _length(length) , _length(length)
, _elt_type(elt_type) , _elt_type(elt_type)
, _mismatched(mismatched)
{ {
set_flag(Instruction::NeedsRangeCheckFlag, true); set_flag(Instruction::NeedsRangeCheckFlag, true);
ASSERT_VALUES ASSERT_VALUES
@ -929,6 +931,7 @@ BASE(AccessIndexed, AccessArray)
Value index() const { return _index; } Value index() const { return _index; }
Value length() const { return _length; } Value length() const { return _length; }
BasicType elt_type() const { return _elt_type; } BasicType elt_type() const { return _elt_type; }
bool mismatched() const { return _mismatched; }
void clear_length() { _length = NULL; } void clear_length() { _length = NULL; }
// perform elimination of range checks involving constants // perform elimination of range checks involving constants
@ -945,8 +948,8 @@ LEAF(LoadIndexed, AccessIndexed)
public: public:
// creation // creation
LoadIndexed(Value array, Value index, Value length, BasicType elt_type, ValueStack* state_before) LoadIndexed(Value array, Value index, Value length, BasicType elt_type, ValueStack* state_before, bool mismatched = false)
: AccessIndexed(array, index, length, elt_type, state_before) : AccessIndexed(array, index, length, elt_type, state_before, mismatched)
, _explicit_null_check(NULL) {} , _explicit_null_check(NULL) {}
// accessors // accessors
@ -974,8 +977,9 @@ LEAF(StoreIndexed, AccessIndexed)
public: public:
// creation // creation
StoreIndexed(Value array, Value index, Value length, BasicType elt_type, Value value, ValueStack* state_before, bool check_boolean) StoreIndexed(Value array, Value index, Value length, BasicType elt_type, Value value, ValueStack* state_before,
: AccessIndexed(array, index, length, elt_type, state_before) bool check_boolean, bool mismatched = false)
: AccessIndexed(array, index, length, elt_type, state_before, mismatched)
, _value(value), _profiled_method(NULL), _profiled_bci(0), _check_boolean(check_boolean) , _value(value), _profiled_method(NULL), _profiled_bci(0), _check_boolean(check_boolean)
{ {
set_flag(NeedsWriteBarrierFlag, (as_ValueType(elt_type)->is_object())); set_flag(NeedsWriteBarrierFlag, (as_ValueType(elt_type)->is_object()));

View File

@ -1387,7 +1387,6 @@ Instruction* LIRGenerator::instruction_for_vreg(int reg_num) {
void LIRGenerator::set_vreg_flag(int vreg_num, VregFlag f) { void LIRGenerator::set_vreg_flag(int vreg_num, VregFlag f) {
if (_vreg_flags.size_in_bits() == 0) { if (_vreg_flags.size_in_bits() == 0) {
BitMap2D temp(100, num_vreg_flags); BitMap2D temp(100, num_vreg_flags);
temp.clear();
_vreg_flags = temp; _vreg_flags = temp;
} }
_vreg_flags.at_put_grow(vreg_num, f, true); _vreg_flags.at_put_grow(vreg_num, f, true);

View File

@ -562,14 +562,13 @@ void LinearScan::compute_local_live_sets() {
LIR_OpVisitState visitor; LIR_OpVisitState visitor;
BitMap2D local_interval_in_loop = BitMap2D(_num_virtual_regs, num_loops()); BitMap2D local_interval_in_loop = BitMap2D(_num_virtual_regs, num_loops());
local_interval_in_loop.clear();
// iterate all blocks // iterate all blocks
for (int i = 0; i < num_blocks; i++) { for (int i = 0; i < num_blocks; i++) {
BlockBegin* block = block_at(i); BlockBegin* block = block_at(i);
ResourceBitMap live_gen(live_size); live_gen.clear(); ResourceBitMap live_gen(live_size);
ResourceBitMap live_kill(live_size); live_kill.clear(); ResourceBitMap live_kill(live_size);
if (block->is_set(BlockBegin::exception_entry_flag)) { if (block->is_set(BlockBegin::exception_entry_flag)) {
// Phi functions at the begin of an exception handler are // Phi functions at the begin of an exception handler are
@ -715,8 +714,8 @@ void LinearScan::compute_local_live_sets() {
block->set_live_gen (live_gen); block->set_live_gen (live_gen);
block->set_live_kill(live_kill); block->set_live_kill(live_kill);
block->set_live_in (ResourceBitMap(live_size)); block->live_in().clear(); block->set_live_in (ResourceBitMap(live_size));
block->set_live_out (ResourceBitMap(live_size)); block->live_out().clear(); block->set_live_out (ResourceBitMap(live_size));
TRACE_LINEAR_SCAN(4, tty->print("live_gen B%d ", block->block_id()); print_bitmap(block->live_gen())); TRACE_LINEAR_SCAN(4, tty->print("live_gen B%d ", block->block_id()); print_bitmap(block->live_gen()));
TRACE_LINEAR_SCAN(4, tty->print("live_kill B%d ", block->block_id()); print_bitmap(block->live_kill())); TRACE_LINEAR_SCAN(4, tty->print("live_kill B%d ", block->block_id()); print_bitmap(block->live_kill()));
@ -741,7 +740,7 @@ void LinearScan::compute_global_live_sets() {
bool change_occurred; bool change_occurred;
bool change_occurred_in_block; bool change_occurred_in_block;
int iteration_count = 0; int iteration_count = 0;
ResourceBitMap live_out(live_set_size()); live_out.clear(); // scratch set for calculations ResourceBitMap live_out(live_set_size()); // scratch set for calculations
// Perform a backward dataflow analysis to compute live_out and live_in for each block. // Perform a backward dataflow analysis to compute live_out and live_in for each block.
// The loop is executed until a fixpoint is reached (no changes in an iteration) // The loop is executed until a fixpoint is reached (no changes in an iteration)
@ -827,7 +826,6 @@ void LinearScan::compute_global_live_sets() {
// check that the live_in set of the first block is empty // check that the live_in set of the first block is empty
ResourceBitMap live_in_args(ir()->start()->live_in().size()); ResourceBitMap live_in_args(ir()->start()->live_in().size());
live_in_args.clear();
if (!ir()->start()->live_in().is_same(live_in_args)) { if (!ir()->start()->live_in().is_same(live_in_args)) {
#ifdef ASSERT #ifdef ASSERT
tty->print_cr("Error: live_in set of first block must be empty (when this fails, virtual registers are used before they are defined)"); tty->print_cr("Error: live_in set of first block must be empty (when this fails, virtual registers are used before they are defined)");
@ -1774,8 +1772,8 @@ void LinearScan::resolve_data_flow() {
int num_blocks = block_count(); int num_blocks = block_count();
MoveResolver move_resolver(this); MoveResolver move_resolver(this);
ResourceBitMap block_completed(num_blocks); block_completed.clear(); ResourceBitMap block_completed(num_blocks);
ResourceBitMap already_resolved(num_blocks); already_resolved.clear(); ResourceBitMap already_resolved(num_blocks);
int i; int i;
for (i = 0; i < num_blocks; i++) { for (i = 0; i < num_blocks; i++) {
@ -3750,7 +3748,6 @@ void MoveResolver::verify_before_resolve() {
ResourceBitMap used_regs(LinearScan::nof_regs + allocator()->frame_map()->argcount() + allocator()->max_spills()); ResourceBitMap used_regs(LinearScan::nof_regs + allocator()->frame_map()->argcount() + allocator()->max_spills());
used_regs.clear();
if (!_multiple_reads_allowed) { if (!_multiple_reads_allowed) {
for (i = 0; i < _mapping_from.length(); i++) { for (i = 0; i < _mapping_from.length(); i++) {
Interval* it = _mapping_from.at(i); Interval* it = _mapping_from.at(i);
@ -6319,7 +6316,6 @@ void ControlFlowOptimizer::delete_unnecessary_jumps(BlockList* code) {
void ControlFlowOptimizer::delete_jumps_to_return(BlockList* code) { void ControlFlowOptimizer::delete_jumps_to_return(BlockList* code) {
#ifdef ASSERT #ifdef ASSERT
ResourceBitMap return_converted(BlockBegin::number_of_blocks()); ResourceBitMap return_converted(BlockBegin::number_of_blocks());
return_converted.clear();
#endif #endif
for (int i = code->length() - 1; i >= 0; i--) { for (int i = code->length() - 1; i >= 0; i--) {

View File

@ -53,7 +53,6 @@ class ValueSet: public CompilationResourceObj {
}; };
inline ValueSet::ValueSet() : _map(Instruction::number_of_instructions()) { inline ValueSet::ValueSet() : _map(Instruction::number_of_instructions()) {
_map.clear();
} }

View File

@ -449,7 +449,6 @@ ResourceBitMap ciMethod::live_local_oops_at_bci(int bci) {
OopMapCache::compute_one_oop_map(get_Method(), bci, &mask); OopMapCache::compute_one_oop_map(get_Method(), bci, &mask);
int mask_size = max_locals(); int mask_size = max_locals();
ResourceBitMap result(mask_size); ResourceBitMap result(mask_size);
result.clear();
int i; int i;
for (i = 0; i < mask_size ; i++ ) { for (i = 0; i < mask_size ; i++ ) {
if (mask.is_oop(i)) result.set_bit(i); if (mask.is_oop(i)) result.set_bit(i);

View File

@ -4673,6 +4673,7 @@ void ClassFileParser::verify_legal_utf8(const unsigned char* buffer,
} }
// Unqualified names may not contain the characters '.', ';', '[', or '/'. // Unqualified names may not contain the characters '.', ';', '[', or '/'.
// In class names, '/' separates unqualified names. This is verified in this function also.
// Method names also may not contain the characters '<' or '>', unless <init> // Method names also may not contain the characters '<' or '>', unless <init>
// or <clinit>. Note that method names may not be <init> or <clinit> in this // or <clinit>. Note that method names may not be <init> or <clinit> in this
// method. Because these names have been checked as special cases before // method. Because these names have been checked as special cases before
@ -4698,9 +4699,17 @@ bool ClassFileParser::verify_unqualified_name(const char* name,
if (ch == ';' || ch == '[' ) { if (ch == ';' || ch == '[' ) {
return false; // do not permit '.', ';', or '[' return false; // do not permit '.', ';', or '['
} }
if (type != ClassFileParser::LegalClass && ch == '/') { if (ch == '/') {
// check for '//' or leading or trailing '/' which are not legal
// unqualified name must not be empty
if (type == ClassFileParser::LegalClass) {
if (p == name || p+1 >= name+length || *(p+1) == '/') {
return false;
}
} else {
return false; // do not permit '/' unless it's class name return false; // do not permit '/' unless it's class name
} }
}
if (type == ClassFileParser::LegalMethod && (ch == '<' || ch == '>')) { if (type == ClassFileParser::LegalMethod && (ch == '<' || ch == '>')) {
return false; // do not permit '<' or '>' in method names return false; // do not permit '<' or '>' in method names
} }
@ -5347,15 +5356,7 @@ void ClassFileParser::fill_instance_klass(InstanceKlass* ik, bool changed_by_loa
if (!is_internal()) { if (!is_internal()) {
if (log_is_enabled(Info, class, load)) { if (log_is_enabled(Info, class, load)) {
ResourceMark rm; ResourceMark rm;
const char* module_name = NULL; const char* module_name = (module_entry->name() == NULL) ? UNNAMED_MODULE : module_entry->name()->as_C_string();
static const size_t modules_image_name_len = strlen(MODULES_IMAGE_NAME);
size_t stream_len = strlen(_stream->source());
// See if _stream->source() ends in "modules"
if (module_entry->is_named() && modules_image_name_len < stream_len &&
(strncmp(_stream->source() + stream_len - modules_image_name_len,
MODULES_IMAGE_NAME, modules_image_name_len) == 0)) {
module_name = module_entry->name()->as_C_string();
}
if (log_is_enabled(Info, class, load)) { if (log_is_enabled(Info, class, load)) {
ik->print_loading_log(LogLevel::Info, _loader_data, module_name, _stream); ik->print_loading_log(LogLevel::Info, _loader_data, module_name, _stream);

View File

@ -181,26 +181,59 @@ bool ClassLoader::string_ends_with(const char* str, const char* str_to_find) {
} }
// Used to obtain the package name from a fully qualified class name. // Used to obtain the package name from a fully qualified class name.
// It is the responsibility of the caller to establish ResourceMark. // It is the responsibility of the caller to establish a ResourceMark.
const char* ClassLoader::package_from_name(const char* class_name) { const char* ClassLoader::package_from_name(const char* const class_name, bool* bad_class_name) {
const char* last_slash = strrchr(class_name, '/'); if (class_name == NULL) {
if (bad_class_name != NULL) {
*bad_class_name = true;
}
return NULL;
}
if (bad_class_name != NULL) {
*bad_class_name = false;
}
const char* const last_slash = strrchr(class_name, '/');
if (last_slash == NULL) { if (last_slash == NULL) {
// No package name // No package name
return NULL; return NULL;
} }
int length = last_slash - class_name;
// A class name could have just the slash character in the name, char* class_name_ptr = (char*) class_name;
// resulting in a negative length. // Skip over '['s
if (*class_name_ptr == '[') {
do {
class_name_ptr++;
} while (*class_name_ptr == '[');
// Fully qualified class names should not contain a 'L'.
// Set bad_class_name to true to indicate that the package name
// could not be obtained due to an error condition.
// In this situation, is_same_class_package returns false.
if (*class_name_ptr == 'L') {
if (bad_class_name != NULL) {
*bad_class_name = true;
}
return NULL;
}
}
int length = last_slash - class_name_ptr;
// A class name could have just the slash character in the name.
if (length <= 0) { if (length <= 0) {
// No package name // No package name
if (bad_class_name != NULL) {
*bad_class_name = true;
}
return NULL; return NULL;
} }
// drop name after last slash (including slash) // drop name after last slash (including slash)
// Ex., "java/lang/String.class" => "java/lang" // Ex., "java/lang/String.class" => "java/lang"
char* pkg_name = NEW_RESOURCE_ARRAY(char, length + 1); char* pkg_name = NEW_RESOURCE_ARRAY(char, length + 1);
strncpy(pkg_name, class_name, length); strncpy(pkg_name, class_name_ptr, length);
*(pkg_name+length) = '\0'; *(pkg_name+length) = '\0';
return (const char *)pkg_name; return (const char *)pkg_name;
@ -228,8 +261,9 @@ ClassPathDirEntry::ClassPathDirEntry(const char* dir) : ClassPathEntry() {
ClassFileStream* ClassPathDirEntry::open_stream(const char* name, TRAPS) { ClassFileStream* ClassPathDirEntry::open_stream(const char* name, TRAPS) {
// construct full path name // construct full path name
char path[JVM_MAXPATHLEN]; char* path = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, JVM_MAXPATHLEN);
if (jio_snprintf(path, sizeof(path), "%s%s%s", _dir, os::file_separator(), name) == -1) { if (jio_snprintf(path, JVM_MAXPATHLEN, "%s%s%s", _dir, os::file_separator(), name) == -1) {
FREE_RESOURCE_ARRAY(char, path, JVM_MAXPATHLEN);
return NULL; return NULL;
} }
// check if file exists // check if file exists
@ -256,6 +290,7 @@ ClassFileStream* ClassPathDirEntry::open_stream(const char* name, TRAPS) {
if (UsePerfData) { if (UsePerfData) {
ClassLoader::perf_sys_classfile_bytes_read()->inc(num_read); ClassLoader::perf_sys_classfile_bytes_read()->inc(num_read);
} }
FREE_RESOURCE_ARRAY(char, path, JVM_MAXPATHLEN);
// Resource allocated // Resource allocated
return new ClassFileStream(buffer, return new ClassFileStream(buffer,
st.st_size, st.st_size,
@ -264,6 +299,7 @@ ClassFileStream* ClassPathDirEntry::open_stream(const char* name, TRAPS) {
} }
} }
} }
FREE_RESOURCE_ARRAY(char, path, JVM_MAXPATHLEN);
return NULL; return NULL;
} }
@ -344,9 +380,9 @@ u1* ClassPathZipEntry::open_versioned_entry(const char* name, jint* filesize, TR
if (is_multi_ver) { if (is_multi_ver) {
int n; int n;
char entry_name[JVM_MAXPATHLEN]; char* entry_name = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, JVM_MAXPATHLEN);
if (version > 0) { if (version > 0) {
n = jio_snprintf(entry_name, sizeof(entry_name), "META-INF/versions/%d/%s", version, name); n = jio_snprintf(entry_name, JVM_MAXPATHLEN, "META-INF/versions/%d/%s", version, name);
entry_name[n] = '\0'; entry_name[n] = '\0';
buffer = open_entry((const char*)entry_name, filesize, false, CHECK_NULL); buffer = open_entry((const char*)entry_name, filesize, false, CHECK_NULL);
if (buffer == NULL) { if (buffer == NULL) {
@ -355,7 +391,7 @@ u1* ClassPathZipEntry::open_versioned_entry(const char* name, jint* filesize, TR
} }
if (buffer == NULL) { if (buffer == NULL) {
for (int i = cur_ver; i >= base_version; i--) { for (int i = cur_ver; i >= base_version; i--) {
n = jio_snprintf(entry_name, sizeof(entry_name), "META-INF/versions/%d/%s", i, name); n = jio_snprintf(entry_name, JVM_MAXPATHLEN, "META-INF/versions/%d/%s", i, name);
entry_name[n] = '\0'; entry_name[n] = '\0';
buffer = open_entry((const char*)entry_name, filesize, false, CHECK_NULL); buffer = open_entry((const char*)entry_name, filesize, false, CHECK_NULL);
if (buffer != NULL) { if (buffer != NULL) {
@ -363,6 +399,7 @@ u1* ClassPathZipEntry::open_versioned_entry(const char* name, jint* filesize, TR
} }
} }
} }
FREE_RESOURCE_ARRAY(char, entry_name, JVM_MAXPATHLEN);
} }
} }
return buffer; return buffer;
@ -508,7 +545,8 @@ bool ctw_visitor(JImageFile* jimage,
const char* name, const char* extension, void* arg) { const char* name, const char* extension, void* arg) {
if (strcmp(extension, "class") == 0) { if (strcmp(extension, "class") == 0) {
Thread* THREAD = Thread::current(); Thread* THREAD = Thread::current();
char path[JIMAGE_MAX_PATH]; ResourceMark rm(THREAD);
char* path = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, JIMAGE_MAX_PATH);
jio_snprintf(path, JIMAGE_MAX_PATH - 1, "%s/%s.class", package, name); jio_snprintf(path, JIMAGE_MAX_PATH - 1, "%s/%s.class", package, name);
ClassLoader::compile_the_world_in(path, *(Handle*)arg, THREAD); ClassLoader::compile_the_world_in(path, *(Handle*)arg, THREAD);
return !HAS_PENDING_EXCEPTION; return !HAS_PENDING_EXCEPTION;
@ -750,9 +788,10 @@ ClassPathEntry* ClassLoader::create_class_path_entry(const char *path, const str
JavaThread* thread = JavaThread::current(); JavaThread* thread = JavaThread::current();
ClassPathEntry* new_entry = NULL; ClassPathEntry* new_entry = NULL;
if ((st->st_mode & S_IFREG) == S_IFREG) { if ((st->st_mode & S_IFREG) == S_IFREG) {
ResourceMark rm(thread);
// Regular file, should be a zip or jimage file // Regular file, should be a zip or jimage file
// Canonicalized filename // Canonicalized filename
char canonical_path[JVM_MAXPATHLEN]; char* canonical_path = NEW_RESOURCE_ARRAY_IN_THREAD(thread, char, JVM_MAXPATHLEN);
if (!get_canonical_path(path, canonical_path, JVM_MAXPATHLEN)) { if (!get_canonical_path(path, canonical_path, JVM_MAXPATHLEN)) {
// This matches the classic VM // This matches the classic VM
if (throw_exception) { if (throw_exception) {
@ -777,14 +816,13 @@ ClassPathEntry* ClassLoader::create_class_path_entry(const char *path, const str
if (zip != NULL && error_msg == NULL) { if (zip != NULL && error_msg == NULL) {
new_entry = new ClassPathZipEntry(zip, path, is_boot_append); new_entry = new ClassPathZipEntry(zip, path, is_boot_append);
} else { } else {
ResourceMark rm(thread);
char *msg; char *msg;
if (error_msg == NULL) { if (error_msg == NULL) {
msg = NEW_RESOURCE_ARRAY(char, strlen(path) + 128); ; msg = NEW_RESOURCE_ARRAY_IN_THREAD(thread, char, strlen(path) + 128); ;
jio_snprintf(msg, strlen(path) + 127, "error in opening JAR file %s", path); jio_snprintf(msg, strlen(path) + 127, "error in opening JAR file %s", path);
} else { } else {
int len = (int)(strlen(path) + strlen(error_msg) + 128); int len = (int)(strlen(path) + strlen(error_msg) + 128);
msg = NEW_RESOURCE_ARRAY(char, len); ; msg = NEW_RESOURCE_ARRAY_IN_THREAD(thread, char, len); ;
jio_snprintf(msg, len - 1, "error in opening JAR file <%s> %s", error_msg, path); jio_snprintf(msg, len - 1, "error in opening JAR file <%s> %s", error_msg, path);
} }
// Don't complain about bad jar files added via -Xbootclasspath/a:. // Don't complain about bad jar files added via -Xbootclasspath/a:.
@ -1112,13 +1150,11 @@ bool ClassLoader::add_package(const char *fullq_class_name, s2 classpath_index,
assert(fullq_class_name != NULL, "just checking"); assert(fullq_class_name != NULL, "just checking");
// Get package name from fully qualified class name. // Get package name from fully qualified class name.
const char *cp = strrchr(fullq_class_name, '/'); ResourceMark rm;
const char *cp = package_from_name(fullq_class_name);
if (cp != NULL) { if (cp != NULL) {
int len = cp - fullq_class_name; PackageEntryTable* pkg_entry_tbl = ClassLoaderData::the_null_class_loader_data()->packages();
PackageEntryTable* pkg_entry_tbl = TempNewSymbol pkg_symbol = SymbolTable::new_symbol(cp, CHECK_false);
ClassLoaderData::the_null_class_loader_data()->packages();
TempNewSymbol pkg_symbol =
SymbolTable::new_symbol(fullq_class_name, len, CHECK_false);
PackageEntry* pkg_entry = pkg_entry_tbl->lookup_only(pkg_symbol); PackageEntry* pkg_entry = pkg_entry_tbl->lookup_only(pkg_symbol);
if (pkg_entry != NULL) { if (pkg_entry != NULL) {
assert(classpath_index != -1, "Unexpected classpath_index"); assert(classpath_index != -1, "Unexpected classpath_index");
@ -1226,11 +1262,9 @@ s2 ClassLoader::classloader_type(Symbol* class_name, ClassPathEntry* e, int clas
// jimage, it is determined by the class path entry. // jimage, it is determined by the class path entry.
jshort loader_type = ClassLoader::APP_LOADER; jshort loader_type = ClassLoader::APP_LOADER;
if (e->is_jrt()) { if (e->is_jrt()) {
int length = 0;
const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length);
if (pkg_string != NULL) {
ResourceMark rm; ResourceMark rm;
TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)pkg_string, length, THREAD); TempNewSymbol pkg_name = InstanceKlass::package_from_name(class_name, CHECK_0);
if (pkg_name != NULL) {
const char* pkg_name_C_string = (const char*)(pkg_name->as_C_string()); const char* pkg_name_C_string = (const char*)(pkg_name->as_C_string());
ClassPathImageEntry* cpie = (ClassPathImageEntry*)e; ClassPathImageEntry* cpie = (ClassPathImageEntry*)e;
JImageFile* jimage = cpie->jimage(); JImageFile* jimage = cpie->jimage();

View File

@ -444,7 +444,9 @@ class ClassLoader: AllStatic {
static bool string_ends_with(const char* str, const char* str_to_find); static bool string_ends_with(const char* str, const char* str_to_find);
// obtain package name from a fully qualified class name // obtain package name from a fully qualified class name
static const char* package_from_name(const char* class_name); // *bad_class_name is set to true if there's a problem with parsing class_name, to
// distinguish from a class_name with no package name, as both cases have a NULL return value
static const char* package_from_name(const char* const class_name, bool* bad_class_name = NULL);
static bool is_jrt(const char* name) { return string_ends_with(name, MODULES_IMAGE_NAME); } static bool is_jrt(const char* name) { return string_ends_with(name, MODULES_IMAGE_NAME); }

View File

@ -54,12 +54,14 @@ public:
const s2 classpath_index, const s2 classpath_index,
instanceKlassHandle result, TRAPS) { instanceKlassHandle result, TRAPS) {
if (ClassLoader::add_package(_file_name, classpath_index, THREAD)) { if (ClassLoader::add_package(_file_name, classpath_index, THREAD)) {
#if INCLUDE_CDS
if (DumpSharedSpaces) { if (DumpSharedSpaces) {
s2 classloader_type = ClassLoader::classloader_type( s2 classloader_type = ClassLoader::classloader_type(
class_name, e, classpath_index, CHECK_(result)); class_name, e, classpath_index, CHECK_(result));
result->set_shared_classpath_index(classpath_index); result->set_shared_classpath_index(classpath_index);
result->set_class_loader_type(classloader_type); result->set_class_loader_type(classloader_type);
} }
#endif
return result; return result;
} else { } else {
return instanceKlassHandle(); // NULL return instanceKlassHandle(); // NULL

View File

@ -3245,6 +3245,15 @@ void java_lang_invoke_MemberName::set_vmindex(oop mname, intptr_t index) {
mname->address_field_put(_vmindex_offset, (address) index); mname->address_field_put(_vmindex_offset, (address) index);
} }
bool java_lang_invoke_MemberName::equals(oop mn1, oop mn2) {
if (mn1 == mn2) {
return true;
}
return (vmtarget(mn1) == vmtarget(mn2) && flags(mn1) == flags(mn2) &&
vmindex(mn1) == vmindex(mn2) &&
clazz(mn1) == clazz(mn2));
}
oop java_lang_invoke_LambdaForm::vmentry(oop lform) { oop java_lang_invoke_LambdaForm::vmentry(oop lform) {
assert(is_instance(lform), "wrong type"); assert(is_instance(lform), "wrong type");
return lform->obj_field(_vmentry_offset); return lform->obj_field(_vmentry_offset);

View File

@ -1114,6 +1114,8 @@ class java_lang_invoke_MemberName: AllStatic {
static int flags_offset_in_bytes() { return _flags_offset; } static int flags_offset_in_bytes() { return _flags_offset; }
static int vmtarget_offset_in_bytes() { return _vmtarget_offset; } static int vmtarget_offset_in_bytes() { return _vmtarget_offset; }
static int vmindex_offset_in_bytes() { return _vmindex_offset; } static int vmindex_offset_in_bytes() { return _vmindex_offset; }
static bool equals(oop mt1, oop mt2);
}; };

View File

@ -201,7 +201,7 @@ ModuleEntryTable::~ModuleEntryTable() {
} }
void ModuleEntryTable::create_unnamed_module(ClassLoaderData* loader_data) { void ModuleEntryTable::create_unnamed_module(ClassLoaderData* loader_data) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
// Each ModuleEntryTable has exactly one unnamed module // Each ModuleEntryTable has exactly one unnamed module
if (loader_data->is_the_null_class_loader_data()) { if (loader_data->is_the_null_class_loader_data()) {
@ -227,7 +227,7 @@ void ModuleEntryTable::create_unnamed_module(ClassLoaderData* loader_data) {
ModuleEntry* ModuleEntryTable::new_entry(unsigned int hash, Handle module_handle, Symbol* name, ModuleEntry* ModuleEntryTable::new_entry(unsigned int hash, Handle module_handle, Symbol* name,
Symbol* version, Symbol* location, Symbol* version, Symbol* location,
ClassLoaderData* loader_data) { ClassLoaderData* loader_data) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
ModuleEntry* entry = (ModuleEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtModule); ModuleEntry* entry = (ModuleEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtModule);
// Initialize everything BasicHashtable would // Initialize everything BasicHashtable would
@ -258,7 +258,7 @@ ModuleEntry* ModuleEntryTable::new_entry(unsigned int hash, Handle module_handle
} }
void ModuleEntryTable::add_entry(int index, ModuleEntry* new_entry) { void ModuleEntryTable::add_entry(int index, ModuleEntry* new_entry) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
Hashtable<Symbol*, mtModule>::add_entry(index, (HashtableEntry<Symbol*, mtModule>*)new_entry); Hashtable<Symbol*, mtModule>::add_entry(index, (HashtableEntry<Symbol*, mtModule>*)new_entry);
} }
@ -268,7 +268,7 @@ ModuleEntry* ModuleEntryTable::locked_create_entry_or_null(Handle module_handle,
Symbol* module_location, Symbol* module_location,
ClassLoaderData* loader_data) { ClassLoaderData* loader_data) {
assert(module_name != NULL, "ModuleEntryTable locked_create_entry_or_null should never be called for unnamed module."); assert(module_name != NULL, "ModuleEntryTable locked_create_entry_or_null should never be called for unnamed module.");
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
// Check if module already exists. // Check if module already exists.
if (lookup_only(module_name) != NULL) { if (lookup_only(module_name) != NULL) {
return NULL; return NULL;
@ -309,7 +309,7 @@ void ModuleEntryTable::purge_all_module_reads() {
} }
void ModuleEntryTable::finalize_javabase(Handle module_handle, Symbol* version, Symbol* location) { void ModuleEntryTable::finalize_javabase(Handle module_handle, Symbol* version, Symbol* location) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
ClassLoaderData* boot_loader_data = ClassLoaderData::the_null_class_loader_data(); ClassLoaderData* boot_loader_data = ClassLoaderData::the_null_class_loader_data();
ModuleEntryTable* module_table = boot_loader_data->modules(); ModuleEntryTable* module_table = boot_loader_data->modules();

View File

@ -568,8 +568,8 @@ void Modules::add_module_exports(jobject from_module, jstring package, jobject t
to_module_entry->is_named() ? to_module_entry->is_named() ?
to_module_entry->name()->as_C_string() : UNNAMED_MODULE); to_module_entry->name()->as_C_string() : UNNAMED_MODULE);
// Do nothing if modules are the same or if package is already exported unqualifiedly. // Do nothing if modules are the same.
if (from_module_entry != to_module_entry && !package_entry->is_unqual_exported()) { if (from_module_entry != to_module_entry) {
package_entry->set_exported(to_module_entry); package_entry->set_exported(to_module_entry);
} }
} }

View File

@ -49,7 +49,7 @@ bool PackageEntry::is_qexported_to(ModuleEntry* m) const {
// Add a module to the package's qualified export list. // Add a module to the package's qualified export list.
void PackageEntry::add_qexport(ModuleEntry* m) { void PackageEntry::add_qexport(ModuleEntry* m) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
if (!has_qual_exports_list()) { if (!has_qual_exports_list()) {
// Lazily create a package's qualified exports list. // Lazily create a package's qualified exports list.
// Initial size is small, do not anticipate export lists to be large. // Initial size is small, do not anticipate export lists to be large.
@ -157,7 +157,7 @@ PackageEntryTable::~PackageEntryTable() {
} }
PackageEntry* PackageEntryTable::new_entry(unsigned int hash, Symbol* name, ModuleEntry* module) { PackageEntry* PackageEntryTable::new_entry(unsigned int hash, Symbol* name, ModuleEntry* module) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
PackageEntry* entry = (PackageEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtModule); PackageEntry* entry = (PackageEntry*) NEW_C_HEAP_ARRAY(char, entry_size(), mtModule);
// Initialize everything BasicHashtable would // Initialize everything BasicHashtable would
@ -180,14 +180,14 @@ PackageEntry* PackageEntryTable::new_entry(unsigned int hash, Symbol* name, Modu
} }
void PackageEntryTable::add_entry(int index, PackageEntry* new_entry) { void PackageEntryTable::add_entry(int index, PackageEntry* new_entry) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
Hashtable<Symbol*, mtModule>::add_entry(index, (HashtableEntry<Symbol*, mtModule>*)new_entry); Hashtable<Symbol*, mtModule>::add_entry(index, (HashtableEntry<Symbol*, mtModule>*)new_entry);
} }
// Create package in loader's package entry table and return the entry. // Create package in loader's package entry table and return the entry.
// If entry already exists, return null. Assume Module lock was taken by caller. // If entry already exists, return null. Assume Module lock was taken by caller.
PackageEntry* PackageEntryTable::locked_create_entry_or_null(Symbol* name, ModuleEntry* module) { PackageEntry* PackageEntryTable::locked_create_entry_or_null(Symbol* name, ModuleEntry* module) {
assert_locked_or_safepoint(Module_lock); assert(Module_lock->owned_by_self(), "should have the Module_lock");
// Check if package already exists. Return NULL if it does. // Check if package already exists. Return NULL if it does.
if (lookup_only(name) != NULL) { if (lookup_only(name) != NULL) {
return NULL; return NULL;

View File

@ -40,11 +40,7 @@
// package is exported to. // package is exported to.
// //
// Packages can be exported in the following 3 ways: // Packages can be exported in the following 3 ways:
// - not exported: the package has not been explicitly qualified to a // - not exported: the package does not have qualified or unqualified exports.
// particular module nor has it been specified to be
// unqualifiedly exported to all modules. If all states
// of exportedness are false, the package is considered
// not exported.
// - qualified exports: the package has been explicitly qualified to at least // - qualified exports: the package has been explicitly qualified to at least
// one particular module or has been qualifiedly exported // one particular module or has been qualifiedly exported
// to all unnamed modules. // to all unnamed modules.
@ -125,6 +121,7 @@ public:
return _is_exported_unqualified; return _is_exported_unqualified;
} }
void set_unqual_exported() { void set_unqual_exported() {
assert(Module_lock->owned_by_self(), "should have the Module_lock");
_is_exported_unqualified = true; _is_exported_unqualified = true;
_is_exported_allUnnamed = false; _is_exported_allUnnamed = false;
_qualified_exports = NULL; _qualified_exports = NULL;

View File

@ -70,6 +70,7 @@
#include "services/threadService.hpp" #include "services/threadService.hpp"
#include "trace/traceMacros.hpp" #include "trace/traceMacros.hpp"
#include "utilities/macros.hpp" #include "utilities/macros.hpp"
#include "utilities/stringUtils.hpp"
#include "utilities/ticks.hpp" #include "utilities/ticks.hpp"
#if INCLUDE_CDS #if INCLUDE_CDS
#include "classfile/sharedClassUtil.hpp" #include "classfile/sharedClassUtil.hpp"
@ -1154,12 +1155,10 @@ Klass* SystemDictionary::resolve_from_stream(Symbol* class_name,
// It is illegal to define classes in the "java." package from // It is illegal to define classes in the "java." package from
// JVM_DefineClass or jni_DefineClass unless you're the bootclassloader // JVM_DefineClass or jni_DefineClass unless you're the bootclassloader
ResourceMark rm(THREAD); ResourceMark rm(THREAD);
char* name = parsed_name->as_C_string(); TempNewSymbol pkg_name = InstanceKlass::package_from_name(parsed_name, CHECK_NULL);
char* index = strrchr(name, '/'); assert(pkg_name != NULL, "Error in parsing package name starting with 'java/'");
*index = '\0'; // chop to just the package name char* name = pkg_name->as_C_string();
while ((index = strchr(name, '/')) != NULL) { StringUtils::replace_no_expand(name, "/", ".");
*index = '.'; // replace '/' with '.' in package name
}
const char* msg_text = "Prohibited package name: "; const char* msg_text = "Prohibited package name: ";
size_t len = strlen(msg_text) + strlen(name) + 1; size_t len = strlen(msg_text) + strlen(name) + 1;
char* message = NEW_RESOURCE_ARRAY(char, len); char* message = NEW_RESOURCE_ARRAY(char, len);
@ -1257,6 +1256,7 @@ instanceKlassHandle SystemDictionary::load_shared_class(
bool SystemDictionary::is_shared_class_visible(Symbol* class_name, bool SystemDictionary::is_shared_class_visible(Symbol* class_name,
instanceKlassHandle ik, instanceKlassHandle ik,
Handle class_loader, TRAPS) { Handle class_loader, TRAPS) {
ResourceMark rm;
int path_index = ik->shared_classpath_index(); int path_index = ik->shared_classpath_index();
SharedClassPathEntry* ent = SharedClassPathEntry* ent =
(SharedClassPathEntry*)FileMapInfo::shared_classpath(path_index); (SharedClassPathEntry*)FileMapInfo::shared_classpath(path_index);
@ -1270,12 +1270,11 @@ bool SystemDictionary::is_shared_class_visible(Symbol* class_name,
TempNewSymbol pkg_name = NULL; TempNewSymbol pkg_name = NULL;
PackageEntry* pkg_entry = NULL; PackageEntry* pkg_entry = NULL;
ModuleEntry* mod_entry = NULL; ModuleEntry* mod_entry = NULL;
int length = 0; const char* pkg_string = NULL;
ClassLoaderData* loader_data = class_loader_data(class_loader); ClassLoaderData* loader_data = class_loader_data(class_loader);
const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length); pkg_name = InstanceKlass::package_from_name(class_name, CHECK_false);
if (pkg_string != NULL) { if (pkg_name != NULL) {
pkg_name = SymbolTable::new_symbol((const char*)pkg_string, pkg_string = pkg_name->as_C_string();
length, CHECK_(false));
if (loader_data != NULL) { if (loader_data != NULL) {
pkg_entry = loader_data->packages()->lookup_only(pkg_name); pkg_entry = loader_data->packages()->lookup_only(pkg_name);
} }
@ -1432,15 +1431,14 @@ instanceKlassHandle SystemDictionary::load_instance_class(Symbol* class_name, Ha
instanceKlassHandle nh = instanceKlassHandle(); // null Handle instanceKlassHandle nh = instanceKlassHandle(); // null Handle
if (class_loader.is_null()) { if (class_loader.is_null()) {
int length = 0; ResourceMark rm;
PackageEntry* pkg_entry = NULL; PackageEntry* pkg_entry = NULL;
bool search_only_bootloader_append = false; bool search_only_bootloader_append = false;
ClassLoaderData *loader_data = class_loader_data(class_loader); ClassLoaderData *loader_data = class_loader_data(class_loader);
// Find the package in the boot loader's package entry table. // Find the package in the boot loader's package entry table.
const jbyte* pkg_string = InstanceKlass::package_from_name(class_name, length); TempNewSymbol pkg_name = InstanceKlass::package_from_name(class_name, CHECK_NULL);
if (pkg_string != NULL) { if (pkg_name != NULL) {
TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)pkg_string, length, CHECK_(nh));
pkg_entry = loader_data->packages()->lookup_only(pkg_name); pkg_entry = loader_data->packages()->lookup_only(pkg_name);
} }
@ -1477,7 +1475,7 @@ instanceKlassHandle SystemDictionary::load_instance_class(Symbol* class_name, Ha
assert(!DumpSharedSpaces, "Archive dumped after module system initialization"); assert(!DumpSharedSpaces, "Archive dumped after module system initialization");
// After the module system has been initialized, check if the class' // After the module system has been initialized, check if the class'
// package is in a module defined to the boot loader. // package is in a module defined to the boot loader.
if (pkg_string == NULL || pkg_entry == NULL || pkg_entry->in_unnamed_module()) { if (pkg_name == NULL || pkg_entry == NULL || pkg_entry->in_unnamed_module()) {
// Class is either in the unnamed package, in a named package // Class is either in the unnamed package, in a named package
// within a module not defined to the boot loader or in a // within a module not defined to the boot loader or in a
// a named package within the unnamed module. In all cases, // a named package within the unnamed module. In all cases,

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -48,7 +48,7 @@ public:
static bool is_shared_class_visible_for_classloader( static bool is_shared_class_visible_for_classloader(
instanceKlassHandle ik, instanceKlassHandle ik,
Handle class_loader, Handle class_loader,
const jbyte* pkg_string, const char* pkg_string,
Symbol* pkg_name, Symbol* pkg_name,
PackageEntry* pkg_entry, PackageEntry* pkg_entry,
ModuleEntry* mod_entry, ModuleEntry* mod_entry,

View File

@ -137,11 +137,6 @@ MethodLiveness::MethodLiveness(Arena* arena, ciMethod* method)
_arena = arena; _arena = arena;
_method = method; _method = method;
_bit_map_size_bits = method->max_locals(); _bit_map_size_bits = method->max_locals();
#ifdef COMPILER1
_bci_block_start.clear();
#endif
} }
void MethodLiveness::compute_liveness() { void MethodLiveness::compute_liveness() {
@ -587,14 +582,6 @@ MethodLiveness::BasicBlock::BasicBlock(MethodLiveness *analyzer, int start, int
new (analyzer->arena()) GrowableArray<MethodLiveness::BasicBlock*>(analyzer->arena(), 5, 0, NULL); new (analyzer->arena()) GrowableArray<MethodLiveness::BasicBlock*>(analyzer->arena(), 5, 0, NULL);
_exception_predecessors = _exception_predecessors =
new (analyzer->arena()) GrowableArray<MethodLiveness::BasicBlock*>(analyzer->arena(), 5, 0, NULL); new (analyzer->arena()) GrowableArray<MethodLiveness::BasicBlock*>(analyzer->arena(), 5, 0, NULL);
_normal_exit.clear();
_exception_exit.clear();
_entry.clear();
// this initialization is not strictly necessary.
// _gen and _kill are cleared at the beginning of compute_gen_kill_range()
_gen.clear();
_kill.clear();
} }
@ -1020,7 +1007,6 @@ MethodLivenessResult MethodLiveness::BasicBlock::get_liveness_at(ciMethod* metho
_last_bci = bci; _last_bci = bci;
} }
answer.clear();
answer.set_union(_normal_exit); answer.set_union(_normal_exit);
answer.set_difference(_kill); answer.set_difference(_kill);
answer.set_union(_gen); answer.set_union(_gen);

View File

@ -1605,6 +1605,9 @@ void CMSCollector::do_compaction_work(bool clear_all_soft_refs) {
_inter_sweep_timer.reset(); _inter_sweep_timer.reset();
_inter_sweep_timer.start(); _inter_sweep_timer.start();
// No longer a need to do a concurrent collection for Metaspace.
MetaspaceGC::set_should_concurrent_collect(false);
gch->post_full_gc_dump(gc_timer); gch->post_full_gc_dump(gc_timer);
gc_timer->register_gc_end(); gc_timer->register_gc_end();

View File

@ -3474,7 +3474,8 @@ void G1CollectedHeap::restore_after_evac_failure() {
double remove_self_forwards_start = os::elapsedTime(); double remove_self_forwards_start = os::elapsedTime();
remove_self_forwarding_pointers(); remove_self_forwarding_pointers();
_preserved_marks_set.restore(workers()); SharedRestorePreservedMarksTaskExecutor task_executor(workers());
_preserved_marks_set.restore(&task_executor);
g1_policy()->phase_times()->record_evac_fail_remove_self_forwards((os::elapsedTime() - remove_self_forwards_start) * 1000.0); g1_policy()->phase_times()->record_evac_fail_remove_self_forwards((os::elapsedTime() - remove_self_forwards_start) * 1000.0);
} }

View File

@ -109,7 +109,7 @@ inline void G1CollectedHeap::old_set_remove(HeapRegion* hr) {
_old_set.remove(hr); _old_set.remove(hr);
} }
// It dirties the cards that cover the block so that so that the post // It dirties the cards that cover the block so that the post
// write barrier never queues anything when updating objects on this // write barrier never queues anything when updating objects on this
// block. It is assumed (and in fact we assert) that the block // block. It is assumed (and in fact we assert) that the block
// belongs to a young region. // belongs to a young region.

View File

@ -80,7 +80,7 @@ void G1StringDedupStat::print_summary(const G1StringDedupStat& last_stat, const
log_info(gc, stringdedup)( log_info(gc, stringdedup)(
"Concurrent String Deduplication " "Concurrent String Deduplication "
G1_STRDEDUP_BYTES_FORMAT_NS "->" G1_STRDEDUP_BYTES_FORMAT_NS "(" G1_STRDEDUP_BYTES_FORMAT_NS "), avg " G1_STRDEDUP_BYTES_FORMAT_NS "->" G1_STRDEDUP_BYTES_FORMAT_NS "(" G1_STRDEDUP_BYTES_FORMAT_NS "), avg "
G1_STRDEDUP_PERCENT_FORMAT_NS ", " G1_STRDEDUP_TIME_FORMAT "]", G1_STRDEDUP_PERCENT_FORMAT_NS ", " G1_STRDEDUP_TIME_FORMAT,
G1_STRDEDUP_BYTES_PARAM(last_stat._new_bytes), G1_STRDEDUP_BYTES_PARAM(last_stat._new_bytes),
G1_STRDEDUP_BYTES_PARAM(last_stat._new_bytes - last_stat._deduped_bytes), G1_STRDEDUP_BYTES_PARAM(last_stat._new_bytes - last_stat._deduped_bytes),
G1_STRDEDUP_BYTES_PARAM(last_stat._deduped_bytes), G1_STRDEDUP_BYTES_PARAM(last_stat._deduped_bytes),

View File

@ -23,6 +23,7 @@
*/ */
#include "precompiled.hpp" #include "precompiled.hpp"
#include "gc/parallel/gcTaskManager.hpp"
#include "gc/parallel/mutableSpace.hpp" #include "gc/parallel/mutableSpace.hpp"
#include "gc/parallel/parallelScavengeHeap.hpp" #include "gc/parallel/parallelScavengeHeap.hpp"
#include "gc/parallel/psOldGen.hpp" #include "gc/parallel/psOldGen.hpp"
@ -237,8 +238,53 @@ void PSPromotionManager::register_preserved_marks(PreservedMarks* preserved_mark
_preserved_marks = preserved_marks; _preserved_marks = preserved_marks;
} }
class ParRestoreGCTask : public GCTask {
private:
const uint _id;
PreservedMarksSet* const _preserved_marks_set;
volatile size_t* const _total_size_addr;
public:
virtual char* name() {
return (char*) "preserved mark restoration task";
}
virtual void do_it(GCTaskManager* manager, uint which){
_preserved_marks_set->get(_id)->restore_and_increment(_total_size_addr);
}
ParRestoreGCTask(uint id,
PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr)
: _id(id),
_preserved_marks_set(preserved_marks_set),
_total_size_addr(total_size_addr) { }
};
class PSRestorePreservedMarksTaskExecutor : public RestorePreservedMarksTaskExecutor {
private:
GCTaskManager* _gc_task_manager;
public:
PSRestorePreservedMarksTaskExecutor(GCTaskManager* gc_task_manager)
: _gc_task_manager(gc_task_manager) { }
void restore(PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr) {
// GCTask / GCTaskQueue are ResourceObjs
ResourceMark rm;
GCTaskQueue* q = GCTaskQueue::create();
for (uint i = 0; i < preserved_marks_set->num(); i += 1) {
q->enqueue(new ParRestoreGCTask(i, preserved_marks_set, total_size_addr));
}
_gc_task_manager->execute_and_wait(q);
}
};
void PSPromotionManager::restore_preserved_marks() { void PSPromotionManager::restore_preserved_marks() {
_preserved_marks_set->restore(PSScavenge::gc_task_manager()); PSRestorePreservedMarksTaskExecutor task_executor(PSScavenge::gc_task_manager());
_preserved_marks_set->restore(&task_executor);
} }
void PSPromotionManager::drain_stacks_depth(bool totally_drain) { void PSPromotionManager::drain_stacks_depth(bool totally_drain) {

View File

@ -739,7 +739,8 @@ void DefNewGeneration::remove_forwarding_pointers() {
eden()->object_iterate(&rspc); eden()->object_iterate(&rspc);
from()->object_iterate(&rspc); from()->object_iterate(&rspc);
_preserved_marks_set.restore(GenCollectedHeap::heap()->workers()); SharedRestorePreservedMarksTaskExecutor task_executor(GenCollectedHeap::heap()->workers());
_preserved_marks_set.restore(&task_executor);
} }
void DefNewGeneration::handle_promotion_failure(oop old) { void DefNewGeneration::handle_promotion_failure(oop old) {

View File

@ -386,7 +386,7 @@ size_t CollectedHeap::max_tlab_size() const {
// initialized by this point, a fact that we assert when doing the // initialized by this point, a fact that we assert when doing the
// card-mark.) // card-mark.)
// (c) G1CollectedHeap(G1) uses two kinds of write barriers. When a // (c) G1CollectedHeap(G1) uses two kinds of write barriers. When a
// G1 concurrent marking is in progress an SATB (pre-write-)barrier is // G1 concurrent marking is in progress an SATB (pre-write-)barrier
// is used to remember the pre-value of any store. Initializing // is used to remember the pre-value of any store. Initializing
// stores will not need this barrier, so we need not worry about // stores will not need this barrier, so we need not worry about
// compensating for the missing pre-barrier here. Turning now // compensating for the missing pre-barrier here. Turning now

View File

@ -28,9 +28,6 @@
#include "memory/allocation.inline.hpp" #include "memory/allocation.inline.hpp"
#include "memory/resourceArea.hpp" #include "memory/resourceArea.hpp"
#include "utilities/macros.hpp" #include "utilities/macros.hpp"
#if INCLUDE_ALL_GCS
#include "gc/parallel/gcTaskManager.hpp"
#endif
void PreservedMarks::restore() { void PreservedMarks::restore() {
while (!_stack.is_empty()) { while (!_stack.is_empty()) {
@ -40,6 +37,15 @@ void PreservedMarks::restore() {
assert_empty(); assert_empty();
} }
void PreservedMarks::restore_and_increment(volatile size_t* const total_size_addr) {
const size_t stack_size = size();
restore();
// Only do the atomic add if the size is > 0.
if (stack_size > 0) {
Atomic::add(stack_size, total_size_addr);
}
}
#ifndef PRODUCT #ifndef PRODUCT
void PreservedMarks::assert_empty() { void PreservedMarks::assert_empty() {
assert(_stack.is_empty(), "stack expected to be empty, size = "SIZE_FORMAT, assert(_stack.is_empty(), "stack expected to be empty, size = "SIZE_FORMAT,
@ -82,13 +88,7 @@ public:
virtual void work(uint worker_id) { virtual void work(uint worker_id) {
uint task_id = 0; uint task_id = 0;
while (!_sub_tasks.is_task_claimed(/* reference */ task_id)) { while (!_sub_tasks.is_task_claimed(/* reference */ task_id)) {
PreservedMarks* const preserved_marks = _preserved_marks_set->get(task_id); _preserved_marks_set->get(task_id)->restore_and_increment(_total_size_addr);
const size_t size = preserved_marks->size();
preserved_marks->restore();
// Only do the atomic add if the size is > 0.
if (size > 0) {
Atomic::add(size, _total_size_addr);
}
} }
_sub_tasks.all_tasks_completed(); _sub_tasks.all_tasks_completed();
} }
@ -104,53 +104,6 @@ public:
} }
}; };
void PreservedMarksSet::restore_internal(WorkGang* workers,
volatile size_t* total_size_addr) {
assert(workers != NULL, "pre-condition");
ParRestoreTask task(workers->active_workers(), this, total_size_addr);
workers->run_task(&task);
}
#if INCLUDE_ALL_GCS
class ParRestoreGCTask : public GCTask {
private:
const uint _id;
PreservedMarksSet* const _preserved_marks_set;
volatile size_t* const _total_size_addr;
public:
virtual char* name() { return (char*) "preserved mark restoration task"; }
virtual void do_it(GCTaskManager* manager, uint which) {
PreservedMarks* const preserved_marks = _preserved_marks_set->get(_id);
const size_t size = preserved_marks->size();
preserved_marks->restore();
// Only do the atomic add if the size is > 0.
if (size > 0) {
Atomic::add(size, _total_size_addr);
}
}
ParRestoreGCTask(uint id,
PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr)
: _id(id),
_preserved_marks_set(preserved_marks_set),
_total_size_addr(total_size_addr) { }
};
void PreservedMarksSet::restore_internal(GCTaskManager* gc_task_manager,
volatile size_t* total_size_addr) {
// GCTask / GCTaskQueue are ResourceObjs
ResourceMark rm;
GCTaskQueue* q = GCTaskQueue::create();
for (uint i = 0; i < num(); i += 1) {
q->enqueue(new ParRestoreGCTask(i, this, total_size_addr));
}
gc_task_manager->execute_and_wait(q);
}
#endif
void PreservedMarksSet::reclaim() { void PreservedMarksSet::reclaim() {
assert_empty(); assert_empty();
@ -176,3 +129,16 @@ void PreservedMarksSet::assert_empty() {
} }
} }
#endif // ndef PRODUCT #endif // ndef PRODUCT
void SharedRestorePreservedMarksTaskExecutor::restore(PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr) {
if (_workers == NULL) {
for (uint i = 0; i < preserved_marks_set->num(); i += 1) {
*total_size_addr += preserved_marks_set->get(i)->size();
preserved_marks_set->get(i)->restore();
}
} else {
ParRestoreTask task(_workers->active_workers(), preserved_marks_set, total_size_addr);
_workers->run_task(&task);
}
}

View File

@ -30,7 +30,7 @@
#include "oops/oop.hpp" #include "oops/oop.hpp"
#include "utilities/stack.hpp" #include "utilities/stack.hpp"
class GCTaskManager; class PreservedMarksSet;
class WorkGang; class WorkGang;
class PreservedMarks VALUE_OBJ_CLASS_SPEC { class PreservedMarks VALUE_OBJ_CLASS_SPEC {
@ -61,6 +61,7 @@ public:
// reclaim the memory taken up by the stack segments. // reclaim the memory taken up by the stack segments.
void restore(); void restore();
void restore_and_increment(volatile size_t* const _total_size_addr);
inline static void init_forwarded_mark(oop obj); inline static void init_forwarded_mark(oop obj);
// Assert the stack is empty and has no cached segments. // Assert the stack is empty and has no cached segments.
@ -75,6 +76,24 @@ public:
virtual void do_object(oop obj); virtual void do_object(oop obj);
}; };
class RestorePreservedMarksTaskExecutor {
public:
void virtual restore(PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr) = 0;
};
class SharedRestorePreservedMarksTaskExecutor : public RestorePreservedMarksTaskExecutor {
private:
WorkGang* _workers;
public:
SharedRestorePreservedMarksTaskExecutor(WorkGang* workers) : _workers(workers) { }
void restore(PreservedMarksSet* preserved_marks_set,
volatile size_t* total_size_addr);
};
class PreservedMarksSet : public CHeapObj<mtGC> { class PreservedMarksSet : public CHeapObj<mtGC> {
private: private:
// true -> _stacks will be allocated in the C heap // true -> _stacks will be allocated in the C heap
@ -91,13 +110,6 @@ private:
// or == NULL if they have not. // or == NULL if they have not.
Padded<PreservedMarks>* _stacks; Padded<PreservedMarks>* _stacks;
// Internal version of restore() that uses a WorkGang for parallelism.
void restore_internal(WorkGang* workers, volatile size_t* total_size_addr);
// Internal version of restore() that uses a GCTaskManager for parallelism.
void restore_internal(GCTaskManager* gc_task_manager,
volatile size_t* total_size_addr);
public: public:
uint num() const { return _num; } uint num() const { return _num; }
@ -111,14 +123,11 @@ public:
// Allocate stack array. // Allocate stack array.
void init(uint num); void init(uint num);
// Itrerate over all stacks, restore all presered marks, and reclaim // Iterate over all stacks, restore all preserved marks, and reclaim
// the memory taken up by the stack segments. If the executor is // the memory taken up by the stack segments.
// NULL, restoration will be done serially. If the executor is not // Supported executors: SharedRestorePreservedMarksTaskExecutor (Serial, CMS, G1),
// NULL, restoration could be done in parallel (when it makes // PSRestorePreservedMarksTaskExecutor (PS).
// sense). Supported executors: WorkGang (Serial, CMS, G1), inline void restore(RestorePreservedMarksTaskExecutor* executor);
// GCTaskManager (PS).
template <class E>
inline void restore(E* executor);
// Reclaim stack array. // Reclaim stack array.
void reclaim(); void reclaim();

View File

@ -49,8 +49,7 @@ inline void PreservedMarks::init_forwarded_mark(oop obj) {
obj->init_mark(); obj->init_mark();
} }
template <class E> inline void PreservedMarksSet::restore(RestorePreservedMarksTaskExecutor* executor) {
inline void PreservedMarksSet::restore(E* executor) {
volatile size_t total_size = 0; volatile size_t total_size = 0;
#ifdef ASSERT #ifdef ASSERT
@ -61,17 +60,7 @@ inline void PreservedMarksSet::restore(E* executor) {
} }
#endif // def ASSERT #endif // def ASSERT
if (executor == NULL) { executor->restore(this, &total_size);
for (uint i = 0; i < _num; i += 1) {
total_size += get(i)->size();
get(i)->restore();
}
} else {
// Right now, if the executor is not NULL we do the work in
// parallel. In the future we might want to do the restoration
// serially, if there's only a small number of marks per stack.
restore_internal(executor, &total_size);
}
assert_empty(); assert_empty();
assert(total_size == total_size_before, assert(total_size == total_size_before,

View File

@ -108,8 +108,9 @@ public:
// do nothing. tlabs must be inited by initialize() calls // do nothing. tlabs must be inited by initialize() calls
} }
static const size_t min_size() { return align_object_size(MinTLABSize / HeapWordSize) + alignment_reserve(); } static size_t min_size() { return align_object_size(MinTLABSize / HeapWordSize) + alignment_reserve(); }
static const size_t max_size() { assert(_max_size != 0, "max_size not set up"); return _max_size; } static size_t max_size() { assert(_max_size != 0, "max_size not set up"); return _max_size; }
static size_t max_size_in_bytes() { return max_size() * BytesPerWord; }
static void set_max_size(size_t max_size) { _max_size = max_size; } static void set_max_size(size_t max_size) { _max_size = max_size; }
HeapWord* start() const { return _start; } HeapWord* start() const { return _start; }

View File

@ -44,11 +44,6 @@ void AbstractWorkGang::initialize_workers() {
vm_exit_out_of_memory(0, OOM_MALLOC_ERROR, "Cannot create GangWorker array."); vm_exit_out_of_memory(0, OOM_MALLOC_ERROR, "Cannot create GangWorker array.");
} }
_active_workers = ParallelGCThreads;
if (UseDynamicNumberOfGCThreads && !FLAG_IS_CMDLINE(ParallelGCThreads)) {
_active_workers = 1U;
}
add_workers(true); add_workers(true);
} }
@ -60,6 +55,10 @@ AbstractGangWorker* AbstractWorkGang::install_worker(uint worker_id) {
} }
void AbstractWorkGang::add_workers(bool initializing) { void AbstractWorkGang::add_workers(bool initializing) {
add_workers(_active_workers, initializing);
}
void AbstractWorkGang::add_workers(uint active_workers, bool initializing) {
os::ThreadType worker_type; os::ThreadType worker_type;
if (are_ConcurrentGC_threads()) { if (are_ConcurrentGC_threads()) {
@ -69,7 +68,7 @@ void AbstractWorkGang::add_workers(bool initializing) {
} }
_created_workers = WorkerManager::add_workers(this, _created_workers = WorkerManager::add_workers(this,
_active_workers, active_workers,
_total_workers, _total_workers,
_created_workers, _created_workers,
worker_type, worker_type,
@ -268,10 +267,11 @@ void WorkGang::run_task(AbstractGangTask* task) {
} }
void WorkGang::run_task(AbstractGangTask* task, uint num_workers) { void WorkGang::run_task(AbstractGangTask* task, uint num_workers) {
guarantee(num_workers <= active_workers(), guarantee(num_workers <= total_workers(),
"Trying to execute task %s with %u workers which is more than the amount of active workers %u.", "Trying to execute task %s with %u workers which is more than the amount of total workers %u.",
task->name(), num_workers, active_workers()); task->name(), num_workers, total_workers());
guarantee(num_workers > 0, "Trying to execute task %s with zero workers", task->name()); guarantee(num_workers > 0, "Trying to execute task %s with zero workers", task->name());
add_workers(num_workers, false);
_dispatcher->coordinator_execute_on_workers(task, num_workers); _dispatcher->coordinator_execute_on_workers(task, num_workers);
} }

View File

@ -170,6 +170,9 @@ class AbstractWorkGang : public CHeapObj<mtInternal> {
// Add GC workers as needed. // Add GC workers as needed.
void add_workers(bool initializing); void add_workers(bool initializing);
// Add GC workers as needed to reach the specified number of workers.
void add_workers(uint active_workers, bool initializing);
// Return the Ith worker. // Return the Ith worker.
AbstractGangWorker* worker(uint i) const; AbstractGangWorker* worker(uint i) const;
@ -214,7 +217,8 @@ public:
virtual void run_task(AbstractGangTask* task); virtual void run_task(AbstractGangTask* task);
// Run a task with the given number of workers, returns // Run a task with the given number of workers, returns
// when the task is done. The number of workers must be at most the number of // when the task is done. The number of workers must be at most the number of
// active workers. // active workers. Additional workers may be created if an insufficient
// number currently exists.
void run_task(AbstractGangTask* task, uint num_workers); void run_task(AbstractGangTask* task, uint num_workers);
protected: protected:

View File

@ -40,31 +40,6 @@
#define JVMCI_ERROR_OK(...) JVMCI_ERROR_(JVMCIEnv::ok, __VA_ARGS__) #define JVMCI_ERROR_OK(...) JVMCI_ERROR_(JVMCIEnv::ok, __VA_ARGS__)
#define CHECK_OK CHECK_(JVMCIEnv::ok) #define CHECK_OK CHECK_(JVMCIEnv::ok)
class ParseClosure : public StackObj {
int _lineNo;
char* _filename;
bool _abort;
protected:
void abort() { _abort = true; }
void warn_and_abort(const char* message) {
warn(message);
abort();
}
void warn(const char* message) {
warning("Error at line %d while parsing %s: %s", _lineNo, _filename == NULL ? "?" : _filename, message);
}
public:
ParseClosure() : _lineNo(0), _filename(NULL), _abort(false) {}
void parse_line(char* line) {
_lineNo++;
do_line(line);
}
virtual void do_line(char* line) = 0;
int lineNo() { return _lineNo; }
bool is_aborted() { return _abort; }
void set_filename(char* path) {_filename = path; _lineNo = 0;}
};
class JVMCIRuntime: public AllStatic { class JVMCIRuntime: public AllStatic {
public: public:
// Constants describing whether JVMCI wants to be able to adjust the compilation // Constants describing whether JVMCI wants to be able to adjust the compilation

View File

@ -484,13 +484,13 @@ void LogConfiguration::print_command_line_help(FILE* out) {
" -Xlog:gc::uptime,tid\n" " -Xlog:gc::uptime,tid\n"
"\t Log messages tagged with 'gc' tag using 'info' level to output 'stdout', using 'uptime' and 'tid' decorations.\n\n" "\t Log messages tagged with 'gc' tag using 'info' level to output 'stdout', using 'uptime' and 'tid' decorations.\n\n"
" -Xlog:gc*=info,rt*=off\n" " -Xlog:gc*=info,safepoint*=off\n"
"\t Log messages tagged with at least 'gc' using 'info' level, but turn off logging of messages tagged with 'rt'.\n" "\t Log messages tagged with at least 'gc' using 'info' level, but turn off logging of messages tagged with 'safepoint'.\n"
"\t (Messages tagged with both 'gc' and 'rt' will not be logged.)\n\n" "\t (Messages tagged with both 'gc' and 'safepoint' will not be logged.)\n\n"
" -Xlog:disable -Xlog:rt=trace:rttrace.txt\n" " -Xlog:disable -Xlog:safepoint=trace:safepointtrace.txt\n"
"\t Turn off all logging, including warnings and errors,\n" "\t Turn off all logging, including warnings and errors,\n"
"\t and then enable messages tagged with 'rt' using 'trace' level to file 'rttrace.txt'.\n"); "\t and then enable messages tagged with 'safepoint' using 'trace' level to file 'safepointtrace.txt'.\n");
} }
void LogConfiguration::rotate_all_outputs() { void LogConfiguration::rotate_all_outputs() {

View File

@ -2933,7 +2933,7 @@ void Metaspace::allocate_metaspace_compressed_klass_ptrs(char* requested_addr, a
// Don't use large pages for the class space. // Don't use large pages for the class space.
bool large_pages = false; bool large_pages = false;
#ifndef AARCH64 #if !(defined(AARCH64) || defined(AIX))
ReservedSpace metaspace_rs = ReservedSpace(compressed_class_space_size(), ReservedSpace metaspace_rs = ReservedSpace(compressed_class_space_size(),
_reserve_alignment, _reserve_alignment,
large_pages, large_pages,
@ -2951,12 +2951,19 @@ void Metaspace::allocate_metaspace_compressed_klass_ptrs(char* requested_addr, a
} }
if (! metaspace_rs.is_reserved()) { if (! metaspace_rs.is_reserved()) {
// Try to align metaspace so that we can decode a compressed klass // Aarch64: Try to align metaspace so that we can decode a compressed
// with a single MOVK instruction. We can do this iff the // klass with a single MOVK instruction. We can do this iff the
// compressed class base is a multiple of 4G. // compressed class base is a multiple of 4G.
for (char *a = (char*)align_ptr_up(requested_addr, 4*G); // Aix: Search for a place where we can find memory. If we need to load
// the base, 4G alignment is helpful, too.
size_t increment = AARCH64_ONLY(4*)G;
for (char *a = (char*)align_ptr_up(requested_addr, increment);
a < (char*)(1024*G); a < (char*)(1024*G);
a += 4*G) { a += increment) {
if (a == (char *)(32*G)) {
// Go faster from here on. Zero-based is no longer possible.
increment = 4*G;
}
#if INCLUDE_CDS #if INCLUDE_CDS
if (UseSharedSpaces if (UseSharedSpaces

View File

@ -2182,39 +2182,21 @@ const char* InstanceKlass::signature_name() const {
return dest; return dest;
} }
const jbyte* InstanceKlass::package_from_name(const Symbol* name, int& length) { // Used to obtain the package name from a fully qualified class name.
ResourceMark rm; Symbol* InstanceKlass::package_from_name(const Symbol* name, TRAPS) {
length = 0;
if (name == NULL) { if (name == NULL) {
return NULL; return NULL;
} else { } else {
const jbyte* base_name = name->base(); if (name->utf8_length() <= 0) {
const jbyte* last_slash = UTF8::strrchr(base_name, name->utf8_length(), '/');
if (last_slash == NULL) {
// No package name
return NULL;
} else {
// Skip over '['s
if (*base_name == '[') {
do {
base_name++;
} while (*base_name == '[');
if (*base_name != 'L') {
// Fully qualified class names should not contain a 'L'.
// Set length to -1 to indicate that the package name
// could not be obtained due to an error condition.
// In this situtation, is_same_class_package returns false.
length = -1;
return NULL; return NULL;
} }
ResourceMark rm;
const char* package_name = ClassLoader::package_from_name((const char*) name->as_C_string());
if (package_name == NULL) {
return NULL;
} }
Symbol* pkg_name = SymbolTable::new_symbol(package_name, THREAD);
// Found the package name, look it up in the symbol table. return pkg_name;
length = last_slash - base_name;
assert(length > 0, "Bad length for package name");
return base_name;
}
} }
} }
@ -2230,12 +2212,9 @@ ModuleEntry* InstanceKlass::module() const {
} }
void InstanceKlass::set_package(ClassLoaderData* loader_data, TRAPS) { void InstanceKlass::set_package(ClassLoaderData* loader_data, TRAPS) {
int length = 0; TempNewSymbol pkg_name = package_from_name(name(), CHECK);
const jbyte* base_name = package_from_name(name(), length);
if (base_name != NULL && loader_data != NULL) {
TempNewSymbol pkg_name = SymbolTable::new_symbol((const char*)base_name, length, CHECK);
if (pkg_name != NULL && loader_data != NULL) {
// Find in class loader's package entry table. // Find in class loader's package entry table.
_package_entry = loader_data->packages()->lookup_only(pkg_name); _package_entry = loader_data->packages()->lookup_only(pkg_name);
@ -2331,20 +2310,18 @@ bool InstanceKlass::is_same_class_package(oop class_loader1, const Symbol* class
if (class_loader1 != class_loader2) { if (class_loader1 != class_loader2) {
return false; return false;
} else if (class_name1 == class_name2) { } else if (class_name1 == class_name2) {
return true; // skip painful bytewise comparison return true;
} else { } else {
ResourceMark rm; ResourceMark rm;
// The Symbol*'s are in UTF8 encoding. Since we only need to check explicitly bool bad_class_name = false;
// for ASCII characters ('/', 'L', '['), we can keep them in UTF8 encoding. const char* name1 = ClassLoader::package_from_name((const char*) class_name1->as_C_string(), &bad_class_name);
// Otherwise, we just compare jbyte values between the strings. if (bad_class_name) {
int length1 = 0; return false;
int length2 = 0; }
const jbyte *name1 = package_from_name(class_name1, length1);
const jbyte *name2 = package_from_name(class_name2, length2);
if ((length1 < 0) || (length2 < 0)) { const char* name2 = ClassLoader::package_from_name((const char*) class_name2->as_C_string(), &bad_class_name);
// error occurred parsing package name. if (bad_class_name) {
return false; return false;
} }
@ -2354,13 +2331,13 @@ bool InstanceKlass::is_same_class_package(oop class_loader1, const Symbol* class
return name1 == name2; return name1 == name2;
} }
// Check that package part is identical // Check that package is identical
return UTF8::equal(name1, length1, name2, length2); return (strcmp(name1, name2) == 0);
} }
} }
// Returns true iff super_method can be overridden by a method in targetclassname // Returns true iff super_method can be overridden by a method in targetclassname
// See JSL 3rd edition 8.4.6.1 // See JLS 3rd edition 8.4.6.1
// Assumes name-signature match // Assumes name-signature match
// "this" is InstanceKlass of super_method which must exist // "this" is InstanceKlass of super_method which must exist
// note that the InstanceKlass of the method in the targetclassname has not always been created yet // note that the InstanceKlass of the method in the targetclassname has not always been created yet
@ -2716,7 +2693,7 @@ nmethod* InstanceKlass::lookup_osr_nmethod(const Method* m, int bci, int comp_le
return NULL; return NULL;
} }
bool InstanceKlass::add_member_name(Handle mem_name) { oop InstanceKlass::add_member_name(Handle mem_name, bool intern) {
jweak mem_name_wref = JNIHandles::make_weak_global(mem_name); jweak mem_name_wref = JNIHandles::make_weak_global(mem_name);
MutexLocker ml(MemberNameTable_lock); MutexLocker ml(MemberNameTable_lock);
DEBUG_ONLY(NoSafepointVerifier nsv); DEBUG_ONLY(NoSafepointVerifier nsv);
@ -2726,7 +2703,7 @@ bool InstanceKlass::add_member_name(Handle mem_name) {
// is called! // is called!
Method* method = (Method*)java_lang_invoke_MemberName::vmtarget(mem_name()); Method* method = (Method*)java_lang_invoke_MemberName::vmtarget(mem_name());
if (method->is_obsolete()) { if (method->is_obsolete()) {
return false; return NULL;
} else if (method->is_old()) { } else if (method->is_old()) {
// Replace method with redefined version // Replace method with redefined version
java_lang_invoke_MemberName::set_vmtarget(mem_name(), method_with_idnum(method->method_idnum())); java_lang_invoke_MemberName::set_vmtarget(mem_name(), method_with_idnum(method->method_idnum()));
@ -2735,8 +2712,11 @@ bool InstanceKlass::add_member_name(Handle mem_name) {
if (_member_names == NULL) { if (_member_names == NULL) {
_member_names = new (ResourceObj::C_HEAP, mtClass) MemberNameTable(idnum_allocated_count()); _member_names = new (ResourceObj::C_HEAP, mtClass) MemberNameTable(idnum_allocated_count());
} }
_member_names->add_member_name(mem_name_wref); if (intern) {
return true; return _member_names->find_or_add_member_name(mem_name_wref);
} else {
return _member_names->add_member_name(mem_name_wref);
}
} }
// ----------------------------------------------------------------------------------------------------- // -----------------------------------------------------------------------------------------------------
@ -3027,10 +3007,14 @@ void InstanceKlass::print_loading_log(LogLevel::type type,
if (cfs != NULL) { if (cfs != NULL) {
if (cfs->source() != NULL) { if (cfs->source() != NULL) {
if (module_name != NULL) { if (module_name != NULL) {
if (ClassLoader::is_jrt(cfs->source())) {
log->print(" source: jrt:/%s", module_name); log->print(" source: jrt:/%s", module_name);
} else { } else {
log->print(" source: %s", cfs->source()); log->print(" source: %s", cfs->source());
} }
} else {
log->print(" source: %s", cfs->source());
}
} else if (loader_data == ClassLoaderData::the_null_class_loader_data()) { } else if (loader_data == ClassLoaderData::the_null_class_loader_data()) {
Thread* THREAD = Thread::current(); Thread* THREAD = Thread::current();
Klass* caller = Klass* caller =

View File

@ -1108,7 +1108,7 @@ public:
// Naming // Naming
const char* signature_name() const; const char* signature_name() const;
static const jbyte* package_from_name(const Symbol* name, int& length); static Symbol* package_from_name(const Symbol* name, TRAPS);
// GC specific object visitors // GC specific object visitors
// //
@ -1298,7 +1298,7 @@ public:
// JSR-292 support // JSR-292 support
MemberNameTable* member_names() { return _member_names; } MemberNameTable* member_names() { return _member_names; }
void set_member_names(MemberNameTable* member_names) { _member_names = member_names; } void set_member_names(MemberNameTable* member_names) { _member_names = member_names; }
bool add_member_name(Handle member_name); oop add_member_name(Handle member_name, bool intern);
public: public:
// JVMTI support // JVMTI support

View File

@ -246,7 +246,7 @@ class Method : public Metadata {
int code_size() const { return constMethod()->code_size(); } int code_size() const { return constMethod()->code_size(); }
// method size in words // method size in words
int method_size() const { return sizeof(Method)/wordSize + is_native() ? 2 : 0; } int method_size() const { return sizeof(Method)/wordSize + ( is_native() ? 2 : 0 ); }
// constant pool for Klass* holding this method // constant pool for Klass* holding this method
ConstantPool* constants() const { return constMethod()->constants(); } ConstantPool* constants() const { return constMethod()->constants(); }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -204,7 +204,8 @@ Node* ArrayCopyNode::try_clone_instance(PhaseGVN *phase, bool can_reshape, int c
} }
if (!finish_transform(phase, can_reshape, ctl, mem)) { if (!finish_transform(phase, can_reshape, ctl, mem)) {
return NULL; // Return NodeSentinel to indicate that the transform failed
return NodeSentinel;
} }
return mem; return mem;
@ -222,6 +223,7 @@ bool ArrayCopyNode::prepare_array_copy(PhaseGVN *phase, bool can_reshape,
Node* dest = in(ArrayCopyNode::Dest); Node* dest = in(ArrayCopyNode::Dest);
const Type* src_type = phase->type(src); const Type* src_type = phase->type(src);
const TypeAryPtr* ary_src = src_type->isa_aryptr(); const TypeAryPtr* ary_src = src_type->isa_aryptr();
assert(ary_src != NULL, "should be an array copy/clone");
if (is_arraycopy() || is_copyofrange() || is_copyof()) { if (is_arraycopy() || is_copyofrange() || is_copyof()) {
const Type* dest_type = phase->type(dest); const Type* dest_type = phase->type(dest);
@ -520,7 +522,7 @@ Node *ArrayCopyNode::Ideal(PhaseGVN *phase, bool can_reshape) {
Node* mem = try_clone_instance(phase, can_reshape, count); Node* mem = try_clone_instance(phase, can_reshape, count);
if (mem != NULL) { if (mem != NULL) {
return mem; return (mem == NodeSentinel) ? NULL : mem;
} }
Node* adr_src = NULL; Node* adr_src = NULL;
@ -627,31 +629,37 @@ bool ArrayCopyNode::may_modify(const TypeOopPtr *t_oop, PhaseTransform *phase) {
return CallNode::may_modify_arraycopy_helper(dest_t, t_oop, phase); return CallNode::may_modify_arraycopy_helper(dest_t, t_oop, phase);
} }
bool ArrayCopyNode::may_modify_helper(const TypeOopPtr *t_oop, Node* n, PhaseTransform *phase) { bool ArrayCopyNode::may_modify_helper(const TypeOopPtr *t_oop, Node* n, PhaseTransform *phase, ArrayCopyNode*& ac) {
if (n->is_Proj()) { if (n->is_Proj()) {
n = n->in(0); n = n->in(0);
if (n->is_Call() && n->as_Call()->may_modify(t_oop, phase)) { if (n->is_Call() && n->as_Call()->may_modify(t_oop, phase)) {
if (n->isa_ArrayCopy() != NULL) {
ac = n->as_ArrayCopy();
}
return true; return true;
} }
} }
return false; return false;
} }
bool ArrayCopyNode::may_modify(const TypeOopPtr *t_oop, MemBarNode* mb, PhaseTransform *phase) { bool ArrayCopyNode::may_modify(const TypeOopPtr *t_oop, MemBarNode* mb, PhaseTransform *phase, ArrayCopyNode*& ac) {
Node* mem = mb->in(TypeFunc::Memory); Node* mem = mb->in(TypeFunc::Memory);
if (mem->is_MergeMem()) { if (mem->is_MergeMem()) {
Node* n = mem->as_MergeMem()->memory_at(Compile::AliasIdxRaw); Node* n = mem->as_MergeMem()->memory_at(Compile::AliasIdxRaw);
if (may_modify_helper(t_oop, n, phase)) { if (may_modify_helper(t_oop, n, phase, ac)) {
return true; return true;
} else if (n->is_Phi()) { } else if (n->is_Phi()) {
for (uint i = 1; i < n->req(); i++) { for (uint i = 1; i < n->req(); i++) {
if (n->in(i) != NULL) { if (n->in(i) != NULL) {
if (may_modify_helper(t_oop, n->in(i), phase)) { if (may_modify_helper(t_oop, n->in(i), phase, ac)) {
return true; return true;
} }
} }
} }
} else if (n->Opcode() == Op_StoreCM) {
// Ignore card mark stores
return may_modify_helper(t_oop, n->in(MemNode::Memory), phase, ac);
} }
} }

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -107,7 +107,7 @@ private:
BasicType copy_type, const Type* value_type, int count); BasicType copy_type, const Type* value_type, int count);
bool finish_transform(PhaseGVN *phase, bool can_reshape, bool finish_transform(PhaseGVN *phase, bool can_reshape,
Node* ctl, Node *mem); Node* ctl, Node *mem);
static bool may_modify_helper(const TypeOopPtr *t_oop, Node* n, PhaseTransform *phase); static bool may_modify_helper(const TypeOopPtr *t_oop, Node* n, PhaseTransform *phase, ArrayCopyNode*& ac);
public: public:
@ -162,7 +162,7 @@ public:
bool is_alloc_tightly_coupled() const { return _alloc_tightly_coupled; } bool is_alloc_tightly_coupled() const { return _alloc_tightly_coupled; }
static bool may_modify(const TypeOopPtr *t_oop, MemBarNode* mb, PhaseTransform *phase); static bool may_modify(const TypeOopPtr *t_oop, MemBarNode* mb, PhaseTransform *phase, ArrayCopyNode*& ac);
bool modifies(intptr_t offset_lo, intptr_t offset_hi, PhaseTransform* phase, bool must_modify); bool modifies(intptr_t offset_lo, intptr_t offset_hi, PhaseTransform* phase, bool must_modify);
#ifndef PRODUCT #ifndef PRODUCT

View File

@ -4306,8 +4306,15 @@ void GraphKit::g1_write_barrier_post(Node* oop_store,
} __ end_if(); } __ end_if();
} __ end_if(); } __ end_if();
} else { } else {
// Object.clone() instrinsic uses this path. // The Object.clone() intrinsic uses this path if !ReduceInitialCardMarks.
// We don't need a barrier here if the destination is a newly allocated object
// in Eden. Otherwise, GC verification breaks because we assume that cards in Eden
// are set to 'g1_young_gen' (see G1SATBCardTableModRefBS::verify_g1_young_region()).
assert(!use_ReduceInitialCardMarks(), "can only happen with card marking");
Node* card_val = __ load(__ ctrl(), card_adr, TypeInt::INT, T_BYTE, Compile::AliasIdxRaw);
__ if_then(card_val, BoolTest::ne, young_card); {
g1_mark_card(ideal, card_adr, oop_store, alias_idx, index, index_adr, buffer, tf); g1_mark_card(ideal, card_adr, oop_store, alias_idx, index, index_adr, buffer, tf);
} __ end_if();
} }
// Final sync IdealKit and GraphKit. // Final sync IdealKit and GraphKit.

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2005, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2005, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -32,6 +32,7 @@
#include "opto/cfgnode.hpp" #include "opto/cfgnode.hpp"
#include "opto/compile.hpp" #include "opto/compile.hpp"
#include "opto/convertnode.hpp" #include "opto/convertnode.hpp"
#include "opto/graphKit.hpp"
#include "opto/locknode.hpp" #include "opto/locknode.hpp"
#include "opto/loopnode.hpp" #include "opto/loopnode.hpp"
#include "opto/macro.hpp" #include "opto/macro.hpp"
@ -263,7 +264,7 @@ void PhaseMacroExpand::eliminate_card_mark(Node* p2x) {
// checks if the store done to a different from the value's region. // checks if the store done to a different from the value's region.
// And replace Cmp with #0 (false) to collapse G1 post barrier. // And replace Cmp with #0 (false) to collapse G1 post barrier.
Node* xorx = p2x->find_out_with(Op_XorX); Node* xorx = p2x->find_out_with(Op_XorX);
assert(xorx != NULL, "missing G1 post barrier"); if (xorx != NULL) {
Node* shift = xorx->unique_out(); Node* shift = xorx->unique_out();
Node* cmpx = shift->unique_out(); Node* cmpx = shift->unique_out();
assert(cmpx->is_Cmp() && cmpx->unique_out()->is_Bool() && assert(cmpx->is_Cmp() && cmpx->unique_out()->is_Bool() &&
@ -299,6 +300,23 @@ void PhaseMacroExpand::eliminate_card_mark(Node* p2x) {
} }
} }
} }
} else {
assert(!GraphKit::use_ReduceInitialCardMarks(), "can only happen with card marking");
// This is a G1 post barrier emitted by the Object.clone() intrinsic.
// Search for the CastP2X->URShiftX->AddP->LoadB->Cmp path which checks if the card
// is marked as young_gen and replace the Cmp with 0 (false) to collapse the barrier.
Node* shift = p2x->find_out_with(Op_URShiftX);
assert(shift != NULL, "missing G1 post barrier");
Node* addp = shift->unique_out();
Node* load = addp->find_out_with(Op_LoadB);
assert(load != NULL, "missing G1 post barrier");
Node* cmpx = load->unique_out();
assert(cmpx->is_Cmp() && cmpx->unique_out()->is_Bool() &&
cmpx->unique_out()->as_Bool()->_test._test == BoolTest::ne,
"missing card value check in G1 post barrier");
_igvn.replace_node(cmpx, makecon(TypeInt::CC_EQ));
// There is no G1 pre barrier in this case
}
// Now CastP2X can be removed since it is used only on dead path // Now CastP2X can be removed since it is used only on dead path
// which currently still alive until igvn optimize it. // which currently still alive until igvn optimize it.
assert(p2x->outcnt() == 0 || p2x->unique_out()->Opcode() == Op_URShiftX, ""); assert(p2x->outcnt() == 0 || p2x->unique_out()->Opcode() == Op_URShiftX, "");
@ -326,17 +344,15 @@ static Node *scan_mem_chain(Node *mem, int alias_idx, int offset, Node *start_me
CallNode *call = in->as_Call(); CallNode *call = in->as_Call();
if (call->may_modify(tinst, phase)) { if (call->may_modify(tinst, phase)) {
assert(call->is_ArrayCopy(), "ArrayCopy is the only call node that doesn't make allocation escape"); assert(call->is_ArrayCopy(), "ArrayCopy is the only call node that doesn't make allocation escape");
if (call->as_ArrayCopy()->modifies(offset, offset, phase, false)) { if (call->as_ArrayCopy()->modifies(offset, offset, phase, false)) {
return in; return in;
} }
} }
mem = in->in(TypeFunc::Memory); mem = in->in(TypeFunc::Memory);
} else if (in->is_MemBar()) { } else if (in->is_MemBar()) {
if (ArrayCopyNode::may_modify(tinst, in->as_MemBar(), phase)) { ArrayCopyNode* ac = NULL;
assert(in->in(0)->is_Proj() && in->in(0)->in(0)->is_ArrayCopy(), "should be arraycopy"); if (ArrayCopyNode::may_modify(tinst, in->as_MemBar(), phase, ac)) {
ArrayCopyNode* ac = in->in(0)->in(0)->as_ArrayCopy(); assert(ac != NULL && ac->is_clonebasic(), "Only basic clone is a non escaping clone");
assert(ac->is_clonebasic(), "Only basic clone is a non escaping clone");
return ac; return ac;
} }
mem = in->in(TypeFunc::Memory); mem = in->in(TypeFunc::Memory);

View File

@ -160,7 +160,8 @@ Node *MemNode::optimize_simple_memory_chain(Node *mchain, const TypeOopPtr *t_oo
} }
} }
} else if (proj_in->is_MemBar()) { } else if (proj_in->is_MemBar()) {
if (ArrayCopyNode::may_modify(t_oop, proj_in->as_MemBar(), phase)) { ArrayCopyNode* ac = NULL;
if (ArrayCopyNode::may_modify(t_oop, proj_in->as_MemBar(), phase, ac)) {
break; break;
} }
result = proj_in->in(TypeFunc::Memory); result = proj_in->in(TypeFunc::Memory);
@ -657,7 +658,8 @@ Node* MemNode::find_previous_store(PhaseTransform* phase) {
continue; // (a) advance through independent call memory continue; // (a) advance through independent call memory
} }
} else if (mem->is_Proj() && mem->in(0)->is_MemBar()) { } else if (mem->is_Proj() && mem->in(0)->is_MemBar()) {
if (ArrayCopyNode::may_modify(addr_t, mem->in(0)->as_MemBar(), phase)) { ArrayCopyNode* ac = NULL;
if (ArrayCopyNode::may_modify(addr_t, mem->in(0)->as_MemBar(), phase, ac)) {
break; break;
} }
mem = mem->in(0)->in(TypeFunc::Memory); mem = mem->in(0)->in(TypeFunc::Memory);

View File

@ -166,14 +166,11 @@ class Parse : public GraphKit {
int _all_successors; // Include exception paths also. int _all_successors; // Include exception paths also.
Block** _successors; Block** _successors;
// Use init_node/init_graph to initialize Blocks.
// Block() : _live_locals((uintptr_t*)NULL,0) { ShouldNotReachHere(); }
Block() : _live_locals() { ShouldNotReachHere(); }
public: public:
// Set up the block data structure itself. // Set up the block data structure itself.
void init_node(Parse* outer, int po); Block(Parse* outer, int rpo);
// Set up the block's relations to other blocks. // Set up the block's relations to other blocks.
void init_graph(Parse* outer); void init_graph(Parse* outer);

View File

@ -1235,29 +1235,33 @@ void Parse::init_blocks() {
// Create the blocks. // Create the blocks.
_block_count = flow()->block_count(); _block_count = flow()->block_count();
_blocks = NEW_RESOURCE_ARRAY(Block, _block_count); _blocks = NEW_RESOURCE_ARRAY(Block, _block_count);
Copy::zero_to_bytes(_blocks, sizeof(Block)*_block_count);
int rpo;
// Initialize the structs. // Initialize the structs.
for (rpo = 0; rpo < block_count(); rpo++) { for (int rpo = 0; rpo < block_count(); rpo++) {
Block* block = rpo_at(rpo); Block* block = rpo_at(rpo);
block->init_node(this, rpo); new(block) Block(this, rpo);
} }
// Collect predecessor and successor information. // Collect predecessor and successor information.
for (rpo = 0; rpo < block_count(); rpo++) { for (int rpo = 0; rpo < block_count(); rpo++) {
Block* block = rpo_at(rpo); Block* block = rpo_at(rpo);
block->init_graph(this); block->init_graph(this);
} }
} }
//-------------------------------init_node------------------------------------- //-------------------------------init_node-------------------------------------
void Parse::Block::init_node(Parse* outer, int rpo) { Parse::Block::Block(Parse* outer, int rpo) : _live_locals() {
_flow = outer->flow()->rpo_at(rpo); _flow = outer->flow()->rpo_at(rpo);
_pred_count = 0; _pred_count = 0;
_preds_parsed = 0; _preds_parsed = 0;
_count = 0; _count = 0;
_is_parsed = false;
_is_handler = false;
_has_merged_backedge = false;
_start_map = NULL;
_num_successors = 0;
_all_successors = 0;
_successors = NULL;
assert(pred_count() == 0 && preds_parsed() == 0, "sanity"); assert(pred_count() == 0 && preds_parsed() == 0, "sanity");
assert(!(is_merged() || is_parsed() || is_handler() || has_merged_backedge()), "sanity"); assert(!(is_merged() || is_parsed() || is_handler() || has_merged_backedge()), "sanity");
assert(_live_locals.size() == 0, "sanity"); assert(_live_locals.size() == 0, "sanity");

View File

@ -695,7 +695,7 @@ JVM_ENTRY(jobject, JVM_Clone(JNIEnv* env, jobject handle))
// This can safepoint and redefine method, so need both new_obj and method // This can safepoint and redefine method, so need both new_obj and method
// in a handle, for two different reasons. new_obj can move, method can be // in a handle, for two different reasons. new_obj can move, method can be
// deleted if nothing is using it on the stack. // deleted if nothing is using it on the stack.
m->method_holder()->add_member_name(new_obj()); m->method_holder()->add_member_name(new_obj(), false);
} }
} }

View File

@ -319,15 +319,7 @@ jvmtiError
JvmtiEnv::GetObjectSize(jobject object, jlong* size_ptr) { JvmtiEnv::GetObjectSize(jobject object, jlong* size_ptr) {
oop mirror = JNIHandles::resolve_external_guard(object); oop mirror = JNIHandles::resolve_external_guard(object);
NULL_CHECK(mirror, JVMTI_ERROR_INVALID_OBJECT); NULL_CHECK(mirror, JVMTI_ERROR_INVALID_OBJECT);
if (mirror->klass() == SystemDictionary::Class_klass() &&
!java_lang_Class::is_primitive(mirror)) {
Klass* k = java_lang_Class::as_Klass(mirror);
assert(k != NULL, "class for non-primitive mirror must exist");
*size_ptr = (jlong)k->size() * wordSize;
} else {
*size_ptr = (jlong)mirror->size() * wordSize; *size_ptr = (jlong)mirror->size() * wordSize;
}
return JVMTI_ERROR_NONE; return JVMTI_ERROR_NONE;
} /* end GetObjectSize */ } /* end GetObjectSize */

View File

@ -178,7 +178,7 @@ oop MethodHandles::init_MemberName(Handle mname, Handle target) {
return NULL; return NULL;
} }
oop MethodHandles::init_method_MemberName(Handle mname, CallInfo& info) { oop MethodHandles::init_method_MemberName(Handle mname, CallInfo& info, bool intern) {
assert(info.resolved_appendix().is_null(), "only normal methods here"); assert(info.resolved_appendix().is_null(), "only normal methods here");
methodHandle m = info.resolved_method(); methodHandle m = info.resolved_method();
assert(m.not_null(), "null method handle"); assert(m.not_null(), "null method handle");
@ -279,13 +279,7 @@ oop MethodHandles::init_method_MemberName(Handle mname, CallInfo& info) {
// If relevant, the vtable or itable value is stored as vmindex. // If relevant, the vtable or itable value is stored as vmindex.
// This is done eagerly, since it is readily available without // This is done eagerly, since it is readily available without
// constructing any new objects. // constructing any new objects.
// TO DO: maybe intern mname_oop return m->method_holder()->add_member_name(mname, intern);
if (m->method_holder()->add_member_name(mname)) {
return mname();
} else {
// Redefinition caused this to fail. Return NULL (and an exception?)
return NULL;
}
} }
oop MethodHandles::init_field_MemberName(Handle mname, fieldDescriptor& fd, bool is_setter) { oop MethodHandles::init_field_MemberName(Handle mname, fieldDescriptor& fd, bool is_setter) {
@ -975,7 +969,9 @@ int MethodHandles::find_MemberNames(KlassHandle k,
if (!java_lang_invoke_MemberName::is_instance(result())) if (!java_lang_invoke_MemberName::is_instance(result()))
return -99; // caller bug! return -99; // caller bug!
CallInfo info(m); CallInfo info(m);
oop saved = MethodHandles::init_method_MemberName(result, info); // Since this is going through the methods to create MemberNames, don't search
// for matching methods already in the table
oop saved = MethodHandles::init_method_MemberName(result, info, /*intern*/false);
if (saved != result()) if (saved != result())
results->obj_at_put(rfill-1, saved); // show saved instance to user results->obj_at_put(rfill-1, saved); // show saved instance to user
} else if (++overflow >= overflow_limit) { } else if (++overflow >= overflow_limit) {
@ -1056,9 +1052,34 @@ MemberNameTable::~MemberNameTable() {
} }
} }
void MemberNameTable::add_member_name(jweak mem_name_wref) { oop MemberNameTable::add_member_name(jweak mem_name_wref) {
assert_locked_or_safepoint(MemberNameTable_lock); assert_locked_or_safepoint(MemberNameTable_lock);
this->push(mem_name_wref); this->push(mem_name_wref);
return JNIHandles::resolve(mem_name_wref);
}
oop MemberNameTable::find_or_add_member_name(jweak mem_name_wref) {
assert_locked_or_safepoint(MemberNameTable_lock);
oop new_mem_name = JNIHandles::resolve(mem_name_wref);
// Find matching member name in the list.
// This is linear because these because these are short lists.
int len = this->length();
int new_index = len;
for (int idx = 0; idx < len; idx++) {
oop mname = JNIHandles::resolve(this->at(idx));
if (mname == NULL) {
new_index = idx;
continue;
}
if (java_lang_invoke_MemberName::equals(new_mem_name, mname)) {
JNIHandles::destroy_weak_global(mem_name_wref);
return mname;
}
}
// Not found, push the new one, or reuse empty slot
this->at_put_grow(new_index, mem_name_wref);
return new_mem_name;
} }
#if INCLUDE_JVMTI #if INCLUDE_JVMTI

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2008, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2008, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -66,7 +66,7 @@ class MethodHandles: AllStatic {
static Handle new_MemberName(TRAPS); // must be followed by init_MemberName static Handle new_MemberName(TRAPS); // must be followed by init_MemberName
static oop init_MemberName(Handle mname_h, Handle target_h); // compute vmtarget/vmindex from target static oop init_MemberName(Handle mname_h, Handle target_h); // compute vmtarget/vmindex from target
static oop init_field_MemberName(Handle mname_h, fieldDescriptor& fd, bool is_setter = false); static oop init_field_MemberName(Handle mname_h, fieldDescriptor& fd, bool is_setter = false);
static oop init_method_MemberName(Handle mname_h, CallInfo& info); static oop init_method_MemberName(Handle mname_h, CallInfo& info, bool intern = true);
static int method_ref_kind(Method* m, bool do_dispatch_if_possible = true); static int method_ref_kind(Method* m, bool do_dispatch_if_possible = true);
static int find_MemberNames(KlassHandle k, Symbol* name, Symbol* sig, static int find_MemberNames(KlassHandle k, Symbol* name, Symbol* sig,
int mflags, KlassHandle caller, int mflags, KlassHandle caller,
@ -253,7 +253,8 @@ class MemberNameTable : public GrowableArray<jweak> {
public: public:
MemberNameTable(int methods_cnt); MemberNameTable(int methods_cnt);
~MemberNameTable(); ~MemberNameTable();
void add_member_name(jweak mem_name_ref); oop add_member_name(jweak mem_name_ref);
oop find_or_add_member_name(jweak mem_name_ref);
#if INCLUDE_JVMTI #if INCLUDE_JVMTI
// RedefineClasses() API support: // RedefineClasses() API support:

View File

@ -356,19 +356,6 @@ UNSAFE_ENTRY(jobject, Unsafe_GetUncompressedObject(JNIEnv *env, jobject unsafe,
return JNIHandles::make_local(env, v); return JNIHandles::make_local(env, v);
} UNSAFE_END } UNSAFE_END
UNSAFE_ENTRY(jclass, Unsafe_GetJavaMirror(JNIEnv *env, jobject unsafe, jlong metaspace_klass)) {
Klass* klass = (Klass*) (address) metaspace_klass;
return (jclass) JNIHandles::make_local(klass->java_mirror());
} UNSAFE_END
UNSAFE_ENTRY(jlong, Unsafe_GetKlassPointer(JNIEnv *env, jobject unsafe, jobject obj)) {
oop o = JNIHandles::resolve(obj);
jlong klass = (jlong) (address) o->klass();
return klass;
} UNSAFE_END
#ifndef SUPPORTS_NATIVE_CX8 #ifndef SUPPORTS_NATIVE_CX8
// VM_Version::supports_cx8() is a surrogate for 'supports atomic long memory ops'. // VM_Version::supports_cx8() is a surrogate for 'supports atomic long memory ops'.
@ -1152,8 +1139,6 @@ static JNINativeMethod jdk_internal_misc_Unsafe_methods[] = {
{CC "putObjectVolatile",CC "(" OBJ "J" OBJ ")V", FN_PTR(Unsafe_PutObjectVolatile)}, {CC "putObjectVolatile",CC "(" OBJ "J" OBJ ")V", FN_PTR(Unsafe_PutObjectVolatile)},
{CC "getUncompressedObject", CC "(" ADR ")" OBJ, FN_PTR(Unsafe_GetUncompressedObject)}, {CC "getUncompressedObject", CC "(" ADR ")" OBJ, FN_PTR(Unsafe_GetUncompressedObject)},
{CC "getJavaMirror", CC "(" ADR ")" CLS, FN_PTR(Unsafe_GetJavaMirror)},
{CC "getKlassPointer", CC "(" OBJ ")" ADR, FN_PTR(Unsafe_GetKlassPointer)},
DECLARE_GETPUTOOP(Boolean, Z), DECLARE_GETPUTOOP(Boolean, Z),
DECLARE_GETPUTOOP(Byte, B), DECLARE_GETPUTOOP(Byte, B),

View File

@ -1433,6 +1433,10 @@ WB_ENTRY(jlong, WB_MetaspaceCapacityUntilGC(JNIEnv* env, jobject wb))
return (jlong) MetaspaceGC::capacity_until_GC(); return (jlong) MetaspaceGC::capacity_until_GC();
WB_END WB_END
WB_ENTRY(jboolean, WB_MetaspaceShouldConcurrentCollect(JNIEnv* env, jobject wb))
return MetaspaceGC::should_concurrent_collect();
WB_END
WB_ENTRY(void, WB_AssertMatchingSafepointCalls(JNIEnv* env, jobject o, jboolean mutexSafepointValue, jboolean attemptedNoSafepointValue)) WB_ENTRY(void, WB_AssertMatchingSafepointCalls(JNIEnv* env, jobject o, jboolean mutexSafepointValue, jboolean attemptedNoSafepointValue))
Monitor::SafepointCheckRequired sfpt_check_required = mutexSafepointValue ? Monitor::SafepointCheckRequired sfpt_check_required = mutexSafepointValue ?
@ -1813,6 +1817,7 @@ static JNINativeMethod methods[] = {
CC"(Ljava/lang/ClassLoader;JJ)V", (void*)&WB_FreeMetaspace }, CC"(Ljava/lang/ClassLoader;JJ)V", (void*)&WB_FreeMetaspace },
{CC"incMetaspaceCapacityUntilGC", CC"(J)J", (void*)&WB_IncMetaspaceCapacityUntilGC }, {CC"incMetaspaceCapacityUntilGC", CC"(J)J", (void*)&WB_IncMetaspaceCapacityUntilGC },
{CC"metaspaceCapacityUntilGC", CC"()J", (void*)&WB_MetaspaceCapacityUntilGC }, {CC"metaspaceCapacityUntilGC", CC"()J", (void*)&WB_MetaspaceCapacityUntilGC },
{CC"metaspaceShouldConcurrentCollect", CC"()Z", (void*)&WB_MetaspaceShouldConcurrentCollect },
{CC"getCPUFeatures", CC"()Ljava/lang/String;", (void*)&WB_GetCPUFeatures }, {CC"getCPUFeatures", CC"()Ljava/lang/String;", (void*)&WB_GetCPUFeatures },
{CC"getNMethod0", CC"(Ljava/lang/reflect/Executable;Z)[Ljava/lang/Object;", {CC"getNMethod0", CC"(Ljava/lang/reflect/Executable;Z)[Ljava/lang/Object;",
(void*)&WB_GetNMethod }, (void*)&WB_GetNMethod },

View File

@ -498,6 +498,19 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread
} }
#endif #endif
if (thread->frames_to_pop_failed_realloc() > 0 && exec_mode != Unpack_uncommon_trap) {
assert(thread->has_pending_exception(), "should have thrown OOME");
thread->set_exception_oop(thread->pending_exception());
thread->clear_pending_exception();
exec_mode = Unpack_exception;
}
#if INCLUDE_JVMCI
if (thread->frames_to_pop_failed_realloc() > 0) {
thread->set_pending_monitorenter(false);
}
#endif
UnrollBlock* info = new UnrollBlock(array->frame_size() * BytesPerWord, UnrollBlock* info = new UnrollBlock(array->frame_size() * BytesPerWord,
caller_adjustment * BytesPerWord, caller_adjustment * BytesPerWord,
caller_was_method_handle ? 0 : callee_parameters, caller_was_method_handle ? 0 : callee_parameters,

View File

@ -446,6 +446,14 @@ void before_exit(JavaThread* thread) {
os::infinite_sleep(); os::infinite_sleep();
} }
EventThreadEnd event;
if (event.should_commit()) {
event.set_thread(THREAD_TRACE_ID(thread));
event.commit();
}
TRACE_VM_EXIT();
// Stop the WatcherThread. We do this before disenrolling various // Stop the WatcherThread. We do this before disenrolling various
// PeriodicTasks to reduce the likelihood of races. // PeriodicTasks to reduce the likelihood of races.
if (PeriodicTask::num_tasks() > 0) { if (PeriodicTask::num_tasks() > 0) {
@ -484,13 +492,6 @@ void before_exit(JavaThread* thread) {
JvmtiExport::post_thread_end(thread); JvmtiExport::post_thread_end(thread);
} }
EventThreadEnd event;
if (event.should_commit()) {
event.set_thread(THREAD_TRACE_ID(thread));
event.commit();
}
// Always call even when there are not JVMTI environments yet, since environments // Always call even when there are not JVMTI environments yet, since environments
// may be attached late and JVMTI must track phases of VM execution // may be attached late and JVMTI must track phases of VM execution
JvmtiExport::post_vm_death(); JvmtiExport::post_vm_death();

View File

@ -274,7 +274,7 @@ void mutex_init() {
def(JfrMsg_lock , Monitor, leaf, true, Monitor::_safepoint_check_always); def(JfrMsg_lock , Monitor, leaf, true, Monitor::_safepoint_check_always);
def(JfrBuffer_lock , Mutex, leaf, true, Monitor::_safepoint_check_never); def(JfrBuffer_lock , Mutex, leaf, true, Monitor::_safepoint_check_never);
def(JfrThreadGroups_lock , Mutex, leaf, true, Monitor::_safepoint_check_always); def(JfrThreadGroups_lock , Mutex, leaf, true, Monitor::_safepoint_check_always);
def(JfrStream_lock , Mutex, nonleaf, true, Monitor::_safepoint_check_never); def(JfrStream_lock , Mutex, leaf+1, true, Monitor::_safepoint_check_never); // ensure to rank lower than 'safepoint'
def(JfrStacktrace_lock , Mutex, special, true, Monitor::_safepoint_check_sometimes); def(JfrStacktrace_lock , Mutex, special, true, Monitor::_safepoint_check_sometimes);
#endif #endif

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 1999, 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -73,10 +73,7 @@ GrowableArray<FilteredField*> *FilteredFieldsMap::_filtered_fields =
void FilteredFieldsMap::initialize() { void FilteredFieldsMap::initialize() {
int offset; int offset = reflect_ConstantPool::oop_offset();
offset = java_lang_Throwable::get_backtrace_offset();
_filtered_fields->append(new FilteredField(SystemDictionary::Throwable_klass(), offset));
offset = reflect_ConstantPool::oop_offset();
_filtered_fields->append(new FilteredField(SystemDictionary::reflect_ConstantPool_klass(), offset)); _filtered_fields->append(new FilteredField(SystemDictionary::reflect_ConstantPool_klass(), offset));
offset = reflect_UnsafeStaticFieldAccessorImpl::base_offset(); offset = reflect_UnsafeStaticFieldAccessorImpl::base_offset();
_filtered_fields->append(new FilteredField(SystemDictionary::reflect_UnsafeStaticFieldAccessorImpl_klass(), offset)); _filtered_fields->append(new FilteredField(SystemDictionary::reflect_UnsafeStaticFieldAccessorImpl_klass(), offset));

View File

@ -3405,6 +3405,8 @@ static void call_initPhase1(TRAPS) {
// //
// After phase 2, The VM will begin search classes from -Xbootclasspath/a. // After phase 2, The VM will begin search classes from -Xbootclasspath/a.
static void call_initPhase2(TRAPS) { static void call_initPhase2(TRAPS) {
TraceTime timer("Phase2 initialization", TRACETIME_LOG(Info, modules, startuptime));
Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK); Klass* k = SystemDictionary::resolve_or_fail(vmSymbols::java_lang_System(), true, CHECK);
instanceKlassHandle klass (THREAD, k); instanceKlassHandle klass (THREAD, k);

View File

@ -71,9 +71,12 @@ inline jlong Thread::cooked_allocated_bytes() {
jlong allocated_bytes = OrderAccess::load_acquire(&_allocated_bytes); jlong allocated_bytes = OrderAccess::load_acquire(&_allocated_bytes);
if (UseTLAB) { if (UseTLAB) {
size_t used_bytes = tlab().used_bytes(); size_t used_bytes = tlab().used_bytes();
if ((ssize_t)used_bytes > 0) { if (used_bytes <= ThreadLocalAllocBuffer::max_size_in_bytes()) {
// More-or-less valid tlab. The load_acquire above should ensure // Comparing used_bytes with the maximum allowed size will ensure
// that the result of the add is <= the instantaneous value. // that we don't add the used bytes from a semi-initialized TLAB
// ending up with incorrect values. There is still a race between
// incrementing _allocated_bytes and clearing the TLAB, that might
// cause double counting in rare cases.
return allocated_bytes + used_bytes; return allocated_bytes + used_bytes;
} }
} }

View File

@ -171,6 +171,8 @@ void vframeArrayElement::unpack_on_stack(int caller_actual_parameters,
int exec_mode) { int exec_mode) {
JavaThread* thread = (JavaThread*) Thread::current(); JavaThread* thread = (JavaThread*) Thread::current();
bool realloc_failure_exception = thread->frames_to_pop_failed_realloc() > 0;
// Look at bci and decide on bcp and continuation pc // Look at bci and decide on bcp and continuation pc
address bcp; address bcp;
// C++ interpreter doesn't need a pc since it will figure out what to do when it // C++ interpreter doesn't need a pc since it will figure out what to do when it
@ -204,10 +206,12 @@ void vframeArrayElement::unpack_on_stack(int caller_actual_parameters,
// //
// For Compiler1, deoptimization can occur while throwing a NullPointerException at monitorenter, // For Compiler1, deoptimization can occur while throwing a NullPointerException at monitorenter,
// in which case bcp should point to the monitorenter since it is within the exception's range. // in which case bcp should point to the monitorenter since it is within the exception's range.
//
// For realloc failure exception we just pop frames, skip the guarantee.
assert(*bcp != Bytecodes::_monitorenter || is_top_frame, "a _monitorenter must be a top frame"); assert(*bcp != Bytecodes::_monitorenter || is_top_frame, "a _monitorenter must be a top frame");
assert(thread->deopt_compiled_method() != NULL, "compiled method should be known"); assert(thread->deopt_compiled_method() != NULL, "compiled method should be known");
guarantee(!(thread->deopt_compiled_method()->is_compiled_by_c2() && guarantee(realloc_failure_exception || !(thread->deopt_compiled_method()->is_compiled_by_c2() &&
*bcp == Bytecodes::_monitorenter && *bcp == Bytecodes::_monitorenter &&
exec_mode == Deoptimization::Unpack_exception), exec_mode == Deoptimization::Unpack_exception),
"shouldn't get exception during monitorenter"); "shouldn't get exception during monitorenter");
@ -237,12 +241,17 @@ void vframeArrayElement::unpack_on_stack(int caller_actual_parameters,
// Deoptimization::fetch_unroll_info_helper // Deoptimization::fetch_unroll_info_helper
popframe_preserved_args_size_in_words = in_words(thread->popframe_preserved_args_size_in_words()); popframe_preserved_args_size_in_words = in_words(thread->popframe_preserved_args_size_in_words());
} }
} else if (JvmtiExport::can_force_early_return() && state != NULL && state->is_earlyret_pending()) { } else if (!realloc_failure_exception && JvmtiExport::can_force_early_return() && state != NULL && state->is_earlyret_pending()) {
// Force early return from top frame after deoptimization // Force early return from top frame after deoptimization
#ifndef CC_INTERP #ifndef CC_INTERP
pc = Interpreter::remove_activation_early_entry(state->earlyret_tos()); pc = Interpreter::remove_activation_early_entry(state->earlyret_tos());
#endif #endif
} else { } else {
if (realloc_failure_exception && JvmtiExport::can_force_early_return() && state != NULL && state->is_earlyret_pending()) {
state->clr_earlyret_pending();
state->set_earlyret_oop(NULL);
state->clr_earlyret_value();
}
// Possibly override the previous pc computation of the top (youngest) frame // Possibly override the previous pc computation of the top (youngest) frame
switch (exec_mode) { switch (exec_mode) {
case Deoptimization::Unpack_deopt: case Deoptimization::Unpack_deopt:

View File

@ -48,9 +48,6 @@ public:
static void on_unloading_classes(void) { static void on_unloading_classes(void) {
} }
static void on_vm_error(bool) {
}
}; };
class TraceThreadData { class TraceThreadData {

View File

@ -43,6 +43,8 @@ extern "C" void JNICALL trace_register_natives(JNIEnv*, jclass);
#define TRACE_REGISTER_NATIVES ((void*)((address_word)(&trace_register_natives))) #define TRACE_REGISTER_NATIVES ((void*)((address_word)(&trace_register_natives)))
#define TRACE_START() JNI_OK #define TRACE_START() JNI_OK
#define TRACE_INITIALIZE() JNI_OK #define TRACE_INITIALIZE() JNI_OK
#define TRACE_VM_EXIT()
#define TRACE_VM_ERROR()
#define TRACE_DEFINE_TRACE_ID_METHODS typedef int ___IGNORED_hs_trace_type1 #define TRACE_DEFINE_TRACE_ID_METHODS typedef int ___IGNORED_hs_trace_type1
#define TRACE_DEFINE_TRACE_ID_FIELD typedef int ___IGNORED_hs_trace_type2 #define TRACE_DEFINE_TRACE_ID_FIELD typedef int ___IGNORED_hs_trace_type2

View File

@ -435,7 +435,6 @@ class BitMap2D VALUE_OBJ_CLASS_SPEC {
void clear_bit(idx_t slot_index, idx_t bit_within_slot_index); void clear_bit(idx_t slot_index, idx_t bit_within_slot_index);
void at_put(idx_t slot_index, idx_t bit_within_slot_index, bool value); void at_put(idx_t slot_index, idx_t bit_within_slot_index, bool value);
void at_put_grow(idx_t slot_index, idx_t bit_within_slot_index, bool value); void at_put_grow(idx_t slot_index, idx_t bit_within_slot_index, bool value);
void clear();
}; };
// Closure for iterating over BitMaps // Closure for iterating over BitMaps

View File

@ -367,8 +367,4 @@ inline void BitMap2D::at_put_grow(idx_t slot_index, idx_t bit_within_slot_index,
_map.at_put(bit, value); _map.at_put(bit, value);
} }
inline void BitMap2D::clear() {
_map.clear();
}
#endif // SHARE_VM_UTILITIES_BITMAP_INLINE_HPP #endif // SHARE_VM_UTILITIES_BITMAP_INLINE_HPP

View File

@ -54,10 +54,6 @@
#include "utilities/macros.hpp" #include "utilities/macros.hpp"
#include "utilities/vmError.hpp" #include "utilities/vmError.hpp"
#if INCLUDE_TRACE
#include "trace/tracing.hpp"
#endif
#include <stdio.h> #include <stdio.h>
#ifndef ASSERT #ifndef ASSERT
@ -306,11 +302,6 @@ void report_out_of_shared_space(SharedSpaceType shared_space) {
exit(2); exit(2);
} }
static void notify_tracing() {
#if INCLUDE_TRACE
Tracing::on_vm_error(true);
#endif
}
void report_insufficient_metaspace(size_t required_size) { void report_insufficient_metaspace(size_t required_size) {
warning("\nThe MaxMetaspaceSize of " SIZE_FORMAT " bytes is not large enough.\n" warning("\nThe MaxMetaspaceSize of " SIZE_FORMAT " bytes is not large enough.\n"
@ -334,8 +325,6 @@ void report_java_out_of_memory(const char* message) {
HeapDumper::dump_heap_from_oome(); HeapDumper::dump_heap_from_oome();
} }
notify_tracing();
if (OnOutOfMemoryError && OnOutOfMemoryError[0]) { if (OnOutOfMemoryError && OnOutOfMemoryError[0]) {
VMError::report_java_out_of_memory(message); VMError::report_java_out_of_memory(message);
} }

View File

@ -39,6 +39,7 @@
#include "runtime/vmThread.hpp" #include "runtime/vmThread.hpp"
#include "runtime/vm_operations.hpp" #include "runtime/vm_operations.hpp"
#include "services/memTracker.hpp" #include "services/memTracker.hpp"
#include "trace/traceMacros.hpp"
#include "utilities/debug.hpp" #include "utilities/debug.hpp"
#include "utilities/decoder.hpp" #include "utilities/decoder.hpp"
#include "utilities/defaultStream.hpp" #include "utilities/defaultStream.hpp"
@ -1165,6 +1166,8 @@ void VMError::report_and_die(int id, const char* message, const char* detail_fmt
// are handled properly. // are handled properly.
reset_signal_handlers(); reset_signal_handlers();
TRACE_VM_ERROR();
} else { } else {
// If UseOsErrorReporting we call this for each level of the call stack // If UseOsErrorReporting we call this for each level of the call stack
// while searching for the exception handler. Only the first level needs // while searching for the exception handler. Only the first level needs

View File

@ -34,7 +34,7 @@ groups=TEST.groups [closed/TEST.groups]
# Source files for classes that will be used at the beginning of each test suite run, # Source files for classes that will be used at the beginning of each test suite run,
# to determine additional characteristics of the system for use with the @requires tag. # to determine additional characteristics of the system for use with the @requires tag.
requires.extraPropDefns = ../../test/jtreg-ext/requires/VMProps.java requires.extraPropDefns = ../../test/jtreg-ext/requires/VMProps.java
requires.properties=sun.arch.data.model requires.properties=sun.arch.data.model vm.simpleArch vm.flightRecorder
# Tests using jtreg 4.2 b02 features # Tests using jtreg 4.2 b02 features
requiredVersion=4.2 b02 requiredVersion=4.2 b02

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,10 +23,10 @@
/* /*
* @test * @test
* @bug 8130847 * @bug 8130847 8156760
* @summary Eliminated instance/array written to by an array copy variant must be correctly initialized when reallocated at a deopt * @summary Eliminated instance/array written to by an array copy variant must be correctly initialized when reallocated at a deopt
* @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement TestEliminatedArrayCopyDeopt * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement TestEliminatedArrayCopyDeopt
* * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:+IgnoreUnrecognizedVMOptions -XX:-ReduceInitialCardMarks TestEliminatedArrayCopyDeopt
*/ */
// Test that if an ArrayCopy node is eliminated because it doesn't // Test that if an ArrayCopy node is eliminated because it doesn't

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,12 +23,12 @@
/* /*
* @test * @test
* @bug 6700100 * @bug 6700100 8156760
* @summary small instance clone as loads/stores * @summary small instance clone as loads/stores
* @compile TestInstanceCloneAsLoadsStores.java TestInstanceCloneUtils.java * @compile TestInstanceCloneAsLoadsStores.java TestInstanceCloneUtils.java
* @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:CompileCommand=dontinline,TestInstanceCloneAsLoadsStores::m* TestInstanceCloneAsLoadsStores * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:CompileCommand=dontinline,TestInstanceCloneAsLoadsStores::m* TestInstanceCloneAsLoadsStores
* @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:CompileCommand=dontinline,TestInstanceCloneAsLoadsStores::m* -XX:+IgnoreUnrecognizedVMOptions -XX:+StressArrayCopyMacroNode TestInstanceCloneAsLoadsStores * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:CompileCommand=dontinline,TestInstanceCloneAsLoadsStores::m* -XX:+IgnoreUnrecognizedVMOptions -XX:+StressArrayCopyMacroNode TestInstanceCloneAsLoadsStores
* * @run main/othervm -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:CompileCommand=dontinline,TestInstanceCloneAsLoadsStores::m* -XX:+IgnoreUnrecognizedVMOptions -XX:-ReduceInitialCardMarks TestInstanceCloneAsLoadsStores
*/ */
public class TestInstanceCloneAsLoadsStores extends TestInstanceCloneUtils { public class TestInstanceCloneAsLoadsStores extends TestInstanceCloneUtils {

View File

@ -34,12 +34,12 @@
import jdk.test.lib.*; import jdk.test.lib.*;
public class Test6857159 { public class Test6857159 {
public static void main(String[] args) throws Exception { public static void main(String[] args) throws Throwable {
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder("-Xbatch", "-XX:+PrintCompilation", OutputAnalyzer analyzer = ProcessTools.executeTestJvm("-Xbatch",
"-XX:CompileOnly=Test$ct.run", "Test"); "-XX:+PrintCompilation", "-XX:CompileOnly=Test$ct.run", "Test");
OutputAnalyzer analyzer = new OutputAnalyzer(pb.start());
analyzer.shouldNotContain("COMPILE SKIPPED"); analyzer.shouldNotContain("COMPILE SKIPPED");
analyzer.shouldContain("Test$ct0::run (16 bytes)"); analyzer.shouldContain("Test$ct0::run (16 bytes)");
analyzer.shouldHaveExitValue(0);
} }
} }

View File

@ -1,54 +0,0 @@
#!/bin/sh
#
# Copyright (c) 2009, 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.
#
#
## some tests require path to find test source dir
if [ "${TESTSRC}" = "" ]
then
TESTSRC=${PWD}
echo "TESTSRC not set. Using "${TESTSRC}" as default"
fi
echo "TESTSRC=${TESTSRC}"
## Adding common setup Variables for running shell tests.
. ${TESTSRC}/../../../test_env.sh
set -x
cp ${TESTSRC}/Test6857159.java .
cp ${TESTSRC}/Test6857159.sh .
${COMPILEJAVA}/bin/javac ${TESTJAVACOPTS} -d . Test6857159.java
${TESTJAVA}/bin/java ${TESTOPTS} -Xbatch -XX:+PrintCompilation -XX:CompileOnly=Test6857159\$ct.run Test6857159 > test.out 2>&1
grep "COMPILE SKIPPED" test.out
result=$?
if [ $result -eq 1 ]
then
echo "Passed"
exit 0
else
echo "Failed"
exit 1
fi

View File

@ -25,8 +25,7 @@
* @test * @test
* @bug 6894807 * @bug 6894807
* @summary No ClassCastException for HashAttributeSet constructors if run with -Xcomp * @summary No ClassCastException for HashAttributeSet constructors if run with -Xcomp
* @compile IsInstanceTest.java * @run main IsInstanceTest
* @run shell Test6894807.sh
*/ */
public class IsInstanceTest { public class IsInstanceTest {
@ -35,13 +34,7 @@ public class IsInstanceTest {
BaseInterface baseInterfaceImpl = new BaseInterfaceImpl(); BaseInterface baseInterfaceImpl = new BaseInterfaceImpl();
for (int i = 0; i < 100000; i++) { for (int i = 0; i < 100000; i++) {
if (isInstanceOf(baseInterfaceImpl, ExtendedInterface.class)) { if (isInstanceOf(baseInterfaceImpl, ExtendedInterface.class)) {
System.out.println("Failed at index:" + i); throw new AssertionError("Failed at index:" + i);
System.out.println("Arch: "+System.getProperty("os.arch", "")+
" OS: "+System.getProperty("os.name", "")+
" OSV: "+System.getProperty("os.version", "")+
" Cores: "+Runtime.getRuntime().availableProcessors()+
" JVM: "+System.getProperty("java.version", "")+" "+System.getProperty("sun.arch.data.model", ""));
break;
} }
} }
System.out.println("Done!"); System.out.println("Done!");

View File

@ -1,48 +0,0 @@
#!/bin/sh
#
# Copyright (c) 2012, 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.
#
if [ "${TESTSRC}" = "" ]
then
TESTSRC=${PWD}
echo "TESTSRC not set. Using "${TESTSRC}" as default"
fi
echo "TESTSRC=${TESTSRC}"
## Adding common setup Variables for running shell tests.
. ${TESTSRC}/../../../test_env.sh
${TESTJAVA}${FS}bin${FS}java ${TESTOPTS} IsInstanceTest > test.out 2>&1
cat test.out
grep "Failed at index" test.out
if [ $? = 0 ]
then
echo "Test Failed"
exit 1
else
echo "Test Passed"
exit 0
fi

View File

@ -2,8 +2,10 @@
* @test * @test
* @bug 7070134 * @bug 7070134
* @summary Hotspot crashes with sigsegv from PorterStemmer * @summary Hotspot crashes with sigsegv from PorterStemmer
* * @modules java.base/jdk.internal.misc
* @run shell Test7070134.sh * @library /testlibrary
* @run driver jdk.test.lib.FileInstaller words words
* @run main/othervm -Xbatch Stemmer words
*/ */
/* /*
@ -61,7 +63,7 @@ import java.io.*;
* by calling one of the various stem(something) methods. * by calling one of the various stem(something) methods.
*/ */
class Stemmer public class Stemmer
{ private char[] b; { private char[] b;
private int i, /* offset into b */ private int i, /* offset into b */
i_end, /* offset to end of stemmed word */ i_end, /* offset to end of stemmed word */

View File

@ -1,45 +0,0 @@
#!/bin/sh
#
# Copyright (c) 2011, 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.
#
#
## some tests require path to find test source dir
if [ "${TESTSRC}" = "" ]
then
TESTSRC=${PWD}
echo "TESTSRC not set. Using "${TESTSRC}" as default"
fi
echo "TESTSRC=${TESTSRC}"
## Adding common setup Variables for running shell tests.
. ${TESTSRC}/../../../test_env.sh
set -x
cp ${TESTSRC}/Stemmer.java .
cp ${TESTSRC}/words .
${COMPILEJAVA}/bin/javac ${TESTJAVACOPTS} -d . Stemmer.java
${TESTJAVA}/bin/java ${TESTOPTS} -Xbatch Stemmer words > test.out 2>&1
exit $?

View File

@ -38,6 +38,7 @@ import sun.hotspot.code.BlobType;
/* /*
* @test PoolsIndependenceTest * @test PoolsIndependenceTest
* @modules java.base/jdk.internal.misc * @modules java.base/jdk.internal.misc
* java.management
* @library /testlibrary /test/lib * @library /testlibrary /test/lib
* @build PoolsIndependenceTest * @build PoolsIndependenceTest
* @run main ClassFileInstaller sun.hotspot.WhiteBox * @run main ClassFileInstaller sun.hotspot.WhiteBox

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2016, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -19,29 +19,19 @@
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 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 * or visit www.oracle.com if you need additional information or have any
* questions. * questions.
*
*/ */
/* @test /*
* @bug 8022853 * @test TestSSE4Disabled
* @library /testlibrary * @bug 8158214
* @modules java.base/jdk.internal.misc * @requires (vm.simpleArch == "x64")
* @build jdk.test.lib.* * @summary Test correct execution without SSE 4.
* @run main GetKlassPointerGetJavaMirror * @run main/othervm -Xcomp -XX:UseSSE=3 TestSSE4Disabled
*/ */
public class TestSSE4Disabled {
import static jdk.test.lib.Asserts.*; public static void main(String args[]) {
System.out.println("Passed");
import jdk.test.lib.*;
import jdk.internal.misc.Unsafe;
public class GetKlassPointerGetJavaMirror {
public static void main(String args[]) throws Exception {
Unsafe unsafe = Utils.getUnsafe();
Object o = new GetKlassPointerGetJavaMirror();
final long metaspaceKlass = unsafe.getKlassPointer(o);
Class<?> c = unsafe.getJavaMirror(metaspaceKlass);
assertEquals(o.getClass(), c);
} }
} }

Some files were not shown because too many files have changed in this diff Show More