8213436: Obsolete UseMembar
Reviewed-by: kvn, dholmes, mdoerr, adinn
This commit is contained in:
parent
38a1e5ffbc
commit
f69921f2fc
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2000, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2000, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright (c) 2015, Red Hat Inc. All rights reserved.
|
* Copyright (c) 2015, Red Hat Inc. 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.
|
||||||
*
|
*
|
||||||
@ -65,8 +65,6 @@ define_pd_global(intx, StackReservedPages, DEFAULT_STACK_RESERVED_PAGES);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
define_pd_global(bool, PreserveFramePointer, false);
|
define_pd_global(bool, PreserveFramePointer, false);
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
|
@ -288,10 +288,6 @@ address MacroAssembler::target_addr_for_insn(address insn_addr, unsigned insn) {
|
|||||||
return address(((uint64_t)insn_addr + (offset << 2)));
|
return address(((uint64_t)insn_addr + (offset << 2)));
|
||||||
}
|
}
|
||||||
|
|
||||||
void MacroAssembler::serialize_memory(Register thread, Register tmp) {
|
|
||||||
dsb(Assembler::SY);
|
|
||||||
}
|
|
||||||
|
|
||||||
void MacroAssembler::safepoint_poll(Label& slow_path) {
|
void MacroAssembler::safepoint_poll(Label& slow_path) {
|
||||||
if (SafepointMechanism::uses_thread_local_poll()) {
|
if (SafepointMechanism::uses_thread_local_poll()) {
|
||||||
ldr(rscratch1, Address(rthread, Thread::polling_page_offset()));
|
ldr(rscratch1, Address(rthread, Thread::polling_page_offset()));
|
||||||
|
@ -975,9 +975,6 @@ public:
|
|||||||
Register tmp,
|
Register tmp,
|
||||||
int offset);
|
int offset);
|
||||||
|
|
||||||
// Support for serializing memory accesses between threads
|
|
||||||
void serialize_memory(Register thread, Register tmp);
|
|
||||||
|
|
||||||
// Arithmetics
|
// Arithmetics
|
||||||
|
|
||||||
void addptr(const Address &dst, int32_t src);
|
void addptr(const Address &dst, int32_t src);
|
||||||
|
@ -1950,21 +1950,10 @@ nmethod* SharedRuntime::generate_native_wrapper(MacroAssembler* masm,
|
|||||||
// didn't see any synchronization is progress, and escapes.
|
// didn't see any synchronization is progress, and escapes.
|
||||||
__ mov(rscratch1, _thread_in_native_trans);
|
__ mov(rscratch1, _thread_in_native_trans);
|
||||||
|
|
||||||
if (UseMembar) {
|
__ strw(rscratch1, Address(rthread, JavaThread::thread_state_offset()));
|
||||||
__ strw(rscratch1, Address(rthread, JavaThread::thread_state_offset()));
|
|
||||||
|
|
||||||
// Force this write out before the read below
|
// Force this write out before the read below
|
||||||
__ dmb(Assembler::ISH);
|
__ dmb(Assembler::ISH);
|
||||||
} else {
|
|
||||||
__ lea(rscratch2, Address(rthread, JavaThread::thread_state_offset()));
|
|
||||||
__ stlrw(rscratch1, rscratch2);
|
|
||||||
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(rthread, r2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// check for safepoint operation in progress and/or pending suspend requests
|
// check for safepoint operation in progress and/or pending suspend requests
|
||||||
Label safepoint_in_progress, safepoint_in_progress_done;
|
Label safepoint_in_progress, safepoint_in_progress_done;
|
||||||
|
@ -1394,16 +1394,8 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
|
|||||||
__ lea(rscratch2, Address(rthread, JavaThread::thread_state_offset()));
|
__ lea(rscratch2, Address(rthread, JavaThread::thread_state_offset()));
|
||||||
__ stlrw(rscratch1, rscratch2);
|
__ stlrw(rscratch1, rscratch2);
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ dmb(Assembler::ISH);
|
||||||
__ dmb(Assembler::ISH);
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(rthread, rscratch2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// check for safepoint operation in progress and/or pending suspend requests
|
// check for safepoint operation in progress and/or pending suspend requests
|
||||||
{
|
{
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2008, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2008, 2018, 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
|
||||||
@ -65,8 +65,6 @@ define_pd_global(intx, InlineSmallCode, 1500);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
define_pd_global(bool, PreserveFramePointer, false);
|
define_pd_global(bool, PreserveFramePointer, false);
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
|
@ -69,8 +69,6 @@ define_pd_global(intx, InlineSmallCode, 1500);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
define_pd_global(bool, PreserveFramePointer, false);
|
define_pd_global(bool, PreserveFramePointer, false);
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
|
@ -1302,35 +1302,6 @@ bool MacroAssembler::is_load_from_polling_page(int instruction, void* ucontext,
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MacroAssembler::is_memory_serialization(int instruction, JavaThread* thread, void* ucontext) {
|
|
||||||
#ifdef LINUX
|
|
||||||
ucontext_t* uc = (ucontext_t*) ucontext;
|
|
||||||
|
|
||||||
if (is_stwx(instruction) || is_stwux(instruction)) {
|
|
||||||
int ra = inv_ra_field(instruction);
|
|
||||||
int rb = inv_rb_field(instruction);
|
|
||||||
|
|
||||||
// look up content of ra and rb in ucontext
|
|
||||||
address ra_val=(address)uc->uc_mcontext.regs->gpr[ra];
|
|
||||||
long rb_val=(long)uc->uc_mcontext.regs->gpr[rb];
|
|
||||||
return os::is_memory_serialize_page(thread, ra_val+rb_val);
|
|
||||||
} else if (is_stw(instruction) || is_stwu(instruction)) {
|
|
||||||
int ra = inv_ra_field(instruction);
|
|
||||||
int d1 = inv_d1_field(instruction);
|
|
||||||
|
|
||||||
// look up content of ra in ucontext
|
|
||||||
address ra_val=(address)uc->uc_mcontext.regs->gpr[ra];
|
|
||||||
return os::is_memory_serialize_page(thread, ra_val+d1);
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
// workaround not needed on !LINUX :-)
|
|
||||||
ShouldNotCallThis();
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void MacroAssembler::bang_stack_with_offset(int offset) {
|
void MacroAssembler::bang_stack_with_offset(int offset) {
|
||||||
// When increasing the stack, the old stack pointer will be written
|
// When increasing the stack, the old stack pointer will be written
|
||||||
// to the new top of stack according to the PPC64 abi.
|
// to the new top of stack according to the PPC64 abi.
|
||||||
@ -3046,27 +3017,6 @@ void MacroAssembler::compiler_fast_unlock_object(ConditionRegister flag, Registe
|
|||||||
// flag == NE indicates failure
|
// flag == NE indicates failure
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
void MacroAssembler::serialize_memory(Register thread, Register tmp1, Register tmp2) {
|
|
||||||
srdi(tmp2, thread, os::get_serialize_page_shift_count());
|
|
||||||
|
|
||||||
int mask = os::vm_page_size() - sizeof(int);
|
|
||||||
if (Assembler::is_simm(mask, 16)) {
|
|
||||||
andi(tmp2, tmp2, mask);
|
|
||||||
} else {
|
|
||||||
lis(tmp1, (int)((signed short) (mask >> 16)));
|
|
||||||
ori(tmp1, tmp1, mask & 0x0000ffff);
|
|
||||||
andr(tmp2, tmp2, tmp1);
|
|
||||||
}
|
|
||||||
|
|
||||||
load_const(tmp1, (long) os::get_memory_serialize_page());
|
|
||||||
release();
|
|
||||||
stwx(R0, tmp1, tmp2);
|
|
||||||
}
|
|
||||||
|
|
||||||
void MacroAssembler::safepoint_poll(Label& slow_path, Register temp_reg) {
|
void MacroAssembler::safepoint_poll(Label& slow_path, Register temp_reg) {
|
||||||
if (SafepointMechanism::uses_thread_local_poll()) {
|
if (SafepointMechanism::uses_thread_local_poll()) {
|
||||||
ld(temp_reg, in_bytes(Thread::polling_page_offset()), R16_thread);
|
ld(temp_reg, in_bytes(Thread::polling_page_offset()), R16_thread);
|
||||||
|
@ -395,11 +395,6 @@ class MacroAssembler: public Assembler {
|
|||||||
static bool is_load_from_polling_page(int instruction, void* ucontext/*may be NULL*/,
|
static bool is_load_from_polling_page(int instruction, void* ucontext/*may be NULL*/,
|
||||||
address* polling_address_ptr = NULL);
|
address* polling_address_ptr = NULL);
|
||||||
|
|
||||||
// Check whether instruction is a write access to the memory
|
|
||||||
// serialization page realized by one of the instructions stw, stwu,
|
|
||||||
// stwx, or stwux.
|
|
||||||
static bool is_memory_serialization(int instruction, JavaThread* thread, void* ucontext);
|
|
||||||
|
|
||||||
// Support for NULL-checks
|
// Support for NULL-checks
|
||||||
//
|
//
|
||||||
// Generates code that causes a NULL OS exception if the content of reg is NULL.
|
// Generates code that causes a NULL OS exception if the content of reg is NULL.
|
||||||
@ -645,9 +640,6 @@ class MacroAssembler: public Assembler {
|
|||||||
Register tmp1, Register tmp2, Register tmp3,
|
Register tmp1, Register tmp2, Register tmp3,
|
||||||
bool try_bias = UseBiasedLocking, bool use_rtm = false);
|
bool try_bias = UseBiasedLocking, bool use_rtm = false);
|
||||||
|
|
||||||
// Support for serializing memory accesses between threads
|
|
||||||
void serialize_memory(Register thread, Register tmp1, Register tmp2);
|
|
||||||
|
|
||||||
// Check if safepoint requested and if so branch
|
// Check if safepoint requested and if so branch
|
||||||
void safepoint_poll(Label& slow_path, Register temp_reg);
|
void safepoint_poll(Label& slow_path, Register temp_reg);
|
||||||
|
|
||||||
|
@ -100,12 +100,6 @@ class NativeInstruction {
|
|||||||
return MacroAssembler::is_load_from_polling_page(long_at(0), NULL);
|
return MacroAssembler::is_load_from_polling_page(long_at(0), NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool is_memory_serialization(JavaThread *thread, void *ucontext) {
|
|
||||||
// Is the current instruction a write access of thread to the
|
|
||||||
// memory serialization page?
|
|
||||||
return MacroAssembler::is_memory_serialization(long_at(0), thread, ucontext);
|
|
||||||
}
|
|
||||||
|
|
||||||
address get_stack_bang_address(void *ucontext) {
|
address get_stack_bang_address(void *ucontext) {
|
||||||
// If long_at(0) is not a stack bang, return 0. Otherwise, return
|
// If long_at(0) is not a stack bang, return 0. Otherwise, return
|
||||||
// banged address.
|
// banged address.
|
||||||
|
@ -2430,16 +2430,8 @@ nmethod *SharedRuntime::generate_native_wrapper(MacroAssembler *masm,
|
|||||||
{
|
{
|
||||||
Label no_block, sync;
|
Label no_block, sync;
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below.
|
||||||
// Force this write out before the read below.
|
__ fence();
|
||||||
__ fence();
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(R16_thread, r_temp_4, r_temp_5);
|
|
||||||
}
|
|
||||||
|
|
||||||
Register sync_state_addr = r_temp_4;
|
Register sync_state_addr = r_temp_4;
|
||||||
Register sync_state = r_temp_5;
|
Register sync_state = r_temp_5;
|
||||||
|
@ -1486,16 +1486,7 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
|
|||||||
__ li(R0/*thread_state*/, _thread_in_native_trans);
|
__ li(R0/*thread_state*/, _thread_in_native_trans);
|
||||||
__ release();
|
__ release();
|
||||||
__ stw(R0/*thread_state*/, thread_(thread_state));
|
__ stw(R0/*thread_state*/, thread_(thread_state));
|
||||||
if (UseMembar) {
|
__ fence();
|
||||||
__ fence();
|
|
||||||
}
|
|
||||||
// Write serialization page so that the VM thread can do a pseudo remote
|
|
||||||
// membar. We use the current thread pointer to calculate a thread
|
|
||||||
// specific offset to write to within the page. This minimizes bus
|
|
||||||
// traffic due to cache line collision.
|
|
||||||
else {
|
|
||||||
__ serialize_memory(R16_thread, R11_scratch1, R12_scratch2);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Now before we return to java we must look for a current safepoint
|
// Now before we return to java we must look for a current safepoint
|
||||||
// (a new safepoint can not start since we entered native_trans).
|
// (a new safepoint can not start since we entered native_trans).
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2016, 2017 Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright (c) 2016, 2018 SAP SE. All rights reserved.
|
* Copyright (c) 2016, 2018 SAP SE. 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.
|
||||||
*
|
*
|
||||||
@ -71,8 +71,6 @@ define_pd_global(intx, StackReservedPages, DEFAULT_STACK_RESERVED_PAGE
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
define_pd_global(bool, PreserveFramePointer, false);
|
define_pd_global(bool, PreserveFramePointer, false);
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
|
@ -2685,33 +2685,6 @@ uint MacroAssembler::get_poll_register(address instr_loc) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MacroAssembler::is_memory_serialization(int instruction, JavaThread* thread, void* ucontext) {
|
|
||||||
ShouldNotCallThis();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
void MacroAssembler::serialize_memory(Register thread, Register tmp1, Register tmp2) {
|
|
||||||
assert_different_registers(tmp1, tmp2);
|
|
||||||
z_sllg(tmp2, thread, os::get_serialize_page_shift_count());
|
|
||||||
load_const_optimized(tmp1, (long) os::get_memory_serialize_page());
|
|
||||||
|
|
||||||
int mask = os::get_serialize_page_mask();
|
|
||||||
if (Immediate::is_uimm16(mask)) {
|
|
||||||
z_nill(tmp2, mask);
|
|
||||||
z_llghr(tmp2, tmp2);
|
|
||||||
} else {
|
|
||||||
z_nilf(tmp2, mask);
|
|
||||||
z_llgfr(tmp2, tmp2);
|
|
||||||
}
|
|
||||||
|
|
||||||
z_release();
|
|
||||||
z_st(Z_R0, 0, tmp2, tmp1);
|
|
||||||
}
|
|
||||||
|
|
||||||
void MacroAssembler::safepoint_poll(Label& slow_path, Register temp_reg) {
|
void MacroAssembler::safepoint_poll(Label& slow_path, Register temp_reg) {
|
||||||
if (SafepointMechanism::uses_thread_local_poll()) {
|
if (SafepointMechanism::uses_thread_local_poll()) {
|
||||||
const Address poll_byte_addr(Z_thread, in_bytes(Thread::polling_page_offset()) + 7 /* Big Endian */);
|
const Address poll_byte_addr(Z_thread, in_bytes(Thread::polling_page_offset()) + 7 /* Big Endian */);
|
||||||
|
@ -635,13 +635,6 @@ class MacroAssembler: public Assembler {
|
|||||||
// Extract poll register from instruction.
|
// Extract poll register from instruction.
|
||||||
static uint get_poll_register(address instr_loc);
|
static uint get_poll_register(address instr_loc);
|
||||||
|
|
||||||
// Check if instruction is a write access to the memory serialization page
|
|
||||||
// realized by one of the instructions stw, stwu, stwx, or stwux.
|
|
||||||
static bool is_memory_serialization(int instruction, JavaThread* thread, void* ucontext);
|
|
||||||
|
|
||||||
// Support for serializing memory accesses between threads.
|
|
||||||
void serialize_memory(Register thread, Register tmp1, Register tmp2);
|
|
||||||
|
|
||||||
// Check if safepoint requested and if so branch
|
// Check if safepoint requested and if so branch
|
||||||
void safepoint_poll(Label& slow_path, Register temp_reg);
|
void safepoint_poll(Label& slow_path, Register temp_reg);
|
||||||
|
|
||||||
|
@ -104,12 +104,6 @@ class NativeInstruction {
|
|||||||
return MacroAssembler::get_poll_register(addr_at(0));
|
return MacroAssembler::get_poll_register(addr_at(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool is_memory_serialization(JavaThread *thread, void *ucontext) {
|
|
||||||
// Is the current instruction a write access of thread to the
|
|
||||||
// memory serialization page?
|
|
||||||
return MacroAssembler::is_memory_serialization(long_at(0), thread, ucontext);
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
// The output of __ breakpoint_trap().
|
// The output of __ breakpoint_trap().
|
||||||
|
@ -2161,16 +2161,8 @@ nmethod *SharedRuntime::generate_native_wrapper(MacroAssembler *masm,
|
|||||||
|
|
||||||
save_native_result(masm, ret_type, workspace_slot_offset); // Make Z_R2 available as work reg.
|
save_native_result(masm, ret_type, workspace_slot_offset); // Make Z_R2 available as work reg.
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below.
|
||||||
// Force this write out before the read below.
|
__ z_fence();
|
||||||
__ z_fence();
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(Z_thread, Z_R1, Z_R2);
|
|
||||||
}
|
|
||||||
|
|
||||||
__ safepoint_poll(sync, Z_R1);
|
__ safepoint_poll(sync, Z_R1);
|
||||||
|
|
||||||
|
@ -1598,15 +1598,8 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
|
|||||||
// synchronization is progress, and escapes.
|
// synchronization is progress, and escapes.
|
||||||
|
|
||||||
__ set_thread_state(_thread_in_native_trans);
|
__ set_thread_state(_thread_in_native_trans);
|
||||||
if (UseMembar) {
|
__ z_fence();
|
||||||
__ z_fence();
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote
|
|
||||||
// membar. We use the current thread pointer to calculate a thread
|
|
||||||
// specific offset to write to within the page. This minimizes bus
|
|
||||||
// traffic due to cache line collision.
|
|
||||||
__ serialize_memory(Z_thread, Z_R1, Z_R0);
|
|
||||||
}
|
|
||||||
// Now before we return to java we must look for a current safepoint
|
// Now before we return to java we must look for a current safepoint
|
||||||
// (a new safepoint can not start since we entered native_trans).
|
// (a new safepoint can not start since we entered native_trans).
|
||||||
// We must check here because a current safepoint could be modifying
|
// We must check here because a current safepoint could be modifying
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2000, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2000, 2018, 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
|
||||||
@ -74,8 +74,6 @@ define_pd_global(intx, StackReservedPages, DEFAULT_STACK_RESERVED_PAGES);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
define_pd_global(bool, PreserveFramePointer, false);
|
define_pd_global(bool, PreserveFramePointer, false);
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
|
@ -236,24 +236,6 @@ void MacroAssembler::breakpoint_trap() {
|
|||||||
trap(ST_RESERVED_FOR_USER_0);
|
trap(ST_RESERVED_FOR_USER_0);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
void MacroAssembler::serialize_memory(Register thread, Register tmp1, Register tmp2) {
|
|
||||||
srl(thread, os::get_serialize_page_shift_count(), tmp2);
|
|
||||||
if (Assembler::is_simm13(os::vm_page_size())) {
|
|
||||||
and3(tmp2, (os::vm_page_size() - sizeof(int)), tmp2);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
set((os::vm_page_size() - sizeof(int)), tmp1);
|
|
||||||
and3(tmp2, tmp1, tmp2);
|
|
||||||
}
|
|
||||||
set(os::get_memory_serialize_page(), tmp1);
|
|
||||||
st(G0, tmp1, tmp2);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void MacroAssembler::safepoint_poll(Label& slow_path, bool a, Register thread_reg, Register temp_reg) {
|
void MacroAssembler::safepoint_poll(Label& slow_path, bool a, Register thread_reg, Register temp_reg) {
|
||||||
if (SafepointMechanism::uses_thread_local_poll()) {
|
if (SafepointMechanism::uses_thread_local_poll()) {
|
||||||
ldx(Address(thread_reg, Thread::polling_page_offset()), temp_reg, 0);
|
ldx(Address(thread_reg, Thread::polling_page_offset()), temp_reg, 0);
|
||||||
|
@ -951,9 +951,6 @@ public:
|
|||||||
void breakpoint_trap();
|
void breakpoint_trap();
|
||||||
void breakpoint_trap(Condition c, CC cc);
|
void breakpoint_trap(Condition c, CC cc);
|
||||||
|
|
||||||
// Support for serializing memory accesses between threads
|
|
||||||
void serialize_memory(Register thread, Register tmp1, Register tmp2);
|
|
||||||
|
|
||||||
void safepoint_poll(Label& slow_path, bool a, Register thread_reg, Register temp_reg);
|
void safepoint_poll(Label& slow_path, bool a, Register thread_reg, Register temp_reg);
|
||||||
|
|
||||||
// Stack frame creation/removal
|
// Stack frame creation/removal
|
||||||
|
@ -2372,16 +2372,8 @@ nmethod* SharedRuntime::generate_native_wrapper(MacroAssembler* masm,
|
|||||||
__ set(_thread_in_native_trans, G3_scratch);
|
__ set(_thread_in_native_trans, G3_scratch);
|
||||||
__ st(G3_scratch, G2_thread, JavaThread::thread_state_offset());
|
__ st(G3_scratch, G2_thread, JavaThread::thread_state_offset());
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ membar(Assembler::StoreLoad);
|
||||||
__ membar(Assembler::StoreLoad);
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(G2_thread, G1_scratch, G3_scratch);
|
|
||||||
}
|
|
||||||
|
|
||||||
Label L;
|
Label L;
|
||||||
Address suspend_state(G2_thread, JavaThread::suspend_flags_offset());
|
Address suspend_state(G2_thread, JavaThread::suspend_flags_offset());
|
||||||
|
@ -1374,16 +1374,8 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
|
|||||||
__ set(_thread_in_native_trans, G3_scratch);
|
__ set(_thread_in_native_trans, G3_scratch);
|
||||||
__ st(G3_scratch, thread_state);
|
__ st(G3_scratch, thread_state);
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ membar(Assembler::StoreLoad);
|
||||||
__ membar(Assembler::StoreLoad);
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(G2_thread, G1_scratch, G3_scratch);
|
|
||||||
}
|
|
||||||
|
|
||||||
Label L;
|
Label L;
|
||||||
__ safepoint_poll(L, false, G2_thread, G3_scratch);
|
__ safepoint_poll(L, false, G2_thread, G3_scratch);
|
||||||
|
@ -84,8 +84,6 @@ define_pd_global(intx, StackReservedPages, DEFAULT_STACK_RESERVED_PAGES);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
define_pd_global(size_t, CMSYoungGenPerWorker, 64*M); // default max size of CMS young gen, per GC worker thread
|
define_pd_global(size_t, CMSYoungGenPerWorker, 64*M); // default max size of CMS young gen, per GC worker thread
|
||||||
|
|
||||||
|
@ -3517,22 +3517,6 @@ void MacroAssembler::save_rax(Register tmp) {
|
|||||||
else if (tmp != rax) mov(tmp, rax);
|
else if (tmp != rax) mov(tmp, rax);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
void MacroAssembler::serialize_memory(Register thread, Register tmp) {
|
|
||||||
movl(tmp, thread);
|
|
||||||
shrl(tmp, os::get_serialize_page_shift_count());
|
|
||||||
andl(tmp, (os::vm_page_size() - sizeof(int)));
|
|
||||||
|
|
||||||
Address index(noreg, tmp, Address::times_1);
|
|
||||||
ExternalAddress page(os::get_memory_serialize_page());
|
|
||||||
|
|
||||||
// Size of store must match masking code above
|
|
||||||
movl(as_Address(ArrayAddress(page, index)), tmp);
|
|
||||||
}
|
|
||||||
|
|
||||||
void MacroAssembler::safepoint_poll(Label& slow_path, Register thread_reg, Register temp_reg) {
|
void MacroAssembler::safepoint_poll(Label& slow_path, Register thread_reg, Register temp_reg) {
|
||||||
if (SafepointMechanism::uses_thread_local_poll()) {
|
if (SafepointMechanism::uses_thread_local_poll()) {
|
||||||
#ifdef _LP64
|
#ifdef _LP64
|
||||||
|
@ -644,9 +644,6 @@ class MacroAssembler: public Assembler {
|
|||||||
Register tmp,
|
Register tmp,
|
||||||
int offset);
|
int offset);
|
||||||
|
|
||||||
// Support for serializing memory accesses between threads
|
|
||||||
void serialize_memory(Register thread, Register tmp);
|
|
||||||
|
|
||||||
// If thread_reg is != noreg the code assumes the register passed contains
|
// If thread_reg is != noreg the code assumes the register passed contains
|
||||||
// the thread (required on 64 bit).
|
// the thread (required on 64 bit).
|
||||||
void safepoint_poll(Label& slow_path, Register thread_reg, Register temp_reg);
|
void safepoint_poll(Label& slow_path, Register thread_reg, Register temp_reg);
|
||||||
|
@ -2088,18 +2088,10 @@ nmethod* SharedRuntime::generate_native_wrapper(MacroAssembler* masm,
|
|||||||
// didn't see any synchronization is progress, and escapes.
|
// didn't see any synchronization is progress, and escapes.
|
||||||
__ movl(Address(thread, JavaThread::thread_state_offset()), _thread_in_native_trans);
|
__ movl(Address(thread, JavaThread::thread_state_offset()), _thread_in_native_trans);
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ membar(Assembler::Membar_mask_bits(
|
||||||
__ membar(Assembler::Membar_mask_bits(
|
Assembler::LoadLoad | Assembler::LoadStore |
|
||||||
Assembler::LoadLoad | Assembler::LoadStore |
|
Assembler::StoreLoad | Assembler::StoreStore));
|
||||||
Assembler::StoreLoad | Assembler::StoreStore));
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(thread, rcx);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (AlwaysRestoreFPU) {
|
if (AlwaysRestoreFPU) {
|
||||||
// Make sure the control word is correct.
|
// Make sure the control word is correct.
|
||||||
|
@ -2560,18 +2560,10 @@ nmethod* SharedRuntime::generate_native_wrapper(MacroAssembler* masm,
|
|||||||
// didn't see any synchronization is progress, and escapes.
|
// didn't see any synchronization is progress, and escapes.
|
||||||
__ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_native_trans);
|
__ movl(Address(r15_thread, JavaThread::thread_state_offset()), _thread_in_native_trans);
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ membar(Assembler::Membar_mask_bits(
|
||||||
__ membar(Assembler::Membar_mask_bits(
|
Assembler::LoadLoad | Assembler::LoadStore |
|
||||||
Assembler::LoadLoad | Assembler::LoadStore |
|
Assembler::StoreLoad | Assembler::StoreStore));
|
||||||
Assembler::StoreLoad | Assembler::StoreStore));
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(r15_thread, rcx);
|
|
||||||
}
|
|
||||||
|
|
||||||
Label after_transition;
|
Label after_transition;
|
||||||
|
|
||||||
|
@ -1090,18 +1090,10 @@ address TemplateInterpreterGenerator::generate_native_entry(bool synchronized) {
|
|||||||
__ movl(Address(thread, JavaThread::thread_state_offset()),
|
__ movl(Address(thread, JavaThread::thread_state_offset()),
|
||||||
_thread_in_native_trans);
|
_thread_in_native_trans);
|
||||||
|
|
||||||
if (UseMembar) {
|
// Force this write out before the read below
|
||||||
// Force this write out before the read below
|
__ membar(Assembler::Membar_mask_bits(
|
||||||
__ membar(Assembler::Membar_mask_bits(
|
Assembler::LoadLoad | Assembler::LoadStore |
|
||||||
Assembler::LoadLoad | Assembler::LoadStore |
|
Assembler::StoreLoad | Assembler::StoreStore));
|
||||||
Assembler::StoreLoad | Assembler::StoreStore));
|
|
||||||
} else {
|
|
||||||
// Write serialization page so VM thread can do a pseudo remote membar.
|
|
||||||
// We use the current thread pointer to calculate a thread specific
|
|
||||||
// offset to write to within the page. This minimizes bus traffic
|
|
||||||
// due to cache line collision.
|
|
||||||
__ serialize_memory(thread, rcx);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef _LP64
|
#ifndef _LP64
|
||||||
if (AlwaysRestoreFPU) {
|
if (AlwaysRestoreFPU) {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2000, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2000, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright 2007, 2008, 2009, 2010, 2011 Red Hat, Inc.
|
* Copyright 2007, 2008, 2009, 2010, 2011 Red Hat, Inc.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
@ -69,8 +69,6 @@ define_pd_global(intx, StackReservedPages, DEFAULT_STACK_RESERVED_PAGES);
|
|||||||
define_pd_global(bool, RewriteBytecodes, true);
|
define_pd_global(bool, RewriteBytecodes, true);
|
||||||
define_pd_global(bool, RewriteFrequentPairs, true);
|
define_pd_global(bool, RewriteFrequentPairs, true);
|
||||||
|
|
||||||
define_pd_global(bool, UseMembar, true);
|
|
||||||
|
|
||||||
// GC Ergo Flags
|
// GC Ergo Flags
|
||||||
define_pd_global(size_t, CMSYoungGenPerWorker, 16*M); // default max size of CMS young gen, per GC worker thread
|
define_pd_global(size_t, CMSYoungGenPerWorker, 16*M); // default max size of CMS young gen, per GC worker thread
|
||||||
|
|
||||||
|
@ -130,11 +130,6 @@ public:
|
|||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
// On POSIX platforms the signal handler is global so we just do the write.
|
|
||||||
static void write_memory_serialize_page_with_handler(JavaThread* thread) {
|
|
||||||
write_memory_serialize_page(thread);
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Crash protection for the watcher thread. Wrap the callback
|
* Crash protection for the watcher thread. Wrap the callback
|
||||||
* with a sigsetjmp and in case of a SIGSEGV/SIGBUS we siglongjmp
|
* with a sigsetjmp and in case of a SIGSEGV/SIGBUS we siglongjmp
|
||||||
|
@ -2414,23 +2414,6 @@ LONG WINAPI topLevelExceptionFilter(struct _EXCEPTION_POINTERS* exceptionInfo) {
|
|||||||
}
|
}
|
||||||
#endif // _WIN64
|
#endif // _WIN64
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so just return.
|
|
||||||
if (exception_code == EXCEPTION_ACCESS_VIOLATION) {
|
|
||||||
if (t != NULL && t->is_Java_thread()) {
|
|
||||||
JavaThread* thread = (JavaThread*) t;
|
|
||||||
PEXCEPTION_RECORD exceptionRecord = exceptionInfo->ExceptionRecord;
|
|
||||||
address addr = (address) exceptionRecord->ExceptionInformation[1];
|
|
||||||
if (os::is_memory_serialize_page(thread, addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return EXCEPTION_CONTINUE_EXECUTION;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((exception_code == EXCEPTION_ACCESS_VIOLATION) &&
|
if ((exception_code == EXCEPTION_ACCESS_VIOLATION) &&
|
||||||
VM_Version::is_cpuinfo_segv_addr(pc)) {
|
VM_Version::is_cpuinfo_segv_addr(pc)) {
|
||||||
// Verify that OS save/restore AVX registers.
|
// Verify that OS save/restore AVX registers.
|
||||||
@ -5330,22 +5313,6 @@ bool os::find(address addr, outputStream* st) {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
LONG WINAPI os::win32::serialize_fault_filter(struct _EXCEPTION_POINTERS* e) {
|
|
||||||
DWORD exception_code = e->ExceptionRecord->ExceptionCode;
|
|
||||||
|
|
||||||
if (exception_code == EXCEPTION_ACCESS_VIOLATION) {
|
|
||||||
JavaThread* thread = JavaThread::current();
|
|
||||||
PEXCEPTION_RECORD exceptionRecord = e->ExceptionRecord;
|
|
||||||
address addr = (address) exceptionRecord->ExceptionInformation[1];
|
|
||||||
|
|
||||||
if (os::is_memory_serialize_page(thread, addr)) {
|
|
||||||
return EXCEPTION_CONTINUE_EXECUTION;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return EXCEPTION_CONTINUE_SEARCH;
|
|
||||||
}
|
|
||||||
|
|
||||||
static jint initSock() {
|
static jint initSock() {
|
||||||
WSADATA wsadata;
|
WSADATA wsadata;
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2018, 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
|
||||||
@ -108,9 +108,6 @@ class win32 {
|
|||||||
static address fast_jni_accessor_wrapper(BasicType);
|
static address fast_jni_accessor_wrapper(BasicType);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// filter function to ignore faults on serializations page
|
|
||||||
static LONG WINAPI serialize_fault_filter(struct _EXCEPTION_POINTERS* e);
|
|
||||||
|
|
||||||
// Fast access to current thread
|
// Fast access to current thread
|
||||||
protected:
|
protected:
|
||||||
static int _thread_ptr_offset;
|
static int _thread_ptr_offset;
|
||||||
@ -123,21 +120,6 @@ public:
|
|||||||
static inline int get_thread_ptr_offset() { return _thread_ptr_offset; }
|
static inline int get_thread_ptr_offset() { return _thread_ptr_offset; }
|
||||||
};
|
};
|
||||||
|
|
||||||
static void write_memory_serialize_page_with_handler(JavaThread* thread) {
|
|
||||||
// Due to chained nature of SEH handlers we have to be sure
|
|
||||||
// that our handler is always last handler before an attempt to write
|
|
||||||
// into serialization page - it can fault if we access this page
|
|
||||||
// right in the middle of protect/unprotect sequence by remote
|
|
||||||
// membar logic.
|
|
||||||
// __try/__except are very lightweight operations (only several
|
|
||||||
// instructions not affecting control flow directly on x86)
|
|
||||||
// so we can use it here, on very time critical path
|
|
||||||
__try {
|
|
||||||
write_memory_serialize_page(thread);
|
|
||||||
} __except (win32::serialize_fault_filter((_EXCEPTION_POINTERS*)_exception_info()))
|
|
||||||
{}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Crash protection for the watcher thread. Wrap the callback
|
* Crash protection for the watcher thread. Wrap the callback
|
||||||
* with a __try { call() }
|
* with a __try { call() }
|
||||||
|
@ -469,18 +469,6 @@ JVM_handle_aix_signal(int sig, siginfo_t* info, void* ucVoid, int abort_if_unrec
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
os::is_memory_serialize_page(thread, addr)) {
|
|
||||||
// Synchronization problem in the pseudo memory barrier code (bug id 6546278)
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
run_stub:
|
run_stub:
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1999, 2018, 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
|
||||||
@ -674,17 +674,6 @@ JVM_handle_bsd_signal(int sig,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV || sig == SIGBUS) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef AMD64
|
#ifndef AMD64
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2003, 2016, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright 2007, 2008, 2009, 2010 Red Hat, Inc.
|
* Copyright 2007, 2008, 2009, 2010 Red Hat, Inc.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
@ -206,17 +206,6 @@ JVM_handle_bsd_signal(int sig,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the process
|
|
||||||
// of write protecting the memory serialization page. It write
|
|
||||||
// enables the page immediately after protecting it so we can
|
|
||||||
// just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV || sig == SIGBUS) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until permission is restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// signal-chaining
|
// signal-chaining
|
||||||
|
@ -475,17 +475,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stub != NULL) {
|
if (stub != NULL) {
|
||||||
|
@ -407,16 +407,6 @@ extern "C" int JVM_handle_linux_signal(int sig, siginfo_t* info,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if (sig == SIGSEGV && os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (unsafe_access && stub == NULL) {
|
if (unsafe_access && stub == NULL) {
|
||||||
|
@ -477,20 +477,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
// Si_addr may not be valid due to a bug in the linux-ppc64 kernel (see comment above).
|
|
||||||
// Use is_memory_serialization instead of si_addr.
|
|
||||||
((NativeInstruction*)pc)->is_memory_serialization(thread, ucVoid)) {
|
|
||||||
// Synchronization problem in the pseudo memory barrier code (bug id 6546278)
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stub != NULL) {
|
if (stub != NULL) {
|
||||||
|
@ -477,19 +477,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
// Info->si_addr need not be the exact address, it is only
|
|
||||||
// guaranteed to be on the same page as the address that caused
|
|
||||||
// the SIGSEGV.
|
|
||||||
if ((sig == SIGSEGV) && !UseMembar &&
|
|
||||||
(os::get_memory_serialize_page() ==
|
|
||||||
(address)((uintptr_t)info->si_addr & ~(os::vm_page_size()-1)))) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stub != NULL) {
|
if (stub != NULL) {
|
||||||
|
@ -439,10 +439,6 @@ inline static bool checkFastJNIAccess(address pc, address* stub) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline static bool checkSerializePage(JavaThread* thread, address addr) {
|
|
||||||
return os::is_memory_serialize_page(thread, addr);
|
|
||||||
}
|
|
||||||
|
|
||||||
inline static bool checkZombie(sigcontext* uc, address* pc, address* stub) {
|
inline static bool checkZombie(sigcontext* uc, address* pc, address* stub) {
|
||||||
if (nativeInstruction_at(*pc)->is_zombie()) {
|
if (nativeInstruction_at(*pc)->is_zombie()) {
|
||||||
// zombie method (ld [%g0],%o7 instruction)
|
// zombie method (ld [%g0],%o7 instruction)
|
||||||
@ -542,16 +538,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
pc = address(SIG_PC(uc));
|
pc = address(SIG_PC(uc));
|
||||||
npc = address(SIG_NPC(uc));
|
npc = address(SIG_NPC(uc));
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) && checkSerializePage(thread, (address)info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (checkPrefetch(uc, pc)) {
|
if (checkPrefetch(uc, pc)) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
@ -498,17 +498,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef AMD64
|
#ifndef AMD64
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2003, 2016, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2003, 2018, Oracle and/or its affiliates. All rights reserved.
|
||||||
* Copyright 2007, 2008, 2009, 2010 Red Hat, Inc.
|
* Copyright 2007, 2008, 2009, 2010 Red Hat, Inc.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
@ -220,17 +220,6 @@ JVM_handle_linux_signal(int sig,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the process
|
|
||||||
// of write protecting the memory serialization page. It write
|
|
||||||
// enables the page immediately after protecting it so we can
|
|
||||||
// just return to retry the write.
|
|
||||||
if (sig == SIGSEGV &&
|
|
||||||
os::is_memory_serialize_page(thread, (address) info->si_addr)) {
|
|
||||||
// Block current thread until permission is restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// signal-chaining
|
// signal-chaining
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1999, 2018, 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
|
||||||
@ -519,17 +519,6 @@ JVM_handle_solaris_signal(int sig, siginfo_t* info, void* ucVoid,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so just return.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address)info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stub != NULL) {
|
if (stub != NULL) {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1999, 2017, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1999, 2018, 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
|
||||||
@ -593,17 +593,6 @@ JVM_handle_solaris_signal(int sig, siginfo_t* info, void* ucVoid,
|
|||||||
stub = addr;
|
stub = addr;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if we caught the safepoint code in the
|
|
||||||
// process of write protecting the memory serialization page.
|
|
||||||
// It write enables the page immediately after protecting it
|
|
||||||
// so we can just return to retry the write.
|
|
||||||
if ((sig == SIGSEGV) &&
|
|
||||||
os::is_memory_serialize_page(thread, (address)info->si_addr)) {
|
|
||||||
// Block current thread until the memory serialize page permission restored.
|
|
||||||
os::block_on_serialize_page_trap();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Execution protection violation
|
// Execution protection violation
|
||||||
|
@ -359,9 +359,6 @@ bool JfrThreadSampleClosure::do_sample_thread(JavaThread* thread, JfrStackFrame*
|
|||||||
|
|
||||||
bool ret = false;
|
bool ret = false;
|
||||||
thread->set_trace_flag();
|
thread->set_trace_flag();
|
||||||
if (!UseMembar) {
|
|
||||||
os::serialize_thread_states();
|
|
||||||
}
|
|
||||||
if (JAVA_SAMPLE == type) {
|
if (JAVA_SAMPLE == type) {
|
||||||
if (thread_state_in_java(thread)) {
|
if (thread_state_in_java(thread)) {
|
||||||
ret = sample_thread_in_java(thread, frames, max_frames);
|
ret = sample_thread_in_java(thread, frames, max_frames);
|
||||||
|
@ -532,7 +532,7 @@ static SpecialFlag const special_jvm_flags[] = {
|
|||||||
{ "MaxRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
{ "MaxRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
||||||
{ "MinRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
{ "MinRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
||||||
{ "InitialRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
{ "InitialRAMFraction", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
||||||
{ "UseMembar", JDK_Version::jdk(10), JDK_Version::undefined(), JDK_Version::undefined() },
|
{ "UseMembar", JDK_Version::jdk(10), JDK_Version::jdk(12), JDK_Version::undefined() },
|
||||||
{ "CompilerThreadHintNoPreempt", JDK_Version::jdk(11), JDK_Version::jdk(12), JDK_Version::jdk(13) },
|
{ "CompilerThreadHintNoPreempt", JDK_Version::jdk(11), JDK_Version::jdk(12), JDK_Version::jdk(13) },
|
||||||
{ "VMThreadHintNoPreempt", JDK_Version::jdk(11), JDK_Version::jdk(12), JDK_Version::jdk(13) },
|
{ "VMThreadHintNoPreempt", JDK_Version::jdk(11), JDK_Version::jdk(12), JDK_Version::jdk(13) },
|
||||||
|
|
||||||
|
@ -247,12 +247,6 @@ define_pd_global(uint64_t,MaxRAM, 1ULL*G);
|
|||||||
range(8, 256) \
|
range(8, 256) \
|
||||||
constraint(ObjectAlignmentInBytesConstraintFunc,AtParse) \
|
constraint(ObjectAlignmentInBytesConstraintFunc,AtParse) \
|
||||||
\
|
\
|
||||||
/* UseMembar is theoretically a temp flag used for memory barrier */ \
|
|
||||||
/* removal testing. It was supposed to be removed before FCS but has */ \
|
|
||||||
/* been re-added (see 6401008) */ \
|
|
||||||
product_pd(bool, UseMembar, \
|
|
||||||
"(Unstable) Issues membars on thread state transitions") \
|
|
||||||
\
|
|
||||||
develop(bool, CleanChunkPoolAsync, true, \
|
develop(bool, CleanChunkPoolAsync, true, \
|
||||||
"Clean the chunk pool asynchronously") \
|
"Clean the chunk pool asynchronously") \
|
||||||
\
|
\
|
||||||
|
@ -126,10 +126,6 @@ class VM_HandshakeOneThread: public VM_Handshake {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!UseMembar) {
|
|
||||||
os::serialize_thread_states();
|
|
||||||
}
|
|
||||||
|
|
||||||
log_trace(handshake)("Thread signaled, begin processing by VMThtread");
|
log_trace(handshake)("Thread signaled, begin processing by VMThtread");
|
||||||
jlong start_time = os::elapsed_counter();
|
jlong start_time = os::elapsed_counter();
|
||||||
do {
|
do {
|
||||||
@ -173,10 +169,6 @@ class VM_HandshakeAllThreads: public VM_Handshake {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!UseMembar) {
|
|
||||||
os::serialize_thread_states();
|
|
||||||
}
|
|
||||||
|
|
||||||
log_debug(handshake)("Threads signaled, begin processing blocked threads by VMThtread");
|
log_debug(handshake)("Threads signaled, begin processing blocked threads by VMThtread");
|
||||||
const jlong start_time = os::elapsed_counter();
|
const jlong start_time = os::elapsed_counter();
|
||||||
int number_of_threads_completed = 0;
|
int number_of_threads_completed = 0;
|
||||||
|
@ -79,17 +79,7 @@ class InterfaceSupport: AllStatic {
|
|||||||
private:
|
private:
|
||||||
static void serialize_thread_state_internal(JavaThread* thread, bool needs_exception_handler) {
|
static void serialize_thread_state_internal(JavaThread* thread, bool needs_exception_handler) {
|
||||||
// Make sure new state is seen by VM thread
|
// Make sure new state is seen by VM thread
|
||||||
if (UseMembar) {
|
OrderAccess::fence();
|
||||||
// Force a fence between the write above and read below
|
|
||||||
OrderAccess::fence();
|
|
||||||
} else {
|
|
||||||
// store to serialize page so VM thread can do pseudo remote membar
|
|
||||||
if (needs_exception_handler) {
|
|
||||||
os::write_memory_serialize_page_with_handler(thread);
|
|
||||||
} else {
|
|
||||||
os::write_memory_serialize_page(thread);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -126,9 +116,7 @@ class ThreadStateTransition : public StackObj {
|
|||||||
// transition_and_fence must be used on any thread state transition
|
// transition_and_fence must be used on any thread state transition
|
||||||
// where there might not be a Java call stub on the stack, in
|
// where there might not be a Java call stub on the stack, in
|
||||||
// particular on Windows where the Structured Exception Handler is
|
// particular on Windows where the Structured Exception Handler is
|
||||||
// set up in the call stub. os::write_memory_serialize_page() can
|
// set up in the call stub.
|
||||||
// fault and we can't recover from it on Windows without a SEH in
|
|
||||||
// place.
|
|
||||||
static inline void transition_and_fence(JavaThread *thread, JavaThreadState from, JavaThreadState to) {
|
static inline void transition_and_fence(JavaThread *thread, JavaThreadState from, JavaThreadState to) {
|
||||||
assert(thread->thread_state() == from, "coming from wrong thread state");
|
assert(thread->thread_state() == from, "coming from wrong thread state");
|
||||||
assert((from & 1) == 0 && (to & 1) == 0, "odd numbers are transitions states");
|
assert((from & 1) == 0 && (to & 1) == 0, "odd numbers are transitions states");
|
||||||
|
@ -71,8 +71,6 @@
|
|||||||
|
|
||||||
OSThread* os::_starting_thread = NULL;
|
OSThread* os::_starting_thread = NULL;
|
||||||
address os::_polling_page = NULL;
|
address os::_polling_page = NULL;
|
||||||
volatile int32_t* os::_mem_serialize_page = NULL;
|
|
||||||
uintptr_t os::_serialize_page_mask = 0;
|
|
||||||
volatile unsigned int os::_rand_seed = 1;
|
volatile unsigned int os::_rand_seed = 1;
|
||||||
int os::_processor_count = 0;
|
int os::_processor_count = 0;
|
||||||
int os::_initial_active_processor_count = 0;
|
int os::_initial_active_processor_count = 0;
|
||||||
@ -1351,49 +1349,6 @@ char** os::split_path(const char* path, int* n) {
|
|||||||
return opath;
|
return opath;
|
||||||
}
|
}
|
||||||
|
|
||||||
void os::set_memory_serialize_page(address page) {
|
|
||||||
int count = log2_intptr(sizeof(class JavaThread)) - log2_intptr(64);
|
|
||||||
_mem_serialize_page = (volatile int32_t *)page;
|
|
||||||
// We initialize the serialization page shift count here
|
|
||||||
// We assume a cache line size of 64 bytes
|
|
||||||
assert(SerializePageShiftCount == count, "JavaThread size changed; "
|
|
||||||
"SerializePageShiftCount constant should be %d", count);
|
|
||||||
set_serialize_page_mask((uintptr_t)(vm_page_size() - sizeof(int32_t)));
|
|
||||||
}
|
|
||||||
|
|
||||||
static volatile intptr_t SerializePageLock = 0;
|
|
||||||
|
|
||||||
// This method is called from signal handler when SIGSEGV occurs while the current
|
|
||||||
// thread tries to store to the "read-only" memory serialize page during state
|
|
||||||
// transition.
|
|
||||||
void os::block_on_serialize_page_trap() {
|
|
||||||
log_debug(safepoint)("Block until the serialize page permission restored");
|
|
||||||
|
|
||||||
// When VMThread is holding the SerializePageLock during modifying the
|
|
||||||
// access permission of the memory serialize page, the following call
|
|
||||||
// will block until the permission of that page is restored to rw.
|
|
||||||
// Generally, it is unsafe to manipulate locks in signal handlers, but in
|
|
||||||
// this case, it's OK as the signal is synchronous and we know precisely when
|
|
||||||
// it can occur.
|
|
||||||
Thread::muxAcquire(&SerializePageLock, "set_memory_serialize_page");
|
|
||||||
Thread::muxRelease(&SerializePageLock);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Serialize all thread state variables
|
|
||||||
void os::serialize_thread_states() {
|
|
||||||
// On some platforms such as Solaris & Linux, the time duration of the page
|
|
||||||
// permission restoration is observed to be much longer than expected due to
|
|
||||||
// scheduler starvation problem etc. To avoid the long synchronization
|
|
||||||
// time and expensive page trap spinning, 'SerializePageLock' is used to block
|
|
||||||
// the mutator thread if such case is encountered. See bug 6546278 for details.
|
|
||||||
Thread::muxAcquire(&SerializePageLock, "serialize_thread_states");
|
|
||||||
os::protect_memory((char *)os::get_memory_serialize_page(),
|
|
||||||
os::vm_page_size(), MEM_PROT_READ);
|
|
||||||
os::protect_memory((char *)os::get_memory_serialize_page(),
|
|
||||||
os::vm_page_size(), MEM_PROT_RW);
|
|
||||||
Thread::muxRelease(&SerializePageLock);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns true if the current stack pointer is above the stack shadow
|
// Returns true if the current stack pointer is above the stack shadow
|
||||||
// pages, false otherwise.
|
// pages, false otherwise.
|
||||||
bool os::stack_shadow_pages_available(Thread *thread, const methodHandle& method, address sp) {
|
bool os::stack_shadow_pages_available(Thread *thread, const methodHandle& method, address sp) {
|
||||||
|
@ -100,8 +100,6 @@ class os: AllStatic {
|
|||||||
private:
|
private:
|
||||||
static OSThread* _starting_thread;
|
static OSThread* _starting_thread;
|
||||||
static address _polling_page;
|
static address _polling_page;
|
||||||
static volatile int32_t * _mem_serialize_page;
|
|
||||||
static uintptr_t _serialize_page_mask;
|
|
||||||
public:
|
public:
|
||||||
static size_t _page_sizes[page_sizes_max];
|
static size_t _page_sizes[page_sizes_max];
|
||||||
|
|
||||||
@ -420,54 +418,6 @@ class os: AllStatic {
|
|||||||
static bool is_readable_pointer(const void* p);
|
static bool is_readable_pointer(const void* p);
|
||||||
static bool is_readable_range(const void* from, const void* to);
|
static bool is_readable_range(const void* from, const void* to);
|
||||||
|
|
||||||
// Routines used to serialize the thread state without using membars
|
|
||||||
static void serialize_thread_states();
|
|
||||||
|
|
||||||
// Since we write to the serialize page from every thread, we
|
|
||||||
// want stores to be on unique cache lines whenever possible
|
|
||||||
// in order to minimize CPU cross talk. We pre-compute the
|
|
||||||
// amount to shift the thread* to make this offset unique to
|
|
||||||
// each thread.
|
|
||||||
static int get_serialize_page_shift_count() {
|
|
||||||
return SerializePageShiftCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void set_serialize_page_mask(uintptr_t mask) {
|
|
||||||
_serialize_page_mask = mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
static unsigned int get_serialize_page_mask() {
|
|
||||||
return _serialize_page_mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void set_memory_serialize_page(address page);
|
|
||||||
|
|
||||||
static address get_memory_serialize_page() {
|
|
||||||
return (address)_mem_serialize_page;
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void write_memory_serialize_page(JavaThread *thread) {
|
|
||||||
uintptr_t page_offset = ((uintptr_t)thread >>
|
|
||||||
get_serialize_page_shift_count()) &
|
|
||||||
get_serialize_page_mask();
|
|
||||||
*(volatile int32_t *)((uintptr_t)_mem_serialize_page+page_offset) = 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static bool is_memory_serialize_page(JavaThread *thread, address addr) {
|
|
||||||
if (UseMembar) return false;
|
|
||||||
// Previously this function calculated the exact address of this
|
|
||||||
// thread's serialize page, and checked if the faulting address
|
|
||||||
// was equal. However, some platforms mask off faulting addresses
|
|
||||||
// to the page size, so now we just check that the address is
|
|
||||||
// within the page. This makes the thread argument unnecessary,
|
|
||||||
// but we retain the NULL check to preserve existing behavior.
|
|
||||||
if (thread == NULL) return false;
|
|
||||||
address page = (address) _mem_serialize_page;
|
|
||||||
return addr >= page && addr < (page + os::vm_page_size());
|
|
||||||
}
|
|
||||||
|
|
||||||
static void block_on_serialize_page_trap();
|
|
||||||
|
|
||||||
// threads
|
// threads
|
||||||
|
|
||||||
enum ThreadType {
|
enum ThreadType {
|
||||||
|
@ -213,16 +213,7 @@ void SafepointSynchronize::begin() {
|
|||||||
// writes and reads of both the safepoint state and the Java
|
// writes and reads of both the safepoint state and the Java
|
||||||
// threads state is critical. In order to guarantee that the
|
// threads state is critical. In order to guarantee that the
|
||||||
// memory writes are serialized with respect to each other,
|
// memory writes are serialized with respect to each other,
|
||||||
// the VM thread issues a memory barrier instruction
|
// the VM thread issues a memory barrier instruction.
|
||||||
// (on MP systems). In order to avoid the overhead of issuing
|
|
||||||
// a memory barrier for each Java thread making native calls, each Java
|
|
||||||
// thread performs a write to a single memory page after changing
|
|
||||||
// the thread state. The VM thread performs a sequence of
|
|
||||||
// mprotect OS calls which forces all previous writes from all
|
|
||||||
// Java threads to be serialized. This is done in the
|
|
||||||
// os::serialize_thread_states() call. This has proven to be
|
|
||||||
// much more efficient than executing a membar instruction
|
|
||||||
// on every call to native code.
|
|
||||||
// 3. Running compiled Code
|
// 3. Running compiled Code
|
||||||
// Compiled code reads a global (Safepoint Polling) page that
|
// Compiled code reads a global (Safepoint Polling) page that
|
||||||
// is set to fault if we are trying to get to a safepoint.
|
// is set to fault if we are trying to get to a safepoint.
|
||||||
@ -251,11 +242,6 @@ void SafepointSynchronize::begin() {
|
|||||||
}
|
}
|
||||||
OrderAccess::fence(); // storestore|storeload, global state -> local state
|
OrderAccess::fence(); // storestore|storeload, global state -> local state
|
||||||
|
|
||||||
// Flush all thread states to memory
|
|
||||||
if (!UseMembar) {
|
|
||||||
os::serialize_thread_states();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (SafepointMechanism::uses_global_page_poll()) {
|
if (SafepointMechanism::uses_global_page_poll()) {
|
||||||
// Make interpreter safepoint aware
|
// Make interpreter safepoint aware
|
||||||
Interpreter::notice_safepoints();
|
Interpreter::notice_safepoints();
|
||||||
|
@ -87,17 +87,6 @@ void SafepointMechanism::initialize_header(JavaThread* thread) {
|
|||||||
disarm_local_poll(thread);
|
disarm_local_poll(thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SafepointMechanism::initialize_serialize_page() {
|
|
||||||
if (!UseMembar) {
|
|
||||||
const size_t page_size = os::vm_page_size();
|
|
||||||
char* serialize_page = os::reserve_memory(page_size, NULL, page_size);
|
|
||||||
os::commit_memory_or_exit(serialize_page, page_size, false, "Unable to commit memory serialization page");
|
|
||||||
log_info(os)("Memory Serialize Page address: " INTPTR_FORMAT, p2i(serialize_page));
|
|
||||||
os::set_memory_serialize_page((address)(serialize_page));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void SafepointMechanism::initialize() {
|
void SafepointMechanism::initialize() {
|
||||||
pd_initialize();
|
pd_initialize();
|
||||||
initialize_serialize_page();
|
|
||||||
}
|
}
|
||||||
|
@ -52,7 +52,6 @@ class SafepointMechanism : public AllStatic {
|
|||||||
static inline void block_if_requested_local_poll(JavaThread *thread);
|
static inline void block_if_requested_local_poll(JavaThread *thread);
|
||||||
|
|
||||||
static void default_initialize();
|
static void default_initialize();
|
||||||
static void initialize_serialize_page();
|
|
||||||
|
|
||||||
static void pd_initialize() NOT_AIX({ default_initialize(); });
|
static void pd_initialize() NOT_AIX({ default_initialize(); });
|
||||||
|
|
||||||
|
@ -164,15 +164,6 @@ const int BitsPerSize_t = size_tSize * BitsPerByte;
|
|||||||
// Size of a char[] needed to represent a jint as a string in decimal.
|
// Size of a char[] needed to represent a jint as a string in decimal.
|
||||||
const int jintAsStringSize = 12;
|
const int jintAsStringSize = 12;
|
||||||
|
|
||||||
// In fact this should be
|
|
||||||
// log2_intptr(sizeof(class JavaThread)) - log2_intptr(64);
|
|
||||||
// see os::set_memory_serialize_page()
|
|
||||||
#ifdef _LP64
|
|
||||||
const int SerializePageShiftCount = 4;
|
|
||||||
#else
|
|
||||||
const int SerializePageShiftCount = 3;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// An opaque struct of heap-word width, so that HeapWord* can be a generic
|
// An opaque struct of heap-word width, so that HeapWord* can be a generic
|
||||||
// pointer into the heap. We require that object sizes be measured in
|
// pointer into the heap. We require that object sizes be measured in
|
||||||
// units of heap words, so that that
|
// units of heap words, so that that
|
||||||
|
@ -44,7 +44,6 @@ public class VMDeprecatedOptions {
|
|||||||
{"MaxRAMFraction", "8"},
|
{"MaxRAMFraction", "8"},
|
||||||
{"MinRAMFraction", "2"},
|
{"MinRAMFraction", "2"},
|
||||||
{"InitialRAMFraction", "64"},
|
{"InitialRAMFraction", "64"},
|
||||||
{"UseMembar", "true"},
|
|
||||||
{"TLABStats", "false"},
|
{"TLABStats", "false"},
|
||||||
|
|
||||||
// deprecated alias flags (see also aliased_jvm_flags):
|
// deprecated alias flags (see also aliased_jvm_flags):
|
||||||
|
Loading…
Reference in New Issue
Block a user