Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/mobile
Path: blob/master/src/hotspot/cpu/zero/methodHandles_zero.cpp
40931 views
1
/*
2
* Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved.
3
* Copyright 2009, 2010, 2011 Red Hat, Inc.
4
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
5
*
6
* This code is free software; you can redistribute it and/or modify it
7
* under the terms of the GNU General Public License version 2 only, as
8
* published by the Free Software Foundation.
9
*
10
* This code is distributed in the hope that it will be useful, but WITHOUT
11
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
13
* version 2 for more details (a copy is included in the LICENSE file that
14
* accompanied this code).
15
*
16
* You should have received a copy of the GNU General Public License version
17
* 2 along with this work; if not, write to the Free Software Foundation,
18
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
19
*
20
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
21
* or visit www.oracle.com if you need additional information or have any
22
* questions.
23
*
24
*/
25
26
#include "precompiled.hpp"
27
#include "classfile/javaClasses.inline.hpp"
28
#include "interpreter/interpreter.hpp"
29
#include "interpreter/interpreterRuntime.hpp"
30
#include "interpreter/zero/zeroInterpreterGenerator.hpp"
31
#include "memory/allocation.inline.hpp"
32
#include "memory/resourceArea.hpp"
33
#include "oops/instanceKlass.inline.hpp"
34
#include "oops/klass.inline.hpp"
35
#include "oops/method.inline.hpp"
36
#include "oops/oop.inline.hpp"
37
#include "runtime/frame.inline.hpp"
38
#include "prims/methodHandles.hpp"
39
40
41
void MethodHandles::invoke_target(Method* method, TRAPS) {
42
43
JavaThread *thread = THREAD;
44
ZeroStack *stack = thread->zero_stack();
45
InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();
46
interpreterState istate = frame->interpreter_state();
47
48
// Trim back the stack to put the parameters at the top
49
stack->set_sp(istate->stack() + 1);
50
51
Interpreter::invoke_method(method, method->from_interpreted_entry(), THREAD);
52
53
// Convert the result
54
istate->set_stack(stack->sp() - 1);
55
56
}
57
58
oop MethodHandles::popFromStack(TRAPS) {
59
60
JavaThread *thread = THREAD;
61
InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();
62
interpreterState istate = frame->interpreter_state();
63
intptr_t* topOfStack = istate->stack();
64
65
oop top = STACK_OBJECT(-1);
66
MORE_STACK(-1);
67
istate->set_stack(topOfStack);
68
69
return top;
70
71
}
72
73
void MethodHandles::setup_frame_anchor(JavaThread* thread) {
74
assert(!thread->has_last_Java_frame(), "Do not need to call this otherwise");
75
76
intptr_t *sp = thread->zero_stack()->sp();
77
ZeroFrame *frame = thread->top_zero_frame();
78
while (frame) {
79
if (frame->is_interpreter_frame()) {
80
interpreterState istate = frame->as_interpreter_frame()->interpreter_state();
81
if (istate->self_link() == istate) break;
82
}
83
sp = ((intptr_t *) frame) + 1;
84
frame = frame->next();
85
}
86
87
assert(frame != NULL, "must be");
88
thread->set_last_Java_frame(frame, sp);
89
}
90
91
void MethodHandles::teardown_frame_anchor(JavaThread* thread) {
92
thread->reset_last_Java_frame();
93
}
94
95
void MethodHandles::throw_AME(Klass* rcvr, Method* interface_method, TRAPS) {
96
JavaThread* thread = THREAD;
97
bool has_last_Java_frame = thread->has_last_Java_frame();
98
if (!has_last_Java_frame) {
99
setup_frame_anchor(thread);
100
}
101
InterpreterRuntime::throw_AbstractMethodErrorVerbose(thread, rcvr, interface_method);
102
if (!has_last_Java_frame) {
103
teardown_frame_anchor(thread);
104
}
105
}
106
107
void MethodHandles::throw_NPE(TRAPS) {
108
JavaThread* thread = THREAD;
109
bool has_last_Java_frame = thread->has_last_Java_frame();
110
if (!has_last_Java_frame) {
111
setup_frame_anchor(thread);
112
}
113
InterpreterRuntime::throw_NullPointerException(thread);
114
if (!has_last_Java_frame) {
115
teardown_frame_anchor(thread);
116
}
117
}
118
119
int MethodHandles::method_handle_entry_invokeBasic(Method* method, intptr_t UNUSED, TRAPS) {
120
121
JavaThread *thread = THREAD;
122
InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();
123
interpreterState istate = frame->interpreter_state();
124
intptr_t* topOfStack = istate->stack();
125
126
// 'this' is a MethodHandle. We resolve the target method by accessing this.form.vmentry.vmtarget.
127
int numArgs = method->size_of_parameters();
128
129
oop recv = STACK_OBJECT(-numArgs);
130
if (recv == NULL) {
131
throw_NPE(THREAD);
132
return 0;
133
}
134
135
oop lform1 = java_lang_invoke_MethodHandle::form(recv); // this.form
136
oop vmEntry1 = java_lang_invoke_LambdaForm::vmentry(lform1);
137
Method* vmtarget = (Method*) java_lang_invoke_MemberName::vmtarget(vmEntry1);
138
139
invoke_target(vmtarget, THREAD);
140
141
// No deoptimized frames on the stack
142
return 0;
143
}
144
145
int MethodHandles::method_handle_entry_linkToStaticOrSpecial(Method* method, intptr_t UNUSED, TRAPS) {
146
147
// Pop appendix argument from stack. This is a MemberName which we resolve to the
148
// target method.
149
oop vmentry = popFromStack(THREAD);
150
151
Method* vmtarget = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);
152
153
invoke_target(vmtarget, THREAD);
154
155
return 0;
156
}
157
158
int MethodHandles::method_handle_entry_linkToInterface(Method* method, intptr_t UNUSED, TRAPS) {
159
JavaThread *thread = THREAD;
160
InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();
161
interpreterState istate = frame->interpreter_state();
162
163
// Pop appendix argument from stack. This is a MemberName which we resolve to the
164
// target method.
165
oop vmentry = popFromStack(THREAD);
166
intptr_t* topOfStack = istate->stack();
167
168
// Resolve target method by looking up in the receiver object's itable.
169
Klass* clazz = java_lang_Class::as_Klass(java_lang_invoke_MemberName::clazz(vmentry));
170
intptr_t vmindex = java_lang_invoke_MemberName::vmindex(vmentry);
171
Method* target = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);
172
173
int numArgs = target->size_of_parameters();
174
oop recv = STACK_OBJECT(-numArgs);
175
if (recv == NULL) {
176
throw_NPE(THREAD);
177
return 0;
178
}
179
180
InstanceKlass* klass_part = InstanceKlass::cast(recv->klass());
181
itableOffsetEntry* ki = (itableOffsetEntry*) klass_part->start_of_itable();
182
int i;
183
for ( i = 0 ; i < klass_part->itable_length() ; i++, ki++ ) {
184
if (ki->interface_klass() == clazz) break;
185
}
186
187
itableMethodEntry* im = ki->first_method_entry(recv->klass());
188
Method* vmtarget = im[vmindex].method();
189
// Check that the vmtarget entry is non-null. A null entry means
190
// that the method no longer exists (got deleted) or is private.
191
// Private class methods can never be an implementation of an
192
// interface method. In those cases, throw AME.
193
if (vmtarget != NULL) {
194
invoke_target(vmtarget, THREAD);
195
} else {
196
throw_AME(recv->klass(), target, THREAD);
197
}
198
199
return 0;
200
}
201
202
int MethodHandles::method_handle_entry_linkToVirtual(Method* method, intptr_t UNUSED, TRAPS) {
203
JavaThread *thread = THREAD;
204
205
InterpreterFrame *frame = thread->top_zero_frame()->as_interpreter_frame();
206
interpreterState istate = frame->interpreter_state();
207
208
// Pop appendix argument from stack. This is a MemberName which we resolve to the
209
// target method.
210
oop vmentry = popFromStack(THREAD);
211
intptr_t* topOfStack = istate->stack();
212
213
// Resolve target method by looking up in the receiver object's vtable.
214
intptr_t vmindex = java_lang_invoke_MemberName::vmindex(vmentry);
215
Method* target = (Method*) java_lang_invoke_MemberName::vmtarget(vmentry);
216
217
int numArgs = target->size_of_parameters();
218
oop recv = STACK_OBJECT(-numArgs);
219
if (recv == NULL) {
220
throw_NPE(THREAD);
221
return 0;
222
}
223
224
Klass* clazz = recv->klass();
225
Klass* klass_part = InstanceKlass::cast(clazz);
226
ResourceMark rm(THREAD);
227
klassVtable vtable = klass_part->vtable();
228
Method* vmtarget = vtable.method_at(vmindex);
229
230
invoke_target(vmtarget, THREAD);
231
232
return 0;
233
}
234
235
int MethodHandles::method_handle_entry_invalid(Method* method, intptr_t UNUSED, TRAPS) {
236
ShouldNotReachHere();
237
return 0;
238
}
239
240
address MethodHandles::generate_method_handle_interpreter_entry(MacroAssembler* masm,
241
vmIntrinsics::ID iid) {
242
switch (iid) {
243
case vmIntrinsics::_invokeGeneric:
244
case vmIntrinsics::_compiledLambdaForm:
245
case vmIntrinsics::_linkToNative:
246
// Perhaps surprisingly, the symbolic references visible to Java are not directly used.
247
// They are linked to Java-generated adapters via MethodHandleNatives.linkMethod.
248
// They all allow an appendix argument.
249
return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_invalid);
250
case vmIntrinsics::_invokeBasic:
251
return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_invokeBasic);
252
case vmIntrinsics::_linkToStatic:
253
case vmIntrinsics::_linkToSpecial:
254
return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToStaticOrSpecial);
255
case vmIntrinsics::_linkToInterface:
256
return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToInterface);
257
case vmIntrinsics::_linkToVirtual:
258
return ZeroInterpreterGenerator::generate_entry_impl(masm, (address) MethodHandles::method_handle_entry_linkToVirtual);
259
default:
260
ShouldNotReachHere();
261
return NULL;
262
}
263
}
264
265
#ifndef PRODUCT
266
void MethodHandles::trace_method_handle(MacroAssembler* _masm, const char* adaptername) {
267
// This is just a stub.
268
}
269
#endif //PRODUCT
270
271