Skip to content

Commit 497ef5a

Browse files
committedSep 4, 2020
Change representation of frames pointing into chunks
1 parent d7d8b6a commit 497ef5a

File tree

5 files changed

+89
-52
lines changed

5 files changed

+89
-52
lines changed
 

‎src/hotspot/cpu/x86/continuation_x86.inline.hpp

+1-2
Original file line numberDiff line numberDiff line change
@@ -900,9 +900,8 @@ static frame chunk_top_frame_pd(oop chunk, intptr_t* sp, RegisterMap* map) {
900900
intptr_t** fp_address = (intptr_t**)(sp - 2);
901901
intptr_t* fp = *fp_address;
902902
if (map->update_map()) {
903-
frame::update_map_with_saved_link(map, fp_address);
903+
frame::update_map_with_saved_link(map, (intptr_t**)(intptr_t)(-2 * BytesPerWord)); // for chunk frames, we store offset
904904
}
905-
906905
return frame(sp, sp, fp, pc, ContinuationCodeBlobLookup::find_blob(pc), NULL, true);
907906
}
908907

‎src/hotspot/cpu/x86/frame_x86.hpp

+4-1
Original file line numberDiff line numberDiff line change
@@ -116,7 +116,10 @@
116116
// uses sp() to mean "raw" sp and unextended_sp() to mean the caller's
117117
// original sp we use that convention.
118118

119-
intptr_t* _unextended_sp;
119+
union {
120+
intptr_t* _unextended_sp;
121+
size_t _frame_index; // used by frames in continuation chunks
122+
};
120123
void adjust_unextended_sp() NOT_DEBUG_RETURN;
121124

