Skip to content

Commit f4d8c20

Browse files
author
Tyler Steele
committedOct 31, 2022
8294729: [s390] Implement nmethod entry barriers
Reviewed-by: mdoerr, eosterlund
1 parent 7e88209 commit f4d8c20

10 files changed

+207
-9
lines changed
 

‎src/hotspot/cpu/s390/assembler_s390.hpp

+3
Original file line numberDiff line numberDiff line change
@@ -3121,6 +3121,9 @@ class Assembler : public AbstractAssembler {
31213121
static bool is_z_algr(long x) {
31223122
return (ALGR_ZOPC == (x & RRE_MASK));
31233123
}
3124+
static bool is_z_cfi(long x) {
3125+
return (CFI_ZOPC == (x & RIL_MASK));
3126+
}
31243127
static bool is_z_lb(long x) {
31253128
return (LB_ZOPC == (x & LB_MASK));
31263129
}

‎src/hotspot/cpu/s390/c1_MacroAssembler_s390.cpp

+5
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,8 @@
2727
#include "asm/macroAssembler.inline.hpp"
2828
#include "c1/c1_MacroAssembler.hpp"
2929
#include "c1/c1_Runtime1.hpp"
30+
#include "gc/shared/barrierSet.hpp"
31+
#include "gc/shared/barrierSetAssembler.hpp"
3032
#include "gc/shared/collectedHeap.hpp"
3133
#include "gc/shared/tlab_globals.hpp"
3234
#include "interpreter/interpreter.hpp"
@@ -72,6 +74,9 @@ void C1_MacroAssembler::build_frame(int frame_size_in_bytes, int bang_size_in_by
7274
generate_stack_overflow_check(bang_size_in_bytes);
7375
save_return_pc();
7476
push_frame(frame_size_in_bytes);
77+
78+
BarrierSetAssembler* bs = BarrierSet::barrier_set()->barrier_set_assembler();
79+
bs->nmethod_entry_barrier(this);
7580
}
7681

7782
void C1_MacroAssembler::verified_entry(bool breakAtEntry) {

‎src/hotspot/cpu/s390/gc/shared/barrierSetAssembler_s390.cpp

+29-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2018 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -25,10 +25,13 @@
2525

2626
#include "precompiled.hpp"
2727
#include "asm/macroAssembler.inline.hpp"
28+
#include "gc/shared/barrierSet.hpp"
2829
#include "gc/shared/barrierSetAssembler.hpp"
30+
#include "gc/shared/barrierSetNMethod.hpp"
2931
#include "interpreter/interp_masm.hpp"
3032
#include "oops/compressedOops.hpp"
3133
#include "runtime/jniHandles.hpp"
34+
#include "runtime/stubRoutines.hpp"
3235
#include "utilities/macros.hpp"
3336

3437
#define __ masm->
@@ -119,3 +122,28 @@ void BarrierSetAssembler::try_resolve_jobject_in_native(MacroAssembler* masm, Re
119122
__ z_nill(obj, ~JNIHandles::weak_tag_mask);
120123
__ z_lg(obj, 0, obj); // Resolve (untagged) jobject.
121124
}
125+
126+
void BarrierSetAssembler::nmethod_entry_barrier(MacroAssembler* masm) {
127+
BarrierSetNMethod* bs_nm = BarrierSet::barrier_set()->barrier_set_nmethod();
128+
if (bs_nm == nullptr) {
129+
return;
130+
}
131+
132+
__ block_comment("nmethod_entry_barrier (nmethod_entry_barrier) {");
133+
134+
// Load jump addr:
135+
__ load_const(Z_R1_scratch, (uint64_t)StubRoutines::zarch::nmethod_entry_barrier()); // 2*6 bytes
136+
137+
// Load value from current java object:
138+
__ z_lg(Z_R0_scratch, in_bytes(bs_nm->thread_disarmed_offset()), Z_thread); // 6 bytes
139+
140+
// Compare to current patched value:
141+
__ z_cfi(Z_R0_scratch, /* to be patched */ -1); // 6 bytes (2 + 4 byte imm val)
142+
143+
// Conditional Jump
144+
__ z_larl(Z_R14, (Assembler::instr_len((unsigned long)LARL_ZOPC) + Assembler::instr_len((unsigned long)BCR_ZOPC)) / 2); // 6 bytes
145+
__ z_bcr(Assembler::bcondNotEqual, Z_R1_scratch); // 2 bytes
146+
147+
// Fall through to method body.
148+
__ block_comment("} nmethod_entry_barrier (nmethod_entry_barrier)");
149+
}

‎src/hotspot/cpu/s390/gc/shared/barrierSetAssembler_s390.hpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2018 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -49,6 +49,8 @@ class BarrierSetAssembler: public CHeapObj<mtGC> {
4949
virtual void try_resolve_jobject_in_native(MacroAssembler* masm, Register jni_env,
5050
Register obj, Register tmp, Label& slowpath);
5151

52+
virtual void nmethod_entry_barrier(MacroAssembler* masm);
53+
5254
virtual void barrier_stubs_init() {}
5355
};
5456

‎src/hotspot/cpu/s390/gc/shared/barrierSetNMethod_s390.cpp

+94-5
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2022, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -23,18 +23,107 @@
2323
*/
2424

2525
#include "precompiled.hpp"
26+
#include "asm/assembler.inline.hpp"
27+
#include "code/codeBlob.hpp"
28+
#include "code/nativeInst.hpp"
29+
#include "code/nmethod.hpp"
2630
#include "gc/shared/barrierSetNMethod.hpp"
2731
#include "utilities/debug.hpp"
2832

33+
class NativeMethodBarrier: public NativeInstruction {
34+
private:
35+
static const int PATCHABLE_INSTRUCTION_OFFSET = 3*6; // bytes
36+
37+
address get_barrier_start_address() const {
38+
return NativeInstruction::addr_at(0);
39+
}
40+
41+
address get_patchable_data_address() const {
42+
address inst_addr = get_barrier_start_address() + PATCHABLE_INSTRUCTION_OFFSET;
43+
44+
debug_only(Assembler::is_z_cfi(*((long*)inst_addr)));
45+
return inst_addr + 2;
46+
}
47+
48+
public:
49+
static const int BARRIER_TOTAL_LENGTH = PATCHABLE_INSTRUCTION_OFFSET + 2*6 + 2; // bytes
50+
51+
int get_guard_value() const {
52+
address data_addr = get_patchable_data_address();
53+
// Return guard instruction value
54+
return *((int32_t*)data_addr);
55+
}
56+
57+
void set_guard_value(int value) {
58+
int32_t* data_addr = (int32_t*)get_patchable_data_address();
59+
60+
// Set guard instruction value
61+
*data_addr = value;
62+
}
63+
64+
#ifdef ASSERT
65+
void verify() const {
66+
int offset = 0; // bytes
67+
const address start = get_barrier_start_address();
68+
69+
MacroAssembler::is_load_const(/* address */ start + offset); // two instructions
70+
offset += Assembler::instr_len(&start[offset]);
71+
offset += Assembler::instr_len(&start[offset]);
72+
73+
Assembler::is_z_lg(*((long*)(start + offset)));
74+
offset += Assembler::instr_len(&start[offset]);
75+
76+
Assembler::is_z_cfi(*((long*)(start + offset)));
77+
offset += Assembler::instr_len(&start[offset]);
78+
79+
Assembler::is_z_larl(*((long*)(start + offset)));
80+
offset += Assembler::instr_len(&start[offset]);
81+
82+
Assembler::is_z_bcr(*((long*)(start + offset)));
83+
offset += Assembler::instr_len(&start[offset]);
84+
85+
assert(offset == BARRIER_TOTAL_LENGTH, "check offset == barrier length constant");
86+
}
87+
#endif
88+
89+
};
90+
91+
static NativeMethodBarrier* get_nmethod_barrier(nmethod* nm) {
92+
address barrier_address = nm->code_begin() + nm->frame_complete_offset() - NativeMethodBarrier::BARRIER_TOTAL_LENGTH;
93+
auto barrier = reinterpret_cast<NativeMethodBarrier*>(barrier_address);
94+
95+
debug_only(barrier->verify());
96+
return barrier;
97+
}
98+
2999
void BarrierSetNMethod::deoptimize(nmethod* nm, address* return_address_ptr) {
30-
ShouldNotReachHere();
100+
// Not required on s390 as a valid backchain is present
101+
return;
102+
}
103+
104+
void BarrierSetNMethod::arm(nmethod* nm, int arm_value) {
105+
if (!supports_entry_barrier(nm)) {
106+
return;
107+
}
108+
109+
NativeMethodBarrier* barrier = get_nmethod_barrier(nm);
110+
barrier->set_guard_value(arm_value);
31111
}
32112

33113
void BarrierSetNMethod::disarm(nmethod* nm) {
34-
ShouldNotReachHere();
114+
if (!supports_entry_barrier(nm)) {
115+
return;
116+
}
117+
118+
NativeMethodBarrier* barrier = get_nmethod_barrier(nm);
119+
barrier->set_guard_value(disarmed_value());
35120
}
36121

37122
bool BarrierSetNMethod::is_armed(nmethod* nm) {
38-
ShouldNotReachHere();
39-
return false;
123+
if (!supports_entry_barrier(nm)) {
124+
return false;
125+
}
126+
127+
NativeMethodBarrier* barrier = get_nmethod_barrier(nm);
128+
return barrier->get_guard_value() != disarmed_value();
40129
}

‎src/hotspot/cpu/s390/s390.ad

+13
Original file line numberDiff line numberDiff line change
@@ -810,6 +810,8 @@ void MachConstantBaseNode::format(PhaseRegAlloc* ra_, outputStream* st) const {
810810

811811
//=============================================================================
812812

813+
#include "gc/shared/barrierSetAssembler.hpp"
814+
813815
#if !defined(PRODUCT)
814816
void MachPrologNode::format(PhaseRegAlloc *ra_, outputStream *st) const {
815817
Compile* C = ra_->C;
@@ -837,6 +839,10 @@ void MachPrologNode::format(PhaseRegAlloc *ra_, outputStream *st) const {
837839
}
838840
st->print_cr("push_frame %d", (int)-framesize);
839841
st->print("\t");
842+
843+
if (C->stub_function() == NULL) {
844+
st->print("nmethod entry barrier\n\t");
845+
}
840846
}
841847
#endif
842848

@@ -890,6 +896,13 @@ void MachPrologNode::emit(CodeBuffer &cbuf, PhaseRegAlloc *ra_) const {
890896
ConstantTable& constant_table = C->output()->constant_table();
891897
constant_table.set_table_base_offset(constant_table.calculate_table_base_offset());
892898
}
899+
900+
if (C->stub_function() == NULL) {
901+
BarrierSetAssembler* bs = BarrierSet::barrier_set()->barrier_set_assembler();
902+
bs->nmethod_entry_barrier(&_masm);
903+
}
904+
905+
C->output()->set_frame_complete(cbuf.insts_size());
893906
}
894907

895908
uint MachPrologNode::size(PhaseRegAlloc *ra_) const {

‎src/hotspot/cpu/s390/sharedRuntime_s390.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
#include "code/icBuffer.hpp"
3030
#include "code/vtableStubs.hpp"
3131
#include "compiler/oopMap.hpp"
32+
#include "gc/shared/barrierSetAssembler.hpp"
3233
#include "gc/shared/gcLocker.hpp"
3334
#include "interpreter/interpreter.hpp"
3435
#include "interpreter/interp_masm.hpp"
@@ -1541,6 +1542,9 @@ nmethod *SharedRuntime::generate_native_wrapper(MacroAssembler *masm,
15411542
// Just resize the existing one.
15421543
#endif
15431544

1545+
BarrierSetAssembler* bs = BarrierSet::barrier_set()->barrier_set_assembler();
1546+
bs->nmethod_entry_barrier(masm);
1547+
15441548
wrapper_FrameDone = __ offset();
15451549

15461550
__ verify_thread();

‎src/hotspot/cpu/s390/stubGenerator_s390.cpp

+48
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
#include "registerSaver_s390.hpp"
2929
#include "gc/shared/barrierSet.hpp"
3030
#include "gc/shared/barrierSetAssembler.hpp"
31+
#include "gc/shared/barrierSetNMethod.hpp"
3132
#include "interpreter/interpreter.hpp"
3233
#include "interpreter/interp_masm.hpp"
3334
#include "memory/universe.hpp"
@@ -2857,6 +2858,47 @@ class StubGenerator: public StubCodeGenerator {
28572858
return start;
28582859
}
28592860

2861+
address generate_nmethod_entry_barrier() {
2862+
__ align(CodeEntryAlignment);
2863+
StubCodeMark mark(this, "StubRoutines", "nmethod_entry_barrier");
2864+
2865+
address start = __ pc();
2866+
2867+
int nbytes_volatile = (8 + 5) * BytesPerWord;
2868+
2869+
// VM-Call Prologue
2870+
__ save_return_pc();
2871+
__ push_frame_abi160(nbytes_volatile);
2872+
__ save_volatile_regs(Z_SP, frame::z_abi_160_size, true, false);
2873+
2874+
// Prep arg for VM call
2875+
// Create ptr to stored return_pc in caller frame.
2876+
__ z_la(Z_ARG1, _z_abi(return_pc) + frame::z_abi_160_size + nbytes_volatile, Z_R0, Z_SP);
2877+
2878+
// VM-Call: BarrierSetNMethod::nmethod_stub_entry_barrier(address* return_address_ptr)
2879+
__ call_VM_leaf(CAST_FROM_FN_PTR(address, BarrierSetNMethod::nmethod_stub_entry_barrier));
2880+
__ z_ltr(Z_R0_scratch, Z_RET);
2881+
2882+
// VM-Call Epilogue
2883+
__ restore_volatile_regs(Z_SP, frame::z_abi_160_size, true, false);
2884+
__ pop_frame();
2885+
__ restore_return_pc();
2886+
2887+
// Check return val of VM-Call
2888+
__ z_bcr(Assembler::bcondZero, Z_R14);
2889+
2890+
// Pop frame built in prologue.
2891+
// Required so wrong_method_stub can deduce caller.
2892+
__ pop_frame();
2893+
__ restore_return_pc();
2894+
2895+
// VM-Call indicates deoptimization required
2896+
__ load_const_optimized(Z_R1_scratch, SharedRuntime::get_handle_wrong_method_stub());
2897+
__ z_br(Z_R1_scratch);
2898+
2899+
return start;
2900+
}
2901+
28602902
address generate_cont_thaw(bool return_barrier, bool exception) {
28612903
if (!Continuations::enabled()) return nullptr;
28622904
Unimplemented();
@@ -2998,6 +3040,12 @@ class StubGenerator: public StubCodeGenerator {
29983040
StubRoutines::_sha512_implCompressMB = generate_SHA512_stub(true, "SHA512_multiBlock");
29993041
}
30003042

3043+
// nmethod entry barriers for concurrent class unloading
3044+
BarrierSetNMethod* bs_nm = BarrierSet::barrier_set()->barrier_set_nmethod();
3045+
if (bs_nm != NULL) {
3046+
StubRoutines::zarch::_nmethod_entry_barrier = generate_nmethod_entry_barrier();
3047+
}
3048+
30013049
#ifdef COMPILER2
30023050
if (UseMultiplyToLenIntrinsic) {
30033051
StubRoutines::_multiplyToLen = generate_multiplyToLen();

‎src/hotspot/cpu/s390/stubRoutines_s390.cpp

+3-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2016, 2017, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2016, 2022, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2016, 2017 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -38,6 +38,8 @@ address StubRoutines::zarch::_partial_subtype_check = NULL;
3838
// Comapct string intrinsics: Translate table for string inflate intrinsic. Used by trot instruction.
3939
address StubRoutines::zarch::_trot_table_addr = NULL;
4040

41+
address StubRoutines::zarch::_nmethod_entry_barrier = NULL;
42+
4143
int StubRoutines::zarch::_atomic_memory_operation_lock = StubRoutines::zarch::unlocked;
4244

4345
#define __ masm->

‎src/hotspot/cpu/s390/stubRoutines_s390.hpp

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2016, 2022, Oracle and/or its affiliates. All rights reserved.
33
* Copyright (c) 2016, 2017 SAP SE. All rights reserved.
44
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
55
*
@@ -78,6 +78,8 @@ class zarch {
7878
static address _trot_table_addr;
7979
static jlong _trot_table[TROT_COLUMN_SIZE];
8080

81+
static address _nmethod_entry_barrier;
82+
8183
public:
8284
// Global lock for everyone who needs to use atomic_compare_and_exchange
8385
// or atomic_increment -- should probably use more locks for more
@@ -98,6 +100,8 @@ class zarch {
98100

99101
// Comapct string intrinsics: Translate table for string inflate intrinsic. Used by trot instruction.
100102
static void generate_load_trot_table_addr(MacroAssembler* masm, Register table);
103+
104+
static address nmethod_entry_barrier() { return _nmethod_entry_barrier; }
101105
};
102106

103107
#endif // CPU_S390_STUBROUTINES_S390_HPP

0 commit comments

Comments
 (0)
Please sign in to comment.