1 /*
2 * Copyright (c) 2022, 2024, Oracle and/or its affiliates. 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 #ifndef CPU_RISCV_CONTINUATIONHELPER_RISCV_INLINE_HPP
26 #define CPU_RISCV_CONTINUATIONHELPER_RISCV_INLINE_HPP
27
28 #include "runtime/continuationHelper.hpp"
29
30 #include "runtime/continuationEntry.inline.hpp"
31 #include "runtime/frame.inline.hpp"
32 #include "runtime/registerMap.hpp"
33 #include "utilities/macros.hpp"
34
35 template<typename FKind>
36 static inline intptr_t** link_address(const frame& f) {
37 assert(FKind::is_instance(f), "");
38 return FKind::interpreted
39 ? (intptr_t**)(f.fp() + frame::link_offset)
40 : (intptr_t**)(f.unextended_sp() + f.cb()->frame_size() - 2);
41 }
42
43 static inline void patch_return_pc_with_preempt_stub(frame& f) {
44 if (f.is_runtime_frame()) {
45 // Unlike x86 we don't know where in the callee frame the return pc is
46 // saved so we can't patch the return from the VM call back to Java.
47 // Instead, we will patch the return from the runtime stub back to the
48 // compiled method so that the target returns to the preempt cleanup stub.
49 intptr_t* caller_sp = f.sp() + f.cb()->frame_size();
50 caller_sp[-1] = (intptr_t)StubRoutines::cont_preempt_stub();
51 } else {
52 // The target will check for preemption once it returns to the interpreter
53 // or the native wrapper code and will manually jump to the preempt stub.
54 JavaThread *thread = JavaThread::current();
55 DEBUG_ONLY(Method* m = f.is_interpreted_frame() ? f.interpreter_frame_method() : f.cb()->as_nmethod()->method();)
56 assert(m->is_object_wait0() || thread->interp_at_preemptable_vmcall_cnt() > 0,
57 "preemptable VM call not using call_VM_preemptable");
58 thread->set_preempt_alternate_return(StubRoutines::cont_preempt_stub());
59 }
60 }
61
62 inline int ContinuationHelper::frame_align_words(int size) {
63 #ifdef _LP64
64 return size & 1;
65 #else
66 return 0;
67 #endif
68 }
69
70 inline intptr_t* ContinuationHelper::frame_align_pointer(intptr_t* sp) {
71 #ifdef _LP64
72 sp = align_down(sp, frame::frame_alignment);
73 #endif
74 return sp;
75 }
76
77 template<typename FKind>
78 inline void ContinuationHelper::update_register_map(const frame& f, RegisterMap* map) {
79 frame::update_map_with_saved_link(map, link_address<FKind>(f));
80 }
81
82 inline void ContinuationHelper::update_register_map_with_callee(const frame& f, RegisterMap* map) {
83 frame::update_map_with_saved_link(map, ContinuationHelper::Frame::callee_link_address(f));
84 }
85
86 inline void ContinuationHelper::push_pd(const frame& f) {
87 *(intptr_t**)(f.sp() - 2) = f.fp();
88 }
89
90 inline void ContinuationHelper::set_anchor_to_entry_pd(JavaFrameAnchor* anchor, ContinuationEntry* entry) {
91 anchor->set_last_Java_fp(entry->entry_fp());
92 }
93
94 inline void ContinuationHelper::set_anchor_pd(JavaFrameAnchor* anchor, intptr_t* sp) {
95 intptr_t* fp = *(intptr_t**)(sp - 2);
96 anchor->set_last_Java_fp(fp);
97 }
98
99 #ifdef ASSERT
100 inline bool ContinuationHelper::Frame::assert_frame_laid_out(frame f) {
101 intptr_t* sp = f.sp();
102 address pc = *(address*)(sp - frame::sender_sp_ret_address_offset());
103 intptr_t* fp = *(intptr_t**)(sp - 2);
104 assert(f.raw_pc() == pc, "f.ra_pc: " INTPTR_FORMAT " actual: " INTPTR_FORMAT, p2i(f.raw_pc()), p2i(pc));
105 assert(f.fp() == fp, "f.fp: " INTPTR_FORMAT " actual: " INTPTR_FORMAT, p2i(f.fp()), p2i(fp));
106 return f.raw_pc() == pc && f.fp() == fp;
107 }
108 #endif
109
110 inline intptr_t** ContinuationHelper::Frame::callee_link_address(const frame& f) {
111 return (intptr_t**)(f.sp() - 2);
112 }
113
114 inline address* ContinuationHelper::Frame::return_pc_address(const frame& f) {
115 return (address*)(f.real_fp() - 1);
116 }
117
118 inline address* ContinuationHelper::InterpretedFrame::return_pc_address(const frame& f) {
119 return (address*)(f.fp() + frame::return_addr_offset);
120 }
121
122 inline void ContinuationHelper::InterpretedFrame::patch_sender_sp(frame& f, const frame& caller) {
123 intptr_t* sp = caller.unextended_sp();
124 assert(f.is_interpreted_frame(), "");
125 intptr_t* la = f.addr_at(frame::interpreter_frame_sender_sp_offset);
126 *la = f.is_heap_frame() ? (intptr_t)(sp - f.fp()) : (intptr_t)sp;
127 }
128
129 inline address ContinuationHelper::Frame::real_pc(const frame& f) {
130 address* pc_addr = &(((address*) f.sp())[-1]);
131 return *pc_addr;
132 }
133
134 inline void ContinuationHelper::Frame::patch_pc(const frame& f, address pc) {
135 address* pc_addr = &(((address*) f.sp())[-1]);
136 *pc_addr = pc;
137 }
138
139 inline intptr_t* ContinuationHelper::InterpretedFrame::frame_top(const frame& f, InterpreterOopMap* mask) { // inclusive; this will be copied with the frame
140 // interpreter_frame_last_sp_offset, points to unextended_sp includes arguments in the frame
141 // interpreter_frame_initial_sp_offset excludes expression stack slots
142 int expression_stack_sz = expression_stack_size(f, mask);
143 intptr_t* res = (intptr_t*)f.at_relative(frame::interpreter_frame_initial_sp_offset) - expression_stack_sz;
144 assert(res == (intptr_t*)f.interpreter_frame_monitor_end() - expression_stack_sz, "");
145 assert(res >= f.unextended_sp(),
146 "res: " INTPTR_FORMAT " initial_sp: " INTPTR_FORMAT " last_sp: " INTPTR_FORMAT " unextended_sp: " INTPTR_FORMAT " expression_stack_size: %d",
147 p2i(res), p2i(f.addr_at(frame::interpreter_frame_initial_sp_offset)), f.at_relative_or_null(frame::interpreter_frame_last_sp_offset),
148 p2i(f.unextended_sp()), expression_stack_sz);
149 return res;
150 }
151
152 inline intptr_t* ContinuationHelper::InterpretedFrame::frame_bottom(const frame& f) { // exclusive; this will not be copied with the frame
153 return (intptr_t*)f.at_relative(frame::interpreter_frame_locals_offset) + 1; // exclusive, so we add 1 word
154 }
155
156 inline intptr_t* ContinuationHelper::InterpretedFrame::frame_top(const frame& f, int callee_argsize, bool callee_interpreted) {
157 return f.unextended_sp() + (callee_interpreted ? callee_argsize : 0);
158 }
159
160 inline intptr_t* ContinuationHelper::InterpretedFrame::callers_sp(const frame& f) {
161 return f.fp();
162 }
163
164 #endif // CPU_RISCV_CONTINUATIONFRAMEHELPERS_RISCV_INLINE_HPP