122125
intptr_t* ptr_at_addr(int offset) const {

‎src/hotspot/cpu/x86/frame_x86.inline.hpp

+4
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,10 @@ inline intptr_t* frame::link() const { return (intptr_t*) *(intptr_
225225

226226
inline intptr_t* frame::unextended_sp() const { return _unextended_sp; }
227227

228+
inline size_t frame::frame_index() const { return _frame_index; }
229+
230+
inline void frame::set_frame_index(size_t index) { _frame_index = index; }
231+
228232
inline intptr_t* frame::real_fp() const {
229233
if (_cb != NULL) {
230234
// use the frame size if valid

‎src/hotspot/share/runtime/continuation.cpp

+71-45
Original file line numberDiff line numberDiff line change
@@ -690,7 +690,8 @@ class ContMirror {
690690
static bool is_in_chunk(oop chunk, void* p);
691691
static bool is_usable_in_chunk(oop chunk, void* p);
692692
static inline void reset_chunk_counters(oop chunk);
693-
oop find_chunk(void* p) const;
693+
oop find_chunk(size_t sp_offset, size_t* chunk_offset) const;
694+
oop find_chunk_by_address(void* p) const;
694695

695696
template<op_mode mode> const hframe last_frame();
696697
template<op_mode mode> void set_last_frame(const hframe& f);
@@ -1237,7 +1238,7 @@ inline void ContMirror::reset_chunk_counters(oop chunk) {
12371238
jdk_internal_misc_StackChunk::set_numOops(chunk, -1);
12381239
}
12391240

1240-
oop ContMirror::find_chunk(void* p) const {
1241+
oop ContMirror::find_chunk_by_address(void* p) const {
12411242
for (oop chunk = tail(); chunk != (oop)NULL; chunk = jdk_internal_misc_StackChunk::parent(chunk)) {
12421243
if (is_in_chunk(chunk, p)) {
12431244
assert (is_usable_in_chunk(chunk, p), "");
@@ -1247,6 +1248,23 @@ oop ContMirror::find_chunk(void* p) const {
12471248
return (oop)NULL;
12481249
}
12491250

1251+
oop ContMirror::find_chunk(size_t sp_offset, size_t* chunk_offset) const {
1252+
size_t offset = 0;
1253+
for (oop chunk = tail(); chunk != (oop)NULL; chunk = jdk_internal_misc_StackChunk::parent(chunk)) {
1254+
if (is_empty_chunk(chunk)) continue;
1255+
const size_t end = jdk_internal_misc_StackChunk::size(chunk) - jdk_internal_misc_StackChunk::argsize(chunk);
1256+
const int chunk_sp = jdk_internal_misc_StackChunk::sp(chunk);
1257+
size_t offset_in_chunk = sp_offset - offset;
1258+
if (chunk_sp + offset_in_chunk < end) {
1259+
// tty->print_cr(">>> find_chunk chunk %p sp_offset: %ld offset: %ld, chunk_sp: %d offset_in_chunk: %ld end: %ld", (oopDesc*)chunk, sp_offset, offset, chunk_sp, offset_in_chunk, end);
1260+
if (chunk_offset != NULL) *chunk_offset = offset;
1261+
return chunk;
1262+
}
1263+
offset += end - chunk_sp;
1264+
}
1265+
return (oop)NULL;
1266+
}
1267+
12501268
template<typename Event> void ContMirror::post_jfr_event(Event* e, JavaThread* jt) {
12511269
if (e->should_commit()) {
12521270
log_develop_trace(jvmcont)("JFR event: frames: %d iframes: %d size: %d refs: %d", _e_num_frames, _e_num_interpreted_frames, _e_size, _e_num_refs);
@@ -4905,10 +4923,12 @@ bool Continuation::is_scope_bottom(oop cont_scope, const frame& f, const Registe
49054923

49064924
static frame chunk_top_frame_pd(oop chunk, intptr_t* sp, RegisterMap* map);
49074925

4908-
static frame chunk_top_frame(oop chunk, RegisterMap* map) {
4926+
static frame chunk_top_frame(oop chunk, RegisterMap* map, size_t offset, size_t index) {
49094927
intptr_t* sp = (intptr_t*)InstanceStackChunkKlass::start_of_stack(chunk) + jdk_internal_misc_StackChunk::sp(chunk);
4910-
// tty->print_cr(">>> chunk_top_frame usp: %p", sp);
4911-
return chunk_top_frame_pd(chunk, sp, map);
4928+
frame f = chunk_top_frame_pd(chunk, sp, map);
4929+
f.set_offset_sp(offset);
4930+
f.set_frame_index(index);
4931+
return f;
49124932
}
49134933

49144934
static frame continuation_body_top_frame(ContMirror& cont, RegisterMap* map) {
@@ -4929,7 +4949,7 @@ static frame continuation_top_frame(oop contOop, RegisterMap* map) {
49294949
for (oop chunk = cont.tail(); chunk != (oop)NULL; chunk = jdk_internal_misc_StackChunk::parent(chunk)) {
49304950
if (!ContMirror::is_empty_chunk(chunk)) {
49314951
map->set_in_cont(true, true);
4932-
return chunk_top_frame(chunk, map);
4952+
return chunk_top_frame(chunk, map, 0, 0);
49334953
}
49344954
}
49354955

@@ -5000,29 +5020,45 @@ static frame sender_for_frame(const frame& f, RegisterMap* map) {
50005020
assert (map->in_cont(), "");
50015021
ContMirror cont(map);
50025022

5003-
oop chunk = cont.find_chunk(f.unextended_sp());
5023+
assert (!f.is_empty(), "");
5024+
50045025
bool from_chunk = false;
5005-
if (chunk != (oop)NULL) {
5026+
if (map->in_chunk()) {
50065027
from_chunk = true;
5007-
// we add to the pointer instead of subtracting from the bounds
5008-
if (ContMirror::is_in_chunk(chunk, f.unextended_sp() + f.cb()->frame_size() + jdk_internal_misc_StackChunk::argsize(chunk) + frame_metadata)) {
5009-
map->set_in_cont(false, false); // to prevent infinite recursion
5010-
frame sender = f.sender(map);
5011-
map->set_in_cont(true, true);
5012-
// tty->print_cr(">>> sender_for_frame usp: %p", sender.unextended_sp());
5028+
5029+
size_t frame_index = f.frame_index();
5030+
size_t chunk_offset;
5031+
oop chunk = cont.find_chunk(f.offset_sp(), &chunk_offset);
5032+
assert (!ContMirror::is_empty_chunk(chunk), "");
5033+
assert (chunk != (oop)NULL, "");
5034+
intptr_t* chunk_sp = (intptr_t*)InstanceStackChunkKlass::start_of_stack(chunk) + jdk_internal_misc_StackChunk::sp(chunk);
5035+
intptr_t* sp = chunk_sp + (f.offset_sp() - chunk_offset);
5036+
5037+
frame f = sp_to_frame(sp);
5038+
if (ContMirror::is_in_chunk(chunk, sp + f.cb()->frame_size() + jdk_internal_misc_StackChunk::argsize(chunk) + frame_metadata)) {
5039+
frame sender = sp_to_frame(sp + f.cb()->frame_size());
5040+
50135041
assert (ContMirror::is_usable_in_chunk(chunk, sender.unextended_sp()), "");
5042+
sender.set_offset_sp(chunk_offset + (sender.sp() - chunk_sp));
5043+
sender.set_frame_index(frame_index + 1);
5044+
if (map->update_map()) {
5045+
frame::update_map_with_saved_link(map, (intptr_t**)(intptr_t)(-2 * BytesPerWord)); // for chunk frames, we store offset TODO PD
5046+
}
50145047
return sender;
50155048
}
5049+
const size_t end = jdk_internal_misc_StackChunk::size(chunk) - jdk_internal_misc_StackChunk::argsize(chunk);
5050+
chunk_offset += end - jdk_internal_misc_StackChunk::sp(chunk);
50165051
chunk = jdk_internal_misc_StackChunk::parent(chunk);
50175052
if (chunk != (oop)NULL) {
50185053
assert (!ContMirror::is_empty_chunk(chunk), "");
5019-
return chunk_top_frame(chunk, map);
5054+
return chunk_top_frame(chunk, map, chunk_offset, frame_index + 1);
50205055
}
50215056
assert (map->in_cont(), "");
50225057
if (map->in_chunk()) map->set_in_cont(true, false);
5023-
assert (!map->in_chunk(), "");
50245058
}
50255059

5060+
assert (!map->in_chunk(), "");
5061+
50265062
hframe sender = from_chunk ? cont.last_frame<mode_slow>()
50275063
: cont.from_frame(f).sender<mode_slow>(cont);
50285064

@@ -5158,27 +5194,27 @@ address Continuation::oop_address(objArrayOop ref_stack, int ref_sp, int index)
51585194
bool Continuation::is_in_usable_stack(address addr, const RegisterMap* map) {
51595195
ContMirror cont(map);
51605196

5161-
oop chunk = cont.find_chunk(addr);
5197+
oop chunk = cont.find_chunk_by_address(addr);
51625198
assert (((intptr_t**)addr == Frame::map_link_address(map)) || (map->in_chunk() == (chunk != (oop)NULL)), "map->in_chunk(): %d", map->in_chunk());
51635199
return (chunk != (oop)NULL) ? ContMirror::is_usable_in_chunk(chunk, addr)
51645200
: (cont.is_in_stack(addr) || cont.is_in_ref_stack(addr)
51655201
|| (intptr_t**)addr == java_lang_Continuation::raw_fp_address(cont.mirror())); // TODO PD
51665202
}
51675203

5168-
// address Continuation::usp_offset_to_location(const frame& fr, const RegisterMap* map, const int usp_offset_in_bytes) {
5169-
// return usp_offset_to_location(fr, map, usp_offset_in_bytes, find_oop_in_compiled_frame(fr, map, usp_offset_in_bytes) >= 0);
5170-
// }
5171-
51725204
// if oop, it is narrow iff UseCompressedOops
51735205
address Continuation::usp_offset_to_location(const frame& fr, const RegisterMap* map, const int usp_offset_in_bytes, bool is_oop) {
51745206
assert (fr.is_compiled_frame(), "");
51755207
ContMirror cont(map);
51765208

51775209
assert(map->in_cont(), "");
51785210

5179-
assert (map->in_chunk() == (cont.find_chunk(fr.unextended_sp()) != (oop)NULL), "");
5180-
if (cont.find_chunk(fr.unextended_sp()) != (oop)NULL) {
5181-
return (address)fr.unextended_sp() + usp_offset_in_bytes;
5211+
if (map->in_chunk()) {
5212+
size_t chunk_offset;
5213+
oop chunk = cont.find_chunk(fr.offset_sp(), &chunk_offset);
5214+
assert (chunk != (oop)NULL, "");
5215+
size_t offset_in_chunk = jdk_internal_misc_StackChunk::sp(chunk) + (fr.offset_sp() - chunk_offset);
5216+
intptr_t* sp = (intptr_t*)InstanceStackChunkKlass::start_of_stack(chunk) + offset_in_chunk;
5217+
return (address)sp + usp_offset_in_bytes;
51825218
}
51835219

51845220
assert(!map->in_chunk(), "fr.unextended_sp: " INTPTR_FORMAT, p2i(fr.unextended_sp()));
@@ -5200,8 +5236,8 @@ int Continuation::usp_offset_to_index(const frame& fr, const RegisterMap* map, c
52005236
assert (fr.is_compiled_frame() || is_stub(fr.cb()), "");
52015237
ContMirror cont(map);
52025238

5203-
assert (map->in_chunk() == (cont.find_chunk(fr.unextended_sp()) != (oop)NULL), "");
5204-
if (cont.find_chunk(fr.unextended_sp()) != (oop)NULL) {
5239+
if (map->in_chunk()) {
5240+
assert (cont.find_chunk(fr.offset_sp(), NULL) != (oop)NULL, "");
52055241
return usp_offset_in_bytes;
52065242
}
52075243

@@ -5226,22 +5262,6 @@ int Continuation::usp_offset_to_index(const frame& fr, const RegisterMap* map, c
52265262
return index;
52275263
}
52285264

5229-
// address Continuation::reg_to_location(const frame& fr, const RegisterMap* map, VMReg reg) {
5230-
// return reg_to_location(fr, map, reg, find_oop_in_compiled_frame(fr, map, reg) >= 0);
5231-
// }
5232-
5233-
// address Continuation::reg_to_location(const frame& fr, const RegisterMap* map, VMReg reg, bool is_oop) {
5234-
// assert (map != NULL, "");
5235-
// oop cont;
5236-
// assert (map->in_cont(), "");
5237-
// if (map->in_cont()) {
5238-
// cont = map->cont();
5239-
// } else {
5240-
// cont = get_continutation_for_frame(map->thread(), fr);
5241-
// }
5242-
// return reg_to_location(cont, fr, map, reg, is_oop);
5243-
// }
5244-
52455265
address Continuation::reg_to_location(const frame& fr, const RegisterMap* map, VMReg reg, bool is_oop) {
52465266
assert (map != NULL, "");
52475267
assert (fr.is_compiled_frame(), "");
@@ -5255,9 +5275,15 @@ address Continuation::reg_to_location(const frame& fr, const RegisterMap* map, V
52555275

52565276
ContMirror cont(map);
52575277

5258-
assert (map->in_chunk() == (cont.find_chunk(fr.unextended_sp()) != (oop)NULL), "");
5259-
if (cont.find_chunk(fr.unextended_sp()) != (oop)NULL) {
5260-
return map->location(reg);
5278+
if (map->in_chunk()) {
5279+
intptr_t usp_offset_in_bytes = (intptr_t)map->location(reg); // see usp_offset_to_index for the chunk case
5280+
5281+
size_t chunk_offset;
5282+
oop chunk = cont.find_chunk(fr.offset_sp(), &chunk_offset);
5283+
assert (chunk != (oop)NULL, "");
5284+
size_t offset_in_chunk = jdk_internal_misc_StackChunk::sp(chunk) + (fr.offset_sp() - chunk_offset);
5285+
intptr_t* sp = (intptr_t*)InstanceStackChunkKlass::start_of_stack(chunk) + offset_in_chunk;
5286+
return (address)sp + usp_offset_in_bytes;
52615287
}
52625288

52635289
hframe hf = cont.from_frame(fr);

‎src/hotspot/share/runtime/frame.hpp

+9-4
Original file line numberDiff line numberDiff line change
@@ -54,10 +54,11 @@ class frame {
5454
// Instance variables:
5555
union {
5656
intptr_t* _sp; // stack pointer (from Thread::last_Java_sp)
57+
size_t _offset_sp; // used by frames in continuation chunks
5758
struct {
5859
int _sp;
5960
int _ref_sp;
60-
} _cont_sp;
61+
} _cont_sp; // used by frames in continuation arrays
6162
};
6263
address _pc; // program counter (the next instruction after the call)
6364
mutable CodeBlob* _cb; // CodeBlob that "owns" pc
@@ -104,10 +105,11 @@ class frame {
104105

105106
intptr_t* sp() const { return _sp; }
106107
void set_sp( intptr_t* newsp ) { _sp = newsp; }
108+
void set_offset_sp( size_t newsp ) { _offset_sp = newsp; }
107109

108-
int cont_sp() const { return _cont_sp._sp; }
109-
int cont_ref_sp() const { return _cont_sp._ref_sp; }
110-
int cont_unextended_sp() const;
110+
int cont_sp() const { return _cont_sp._sp; }
111+
int cont_ref_sp() const { return _cont_sp._ref_sp; }
112+
size_t offset_sp() const { return _offset_sp; }
111113

112114
CodeBlob* cb() const { return _cb; }
113115
inline CodeBlob* get_cb() const;
@@ -230,6 +232,9 @@ class frame {
230232
// The frame's original SP, before any extension by an interpreted callee;
231233
// used for packing debug info into vframeArray objects and vframeArray lookup.
232234
intptr_t* unextended_sp() const;
235+
236+
size_t frame_index() const;
237+
void set_frame_index(size_t index);
233238

234239
// returns the stack pointer of the calling frame
235240
intptr_t* sender_sp() const;

0 commit comments

Comments
 (0)
Please sign in to comment.