/* * Copyright (c) 2000, 2022, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2012, 2022 SAP SE. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. *
*/
// consider stack guards when trying to determine "safe" stack pointers // sp must be within the usable part of the stack (not in guards) if (!thread->is_in_usable_stack(sp)) { returnfalse;
}
// Unextended sp must be within the stack if (!thread->is_in_full_stack_checked(unextended_sp)) { returnfalse;
}
// An fp must be within the stack and above (but not equal) sp. bool fp_safe = thread->is_in_stack_range_excl(fp, sp); // An interpreter fp must be fp_safe. // Moreover, it must be at a distance at least the size of the ijava_state structure. bool fp_interp_safe = fp_safe && ((fp - sp) >= ijava_state_size);
// We know sp/unextended_sp are safe, only fp is questionable here
// If the current frame is known to the code cache then we can attempt to // construct the sender and do some validation of it. This goes a long way // toward eliminating issues when we get in frame construction code
if (_cb != NULL) {
// First check if the frame is complete and the test is reliable. // Unfortunately we can only check frame completeness for runtime stubs // and nmethods. Other generic buffer blobs are more problematic // so we just assume they are OK. // Adapter blobs never have a complete frame and are never OK if (!_cb->is_frame_complete_at(_pc)) { if (_cb->is_compiled() || _cb->is_adapter_blob() || _cb->is_runtime_stub()) { returnfalse;
}
}
// Could just be some random pointer within the codeBlob. if (!_cb->code_contains(_pc)) { returnfalse;
}
// Entry frame checks if (is_entry_frame()) { // An entry frame must have a valid fp. return fp_safe && is_entry_frame_valid(thread);
}
if (is_interpreted_frame() && !fp_interp_safe) { returnfalse;
}
// At this point, there still is a chance that fp_safe is false. // In particular, (fp == NULL) might be true. So let's check and // bail out before we actually dereference from fp. if (!fp_safe) { returnfalse;
}
if (Continuation::is_return_barrier_entry(sender_pc)) { // If our sender_pc is the return barrier, then our "real" sender is the continuation entry
frame s = Continuation::continuation_bottom_sender(thread, *this, sender_sp);
sender_sp = s.sp();
sender_pc = s.pc();
}
// We must always be able to find a recognizable pc.
CodeBlob* sender_blob = CodeCache::find_blob(sender_pc); if (sender_blob == NULL) { returnfalse;
}
// It should be safe to construct the sender though it might not be valid.
frame sender(sender_sp, sender_pc);
// Do we have a valid fp?
address sender_fp = (address) sender.fp();
// sender_fp must be within the stack and above (but not // equal) current frame's fp. if (!thread->is_in_stack_range_excl(sender_fp, fp)) { returnfalse;
}
// If the potential sender is the interpreter then we can do some more checking. if (Interpreter::contains(sender_pc)) { return sender.is_interpreted_frame_valid(thread);
}
// Could just be some random pointer within the codeBlob. if (!sender.cb()->code_contains(sender_pc)) { returnfalse;
}
// We should never be able to see an adapter if the current frame is something from code cache. if (sender_blob->is_adapter_blob()) { returnfalse;
}
if (sender.is_entry_frame()) { return sender.is_entry_frame_valid(thread);
}
// Frame size is always greater than zero. If the sender frame size is zero or less, // something is really weird and we better give up. if (sender_blob->frame_size() <= 0) { returnfalse;
}
returntrue;
}
// Must be native-compiled frame. Since sender will try and use fp to find // linkages it must be safe
if (!fp_safe) { returnfalse;
}
returntrue;
}
frame frame::sender_for_entry_frame(RegisterMap *map) const {
assert(map != NULL, "map must be set"); // Java frame called from C; skip all C frames and return top C // frame of that chunk as the sender.
JavaFrameAnchor* jfa = entry_frame_call_wrapper()->anchor();
assert(!entry_frame_is_first(), "next Java fp must be non zero");
assert(jfa->last_Java_sp() > _sp, "must be above this frame on stack");
map->clear();
assert(map->include_argument_oops(), "should be set by clear");
if (jfa->last_Java_pc() != NULL) {
frame fr(jfa->last_Java_sp(), jfa->last_Java_pc()); return fr;
} // Last_java_pc is not set, if we come here from compiled code. The // constructor retrieves the PC from the stack.
frame fr(jfa->last_Java_sp()); return fr;
}
frame frame::sender_for_interpreter_frame(RegisterMap *map) const { // This is the sp before any possible extension (adapter/locals).
intptr_t* unextended_sp = interpreter_frame_sender_sp();
address sender_pc = this->sender_pc(); if (Continuation::is_return_barrier_entry(sender_pc)) { if (map->walk_cont()) { // about to walk into an h-stack return Continuation::top_frame(*this, map);
} else { return Continuation::continuation_bottom_sender(map->thread(), *this, sender_sp());
}
}
bool frame::is_interpreted_frame_valid(JavaThread* thread) const {
assert(is_interpreted_frame(), "Not an interpreted frame"); // These are reasonable sanity checks if (fp() == 0 || (intptr_t(fp()) & (wordSize-1)) != 0) { returnfalse;
} if (sp() == 0 || (intptr_t(sp()) & (wordSize-1)) != 0) { returnfalse;
} int min_frame_slots = (abi_minframe_size + ijava_state_size) / sizeof(intptr_t); if (fp() - min_frame_slots < sp()) { returnfalse;
} // These are hacks to keep us out of trouble. // The problem with these is that they mask other problems if (fp() <= sp()) { // this attempts to deal with unsigned comparison above returnfalse;
}
// do some validation of frame elements
// first the method
Method* m = *interpreter_frame_method_addr();
// validate the method we'd find in this potential sender if (!Method::is_valid_method(m)) returnfalse;
// stack frames shouldn't be much larger than max_stack elements // this test requires the use of unextended_sp which is the sp as seen by // the current frame, and not sp which is the "raw" pc which could point // further because of local variables of the callee method inserted after // method arguments if (fp() - unextended_sp() > 1024 + m->max_stack()*Interpreter::stackElementSize) { returnfalse;
}
if (method->is_native()) { // Prior to calling into the runtime to notify the method exit the possible // result value is saved into the interpreter frame.
address lresult = (address)&(get_ijava_state()->lresult);
address fresult = (address)&(get_ijava_state()->fresult);
switch (method->result_type()) { case T_OBJECT: case T_ARRAY: {
*oop_result = JNIHandles::resolve(*(jobject*)lresult); break;
} // We use std/stfd to store the values. case T_BOOLEAN : value_result->z = (jboolean) *(unsignedlong*)lresult; break; case T_INT : value_result->i = (jint) *(long*)lresult; break; case T_CHAR : value_result->c = (jchar) *(unsignedlong*)lresult; break; case T_SHORT : value_result->s = (jshort) *(long*)lresult; break; case T_BYTE : value_result->z = (jbyte) *(long*)lresult; break; case T_LONG : value_result->j = (jlong) *(long*)lresult; break; case T_FLOAT : value_result->f = (jfloat) *(double*)fresult; break; case T_DOUBLE : value_result->d = (jdouble) *(double*)fresult; break; case T_VOID : /* Nothing to do */ break; default : ShouldNotReachHere();
}
} else {
intptr_t* tos_addr = interpreter_frame_tos_address(); switch (method->result_type()) { case T_OBJECT: case T_ARRAY: {
oop obj = *(oop*)tos_addr;
assert(Universe::is_in_heap_or_null(obj), "sanity check");
*oop_result = obj;
} case T_BOOLEAN : value_result->z = (jboolean) *(jint*)tos_addr; break; case T_BYTE : value_result->b = (jbyte) *(jint*)tos_addr; break; case T_CHAR : value_result->c = (jchar) *(jint*)tos_addr; break; case T_SHORT : value_result->s = (jshort) *(jint*)tos_addr; break; case T_INT : value_result->i = *(jint*)tos_addr; break; case T_LONG : value_result->j = *(jlong*)tos_addr; break; case T_FLOAT : value_result->f = *(jfloat*)tos_addr; break; case T_DOUBLE : value_result->d = *(jdouble*)tos_addr; break; case T_VOID : /* Nothing to do */ break; default : ShouldNotReachHere();
}
} return type;
}
#ifndef PRODUCT
void frame::describe_pd(FrameValues& values, int frame_no) { if (is_interpreted_frame()) { #define DESCRIBE_ADDRESS(name) \
values.describe(frame_no, (intptr_t*)&(get_ijava_state()->name), #name);
intptr_t *frame::initial_deoptimization_info() { // unused... but returns fp() to minimize changes introduced by 7087445 return fp();
}
#ifndef PRODUCT // This is a generic constructor which is only used by pns() in debug.cpp. // fp is dropped and gets determined by backlink.
frame::frame(void* sp, void* fp, void* pc) : frame((intptr_t*)sp, (address)pc) {} #endif
// Pointer beyond the "oldest/deepest" BasicObjectLock on stack.
BasicObjectLock* frame::interpreter_frame_monitor_end() const {
BasicObjectLock* result = (BasicObjectLock*) at(ijava_idx(monitors)); // make sure the pointer points inside the frame
assert(sp() <= (intptr_t*) result, "monitor end should be above the stack pointer");
assert((intptr_t*) result < fp(), "monitor end should be strictly below the frame pointer: result: " INTPTR_FORMAT " fp: " INTPTR_FORMAT, p2i(result), p2i(fp())); return result;
}
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.