Skip to content

Commit 6321d3e

Browse files
committedNov 27, 2023
8316746: Top of lock-stack does not match the unlocked object
Reviewed-by: rrich Backport-of: 7d8adfa855e51a90c2f125fc20a06f9a488e6248
1 parent 2420592 commit 6321d3e

File tree

4 files changed

+117
-82
lines changed

4 files changed

+117
-82
lines changed
 

‎src/hotspot/cpu/ppc/frame_ppc.inline.hpp

-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,6 @@ inline intptr_t* frame::interpreter_frame_mdp_addr() const {
143143
return (intptr_t*) &(get_ijava_state()->mdx);
144144
}
145145

146-
// Pointer beyond the "oldest/deepest" BasicObjectLock on stack.
147146
inline BasicObjectLock* frame::interpreter_frame_monitor_end() const {
148147
return (BasicObjectLock*) get_ijava_state()->monitors;
149148
}

‎src/hotspot/cpu/ppc/interp_masm_ppc_64.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -1978,7 +1978,7 @@ void InterpreterMacroAssembler::profile_parameters_type(Register tmp1, Register
19781978
}
19791979
}
19801980

1981-
// Add a InterpMonitorElem to stack (see frame_sparc.hpp).
1981+
// Add a monitor (see frame_ppc.hpp).
19821982
void InterpreterMacroAssembler::add_monitor_to_stack(bool stack_is_empty, Register Rtemp1, Register Rtemp2) {
19831983

19841984
// Very-local scratch registers.

‎src/hotspot/cpu/ppc/templateTable_ppc_64.cpp

+60-80
Original file line numberDiff line numberDiff line change
@@ -4056,90 +4056,78 @@ void TemplateTable::athrow() {
40564056
// at next monitor exit.
40574057
void TemplateTable::monitorenter() {
40584058
transition(atos, vtos);
4059-
40604059
__ verify_oop(R17_tos);
40614060

4062-
Register Rcurrent_monitor = R11_scratch1,
4063-
Rcurrent_obj = R12_scratch2,
4061+
Register Rcurrent_monitor = R3_ARG1,
4062+
Rcurrent_obj = R4_ARG2,
40644063
Robj_to_lock = R17_tos,
4065-
Rscratch1 = R3_ARG1,
4066-
Rscratch2 = R4_ARG2,
4067-
Rscratch3 = R5_ARG3,
4068-
Rcurrent_obj_addr = R6_ARG4;
4064+
Rscratch1 = R11_scratch1,
4065+
Rscratch2 = R12_scratch2,
4066+
Rbot = R5_ARG3,
4067+
Rfree_slot = R6_ARG4;
4068+
4069+
Label Lfound, Lallocate_new;
4070+
4071+
__ ld(Rscratch1, _abi(callers_sp), R1_SP); // load FP
4072+
__ li(Rfree_slot, 0); // Points to free slot or null.
4073+
4074+
// Set up search loop - start with topmost monitor.
4075+
__ mr(Rcurrent_monitor, R26_monitor);
4076+
__ addi(Rbot, Rscratch1, -frame::ijava_state_size);
40694077

40704078
// ------------------------------------------------------------------------------
40714079
// Null pointer exception.
4072-
__ null_check_throw(Robj_to_lock, -1, R11_scratch1);
4080+
__ null_check_throw(Robj_to_lock, -1, Rscratch1);
40734081

4074-
// Try to acquire a lock on the object.
4075-
// Repeat until succeeded (i.e., until monitorenter returns true).
4082+
// Check if any slot is present => short cut to allocation if not.
4083+
__ cmpld(CCR0, Rcurrent_monitor, Rbot);
4084+
__ beq(CCR0, Lallocate_new);
40764085

40774086
// ------------------------------------------------------------------------------
40784087
// Find a free slot in the monitor block.
4079-
Label Lfound, Lexit, Lallocate_new;
4080-
ConditionRegister found_free_slot = CCR0,
4081-
found_same_obj = CCR1,
4082-
reached_limit = CCR6;
4088+
// Note: The order of the monitors is important for C2 OSR which derives the
4089+
// unlock order from it (see comments for interpreter_frame_monitor_*).
40834090
{
4084-
Label Lloop, Lentry;
4085-
Register Rlimit = Rcurrent_monitor;
4086-
4087-
// Set up search loop - start with topmost monitor.
4088-
__ add(Rcurrent_obj_addr, BasicObjectLock::obj_offset_in_bytes(), R26_monitor);
4091+
Label Lloop, LnotFree, Lexit;
40894092

4090-
__ ld(Rlimit, 0, R1_SP);
4091-
__ addi(Rlimit, Rlimit, - (frame::ijava_state_size + frame::interpreter_frame_monitor_size_in_bytes() - BasicObjectLock::obj_offset_in_bytes())); // Monitor base
4093+
__ bind(Lloop);
4094+
__ ld(Rcurrent_obj, BasicObjectLock::obj_offset_in_bytes(), Rcurrent_monitor);
4095+
// Exit if current entry is for same object; this guarantees, that new monitor
4096+
// used for recursive lock is above the older one.
4097+
__ cmpd(CCR0, Rcurrent_obj, Robj_to_lock);
4098+
__ beq(CCR0, Lexit); // recursive locking
40924099

4093-
// Check if any slot is present => short cut to allocation if not.
4094-
__ cmpld(reached_limit, Rcurrent_obj_addr, Rlimit);
4095-
__ bgt(reached_limit, Lallocate_new);
4100+
__ cmpdi(CCR0, Rcurrent_obj, 0);
4101+
__ bne(CCR0, LnotFree);
4102+
__ mr(Rfree_slot, Rcurrent_monitor); // remember free slot closest to the bottom
4103+
__ bind(LnotFree);
40964104

4097-
// Pre-load topmost slot.
4098-
__ ld(Rcurrent_obj, 0, Rcurrent_obj_addr);
4099-
__ addi(Rcurrent_obj_addr, Rcurrent_obj_addr, frame::interpreter_frame_monitor_size() * wordSize);
4100-
// The search loop.
4101-
__ bind(Lloop);
4102-
// Found free slot?
4103-
__ cmpdi(found_free_slot, Rcurrent_obj, 0);
4104-
// Is this entry for same obj? If so, stop the search and take the found
4105-
// free slot or allocate a new one to enable recursive locking.
4106-
__ cmpd(found_same_obj, Rcurrent_obj, Robj_to_lock);
4107-
__ cmpld(reached_limit, Rcurrent_obj_addr, Rlimit);
4108-
__ beq(found_free_slot, Lexit);
4109-
__ beq(found_same_obj, Lallocate_new);
4110-
__ bgt(reached_limit, Lallocate_new);
4111-
// Check if last allocated BasicLockObj reached.
4112-
__ ld(Rcurrent_obj, 0, Rcurrent_obj_addr);
4113-
__ addi(Rcurrent_obj_addr, Rcurrent_obj_addr, frame::interpreter_frame_monitor_size() * wordSize);
4114-
// Next iteration if unchecked BasicObjectLocks exist on the stack.
4115-
__ b(Lloop);
4105+
__ addi(Rcurrent_monitor, Rcurrent_monitor, frame::interpreter_frame_monitor_size_in_bytes());
4106+
__ cmpld(CCR0, Rcurrent_monitor, Rbot);
4107+
__ bne(CCR0, Lloop);
4108+
__ bind(Lexit);
41164109
}
41174110

41184111
// ------------------------------------------------------------------------------
41194112
// Check if we found a free slot.
4120-
__ bind(Lexit);
4121-
4122-
__ addi(Rcurrent_monitor, Rcurrent_obj_addr, -(frame::interpreter_frame_monitor_size() * wordSize) - BasicObjectLock::obj_offset_in_bytes());
4123-
__ addi(Rcurrent_obj_addr, Rcurrent_obj_addr, - frame::interpreter_frame_monitor_size() * wordSize);
4124-
__ b(Lfound);
4113+
__ cmpdi(CCR0, Rfree_slot, 0);
4114+
__ bne(CCR0, Lfound);
41254115

41264116
// We didn't find a free BasicObjLock => allocate one.
4127-
__ align(32, 12);
41284117
__ bind(Lallocate_new);
41294118
__ add_monitor_to_stack(false, Rscratch1, Rscratch2);
4130-
__ mr(Rcurrent_monitor, R26_monitor);
4131-
__ addi(Rcurrent_obj_addr, R26_monitor, BasicObjectLock::obj_offset_in_bytes());
4119+
__ mr(Rfree_slot, R26_monitor);
41324120

41334121
// ------------------------------------------------------------------------------
41344122
// We now have a slot to lock.
41354123
__ bind(Lfound);
41364124

41374125
// Increment bcp to point to the next bytecode, so exception handling for async. exceptions work correctly.
4138-
// The object has already been poped from the stack, so the expression stack looks correct.
4126+
// The object has already been popped from the stack, so the expression stack looks correct.
41394127
__ addi(R14_bcp, R14_bcp, 1);
41404128

4141-
__ std(Robj_to_lock, 0, Rcurrent_obj_addr);
4142-
__ lock_object(Rcurrent_monitor, Robj_to_lock);
4129+
__ std(Robj_to_lock, BasicObjectLock::obj_offset_in_bytes(), Rfree_slot);
4130+
__ lock_object(Rfree_slot, Robj_to_lock);
41434131

41444132
// Check if there's enough space on the stack for the monitors after locking.
41454133
// This emits a single store.
@@ -4153,46 +4141,40 @@ void TemplateTable::monitorexit() {
41534141
transition(atos, vtos);
41544142
__ verify_oop(R17_tos);
41554143

4156-
Register Rcurrent_monitor = R11_scratch1,
4157-
Rcurrent_obj = R12_scratch2,
4144+
Register Rcurrent_monitor = R3_ARG1,
4145+
Rcurrent_obj = R4_ARG2,
41584146
Robj_to_lock = R17_tos,
4159-
Rcurrent_obj_addr = R3_ARG1,
4160-
Rlimit = R4_ARG2;
4147+
Rscratch = R11_scratch1,
4148+
Rbot = R12_scratch2;
4149+
41614150
Label Lfound, Lillegal_monitor_state;
41624151

4163-
// Check corner case: unbalanced monitorEnter / Exit.
4164-
__ ld(Rlimit, 0, R1_SP);
4165-
__ addi(Rlimit, Rlimit, - (frame::ijava_state_size + frame::interpreter_frame_monitor_size_in_bytes())); // Monitor base
4152+
__ ld(Rscratch, _abi(callers_sp), R1_SP); // load FP
4153+
4154+
// Set up search loop - start with topmost monitor.
4155+
__ mr(Rcurrent_monitor, R26_monitor);
4156+
__ addi(Rbot, Rscratch, -frame::ijava_state_size);
41664157

41674158
// Null pointer check.
4168-
__ null_check_throw(Robj_to_lock, -1, R11_scratch1);
4159+
__ null_check_throw(Robj_to_lock, -1, Rscratch);
41694160

4170-
__ cmpld(CCR0, R26_monitor, Rlimit);
4171-
__ bgt(CCR0, Lillegal_monitor_state);
4161+
// Check corner case: unbalanced monitorEnter / Exit.
4162+
__ cmpld(CCR0, Rcurrent_monitor, Rbot);
4163+
__ beq(CCR0, Lillegal_monitor_state);
41724164

41734165
// Find the corresponding slot in the monitors stack section.
41744166
{
41754167
Label Lloop;
41764168

4177-
// Start with topmost monitor.
4178-
__ addi(Rcurrent_obj_addr, R26_monitor, BasicObjectLock::obj_offset_in_bytes());
4179-
__ addi(Rlimit, Rlimit, BasicObjectLock::obj_offset_in_bytes());
4180-
__ ld(Rcurrent_obj, 0, Rcurrent_obj_addr);
4181-
__ addi(Rcurrent_obj_addr, Rcurrent_obj_addr, frame::interpreter_frame_monitor_size() * wordSize);
4182-
41834169
__ bind(Lloop);
4170+
__ ld(Rcurrent_obj, BasicObjectLock::obj_offset_in_bytes(), Rcurrent_monitor);
41844171
// Is this entry for same obj?
41854172
__ cmpd(CCR0, Rcurrent_obj, Robj_to_lock);
41864173
__ beq(CCR0, Lfound);
41874174

4188-
// Check if last allocated BasicLockObj reached.
4189-
4190-
__ ld(Rcurrent_obj, 0, Rcurrent_obj_addr);
4191-
__ cmpld(CCR0, Rcurrent_obj_addr, Rlimit);
4192-
__ addi(Rcurrent_obj_addr, Rcurrent_obj_addr, frame::interpreter_frame_monitor_size() * wordSize);
4193-
4194-
// Next iteration if unchecked BasicObjectLocks exist on the stack.
4195-
__ ble(CCR0, Lloop);
4175+
__ addi(Rcurrent_monitor, Rcurrent_monitor, frame::interpreter_frame_monitor_size_in_bytes());
4176+
__ cmpld(CCR0, Rcurrent_monitor, Rbot);
4177+
__ bne(CCR0, Lloop);
41964178
}
41974179

41984180
// Fell through without finding the basic obj lock => throw up!
@@ -4202,8 +4184,6 @@ void TemplateTable::monitorexit() {
42024184

42034185
__ align(32, 12);
42044186
__ bind(Lfound);
4205-
__ addi(Rcurrent_monitor, Rcurrent_obj_addr,
4206-
-(frame::interpreter_frame_monitor_size() * wordSize) - BasicObjectLock::obj_offset_in_bytes());
42074187
__ unlock_object(Rcurrent_monitor);
42084188
}
42094189

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2023 SAP SE. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
/*
26+
* @test
27+
* @bug 8316746
28+
* @summary During OSR, locks get transferred from interpreter frame.
29+
* Check that unlocking 2 such locks works in the OSR compiled nmethod.
30+
* Some platforms verify that the unlocking happens in the corrent order.
31+
*
32+
* @run main/othervm -Xbatch TestUnlockOSR
33+
*/
34+
35+
public class TestUnlockOSR {
36+
static void test_method(Object a, Object b, int limit) {
37+
synchronized(a) { // allocate space for monitors
38+
synchronized(b) {
39+
}
40+
} // free space to test allocation in reused space
41+
synchronized(a) { // reuse the space
42+
synchronized(b) {
43+
for (int i = 0; i < limit; i++) {}
44+
}
45+
}
46+
}
47+
48+
public static void main(String[] args) {
49+
Object a = new TestUnlockOSR(),
50+
b = new TestUnlockOSR();
51+
// avoid uncommon trap before last unlocks
52+
for (int i = 0; i < 100; i++) { test_method(a, b, 0); }
53+
// trigger OSR
54+
test_method(a, b, 100000);
55+
}
56+
}

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Nov 27, 2023

@openjdk-notifier[bot]
Please sign in to comment.