Path: blob/master/src/hotspot/cpu/zero/methodHandles_zero.cpp
40931 views
/*1* Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved.2* Copyright 2009, 2010, 2011 Red Hat, Inc.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 it6* under the terms of the GNU General Public License version 2 only, as7* published by the Free Software Foundation.8*9* This code is distributed in the hope that it will be useful, but WITHOUT10* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or11* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License12* version 2 for more details (a copy is included in the LICENSE file that13* accompanied this code).14*15* You should have received a copy of the GNU General Public License version16* 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 USA20* or visit www.oracle.com if you need additional information or have any21* questions.22*23*/2425#include "precompiled.hpp"26#include "classfile/javaClasses.inline.hpp"27#include "interpreter/interpreter.hpp"28#include "interpreter/interpreterRuntime.hpp"29#include "interpreter/zero/zeroInterpreterGenerator.hpp"30#include "memory/allocation.inline.hpp"31#include "memory/resourceArea.hpp"32#include "oops/instanceKlass.inline.hpp"33#include "oops/klass.inline.hpp"34#include "oops/method.inline.hpp"35#include "oops/oop.inline.hpp"36#include "runtime/frame.inline.hpp"37#include "prims/methodHandles.hpp"383940void MethodHandles::invoke_target(Method* method, TRAPS) {4142JavaThread *thread = THREAD;43ZeroStack *stack = thread->zero_stack();44InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();45interpreterState istate = frame->interpreter_state();4647// Trim back the stack to put the parameters at the top48stack->set_sp(istate->stack() + 1);4950Interpreter::invoke_method(method, method->from_interpreted_entry(), THREAD);5152// Convert the result53istate->set_stack(stack->sp() - 1);5455}5657oop MethodHandles::popFromStack(TRAPS) {5859JavaThread *thread = THREAD;60InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();61interpreterState istate = frame->interpreter_state();62intptr_t* topOfStack = istate->stack();6364oop top = STACK_OBJECT(-1);65MORE_STACK(-1);66istate->set_stack(topOfStack);6768return top;6970}7172void MethodHandles::setup_frame_anchor(JavaThread* thread) {73assert(!thread->has_last_Java_frame(), "Do not need to call this otherwise");7475intptr_t *sp = thread->zero_stack()->sp();76ZeroFrame *frame = thread->top_zero_frame();77while (frame) {78if (frame->is_interpreter_frame()) {79interpreterState istate = frame->as_interpreter_frame()->interpreter_state();80if (istate->self_link() == istate) break;81}82sp = ((intptr_t *) frame) + 1;83frame = frame->next();84}8586assert(frame != NULL, "must be");87thread->set_last_Java_frame(frame, sp);88}8990void MethodHandles::teardown_frame_anchor(JavaThread* thread) {91thread->reset_last_Java_frame();92}9394void MethodHandles::throw_AME(Klass* rcvr, Method* interface_method, TRAPS) {95JavaThread* thread = THREAD;96bool has_last_Java_frame = thread->has_last_Java_frame();97if (!has_last_Java_frame) {98setup_frame_anchor(thread);99}100InterpreterRuntime::throw_AbstractMethodErrorVerbose(thread, rcvr, interface_method);101if (!has_last_Java_frame) {102teardown_frame_anchor(thread);103}104}105106void MethodHandles::throw_NPE(TRAPS) {107JavaThread* thread = THREAD;108bool has_last_Java_frame = thread->has_last_Java_frame();109if (!has_last_Java_frame) {110setup_frame_anchor(thread);111}112InterpreterRuntime::throw_NullPointerException(thread);113if (!has_last_Java_frame) {114teardown_frame_anchor(thread);115}116}117118int MethodHandles::method_handle_entry_invokeBasic(Method* method, intptr_t UNUSED, TRAPS) {119120JavaThread *thread = THREAD;121InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();122interpreterState istate = frame->interpreter_state();123intptr_t* topOfStack = istate->stack();124125// 'this' is a MethodHandle. We resolve the target method by accessing this.form.vmentry.vmtarget.126int numArgs = method->size_of_parameters();127128oop recv = STACK_OBJECT(-numArgs);129if (recv == NULL) {130throw_NPE(THREAD);131return 0;132}133134oop lform1 = java_lang_invoke_MethodHandle::form(recv); // this.form135oop vmEntry1 = java_lang_invoke_LambdaForm::vmentry(lform1);136Method* vmtarget = (Method*) java_lang_invoke_MemberName::vmtarget(vmEntry1);137138invoke_target(vmtarget, THREAD);139140// No deoptimized frames on the stack141return 0;142}143144int MethodHandles::method_handle_entry_linkToStaticOrSpecial(Method* method, intptr_t UNUSED, TRAPS) {145146// Pop appendix argument from stack. This is a MemberName which we resolve to the147// target method.148oop vmentry = popFromStack(THREAD);149150Method* vmtarget = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);151152invoke_target(vmtarget, THREAD);153154return 0;155}156157int MethodHandles::method_handle_entry_linkToInterface(Method* method, intptr_t UNUSED, TRAPS) {158JavaThread *thread = THREAD;159InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();160interpreterState istate = frame->interpreter_state();161162// Pop appendix argument from stack. This is a MemberName which we resolve to the163// target method.164oop vmentry = popFromStack(THREAD);165intptr_t* topOfStack = istate->stack();166167// Resolve target method by looking up in the receiver object's itable.168Klass* clazz = java_lang_Class::as_Klass(java_lang_invoke_MemberName::clazz(vmentry));169intptr_t vmindex = java_lang_invoke_MemberName::vmindex(vmentry);170Method* target = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);171172int numArgs = target->size_of_parameters();173oop recv = STACK_OBJECT(-numArgs);174if (recv == NULL) {175throw_NPE(THREAD);176return 0;177}178179InstanceKlass* klass_part = InstanceKlass::cast(recv->klass());180itableOffsetEntry* ki = (itableOffsetEntry*) klass_part->start_of_itable();181int i;182for ( i = 0 ; i < klass_part->itable_length() ; i++, ki++ ) {183if (ki->interface_klass() == clazz) break;184}185186itableMethodEntry* im = ki->first_method_entry(recv->klass());187Method* vmtarget = im[vmindex].method();188// Check that the vmtarget entry is non-null. A null entry means189// that the method no longer exists (got deleted) or is private.190// Private class methods can never be an implementation of an191// interface method. In those cases, throw AME.192if (vmtarget != NULL) {193invoke_target(vmtarget, THREAD);194} else {195throw_AME(recv->klass(), target, THREAD);196}197198return 0;199}200201int MethodHandles::method_handle_entry_linkToVirtual(Method* method, intptr_t UNUSED, TRAPS) {202JavaThread *thread = THREAD;203204InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();205interpreterState istate = frame->interpreter_state();206207// Pop appendix argument from stack. This is a MemberName which we resolve to the208// target method.209oop vmentry = popFromStack(THREAD);210intptr_t* topOfStack = istate->stack();211212// Resolve target method by looking up in the receiver object's vtable.213intptr_t vmindex = java_lang_invoke_MemberName::vmindex(vmentry);214Method* target = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);215216int numArgs = target->size_of_parameters();217oop recv = STACK_OBJECT(-numArgs);218if (recv == NULL) {219throw_NPE(THREAD);220return 0;221}222223Klass* clazz = recv->klass();224Klass* klass_part = InstanceKlass::cast(clazz);225ResourceMark rm(THREAD);226klassVtable vtable = klass_part->vtable();227Method* vmtarget = vtable.method_at(vmindex);228229invoke_target(vmtarget, THREAD);230231return 0;232}233234int MethodHandles::method_handle_entry_invalid(Method* method, intptr_t UNUSED, TRAPS) {235ShouldNotReachHere();236return 0;237}238239address MethodHandles::generate_method_handle_interpreter_entry(MacroAssembler* masm,240vmIntrinsics::ID iid) {241switch (iid) {242case vmIntrinsics::_invokeGeneric:243case vmIntrinsics::_compiledLambdaForm:244case vmIntrinsics::_linkToNative:245// Perhaps surprisingly, the symbolic references visible to Java are not directly used.246// They are linked to Java-generated adapters via MethodHandleNatives.linkMethod.247// They all allow an appendix argument.248return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_invalid);249case vmIntrinsics::_invokeBasic:250return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_invokeBasic);251case vmIntrinsics::_linkToStatic:252case vmIntrinsics::_linkToSpecial:253return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToStaticOrSpecial);254case vmIntrinsics::_linkToInterface:255return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToInterface);256case vmIntrinsics::_linkToVirtual:257return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToVirtual);258default:259ShouldNotReachHere();260return NULL;261}262}263264#ifndef PRODUCT265void MethodHandles::trace_method_handle(MacroAssembler* _masm, const char* adaptername) {266// This is just a stub.267}268#endif //PRODUCT269270271