blob: 24724b01abb9cd82ea1fdba36e92e895846c1175 [file] [log] [blame]
J. Duke81537792007-12-01 00:00:00 +00001/*
Mikael Gerdincc1aaab2017-03-22 15:25:21 +01002 * Copyright (c) 1997, 2017, Oracle and/or its affiliates. All rights reserved.
J. Duke81537792007-12-01 00:00:00 +00003 * 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 *
Erik Trimbleba7c1732010-05-27 19:08:38 -070019 * 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.
J. Duke81537792007-12-01 00:00:00 +000022 *
23 */
24
Stefan Karlsson8006fe82010-11-23 13:22:55 -080025#ifndef SHARE_VM_RUNTIME_JAVACALLS_HPP
26#define SHARE_VM_RUNTIME_JAVACALLS_HPP
27
28#include "memory/allocation.hpp"
Jon Masamitsu5c58d272012-09-01 13:25:18 -040029#include "oops/method.hpp"
Stefan Karlsson8006fe82010-11-23 13:22:55 -080030#include "runtime/handles.hpp"
31#include "runtime/javaFrameAnchor.hpp"
Stefan Karlsson71bcff32016-04-11 08:51:53 +020032#include "runtime/thread.hpp"
Stefan Karlsson8006fe82010-11-23 13:22:55 -080033#include "runtime/vmThread.hpp"
Goetz Lindenmaierda333992016-07-13 12:23:05 +020034#include "utilities/macros.hpp"
35
36#include CPU_HEADER(jniTypes)
Stefan Karlsson8006fe82010-11-23 13:22:55 -080037
J. Duke81537792007-12-01 00:00:00 +000038// A JavaCallWrapper is constructed before each JavaCall and destructed after the call.
39// Its purpose is to allocate/deallocate a new handle block and to save/restore the last
40// Java fp/sp. A pointer to the JavaCallWrapper is stored on the stack.
41
42class JavaCallWrapper: StackObj {
43 friend class VMStructs;
44 private:
45 JavaThread* _thread; // the thread to which this call belongs
46 JNIHandleBlock* _handles; // the saved handle block
Jon Masamitsu5c58d272012-09-01 13:25:18 -040047 Method* _callee_method; // to be able to collect arguments if entry frame is top frame
J. Duke81537792007-12-01 00:00:00 +000048 oop _receiver; // the receiver of the call (if a non-static call)
49
50 JavaFrameAnchor _anchor; // last thread anchor state that we must restore
51
52 JavaValue* _result; // result value
53
54 public:
55 // Construction/destruction
Coleen Phillimoreaf5c6ee2017-07-27 18:06:41 -040056 JavaCallWrapper(const methodHandle& callee_method, Handle receiver, JavaValue* result, TRAPS);
J. Duke81537792007-12-01 00:00:00 +000057 ~JavaCallWrapper();
58
59 // Accessors
60 JavaThread* thread() const { return _thread; }
61 JNIHandleBlock* handles() const { return _handles; }
62
63 JavaFrameAnchor* anchor(void) { return &_anchor; }
64
65 JavaValue* result() const { return _result; }
66 // GC support
Jon Masamitsu5c58d272012-09-01 13:25:18 -040067 Method* callee_method() { return _callee_method; }
J. Duke81537792007-12-01 00:00:00 +000068 oop receiver() { return _receiver; }
69 void oops_do(OopClosure* f);
70
Rickard Bäckman9a672292013-06-12 11:17:39 +020071 bool is_first_frame() const { return _anchor.last_Java_sp() == NULL; }
72
J. Duke81537792007-12-01 00:00:00 +000073};
74
75
76// Encapsulates arguments to a JavaCall (faster, safer, and more convenient than using var-args)
77class JavaCallArguments : public StackObj {
78 private:
79 enum Constants {
80 _default_size = 8 // Must be at least # of arguments in JavaCalls methods
81 };
82
Mikael Gerdincc1aaab2017-03-22 15:25:21 +010083 intptr_t _value_buffer [_default_size + 1];
84 u_char _value_state_buffer[_default_size + 1];
J. Duke81537792007-12-01 00:00:00 +000085
86 intptr_t* _value;
Mikael Gerdincc1aaab2017-03-22 15:25:21 +010087 u_char* _value_state;
J. Duke81537792007-12-01 00:00:00 +000088 int _size;
89 int _max_size;
90 bool _start_at_zero; // Support late setting of receiver
Christian Thalinger16526e02015-10-08 12:49:30 -100091 JVMCI_ONLY(nmethod* _alternative_target;) // Nmethod that should be called instead of normal target
J. Duke81537792007-12-01 00:00:00 +000092
93 void initialize() {
94 // Starts at first element to support set_receiver.
Mikael Gerdincc1aaab2017-03-22 15:25:21 +010095 _value = &_value_buffer[1];
96 _value_state = &_value_state_buffer[1];
J. Duke81537792007-12-01 00:00:00 +000097
J. Duke81537792007-12-01 00:00:00 +000098 _max_size = _default_size;
99 _size = 0;
100 _start_at_zero = false;
Christian Thalinger16526e02015-10-08 12:49:30 -1000101 JVMCI_ONLY(_alternative_target = NULL;)
J. Duke81537792007-12-01 00:00:00 +0000102 }
103
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100104 // Helper for push_oop and the like. The value argument is a
105 // "handle" that refers to an oop. We record the address of the
106 // handle rather than the designated oop. The handle is later
107 // resolved to the oop by parameters(). This delays the exposure of
108 // naked oops until it is GC-safe.
109 template<typename T>
110 inline int push_oop_impl(T handle, int size) {
111 // JNITypes::put_obj expects an oop value, so we play fast and
112 // loose with the type system. The cast from handle type to oop
113 // *must* use a C-style cast. In a product build it performs a
114 // reinterpret_cast. In a debug build (more accurately, in a
115 // CHECK_UNHANDLED_OOPS build) it performs a static_cast, invoking
116 // the debug-only oop class's conversion from void* constructor.
117 JNITypes::put_obj((oop)handle, _value, size); // Updates size.
118 return size; // Return the updated size.
119 }
120
J. Duke81537792007-12-01 00:00:00 +0000121 public:
122 JavaCallArguments() { initialize(); }
123
124 JavaCallArguments(Handle receiver) {
125 initialize();
126 push_oop(receiver);
127 }
128
129 JavaCallArguments(int max_size) {
130 if (max_size > _default_size) {
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100131 _value = NEW_RESOURCE_ARRAY(intptr_t, max_size + 1);
132 _value_state = NEW_RESOURCE_ARRAY(u_char, max_size + 1);
Christian Thalinger0211f972010-04-30 08:37:24 -0700133
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100134 // Reserve room for potential receiver in value and state
135 _value++;
136 _value_state++;
Christian Thalinger0211f972010-04-30 08:37:24 -0700137
J. Duke81537792007-12-01 00:00:00 +0000138 _max_size = max_size;
139 _size = 0;
140 _start_at_zero = false;
Christian Thalinger16526e02015-10-08 12:49:30 -1000141 JVMCI_ONLY(_alternative_target = NULL;)
J. Duke81537792007-12-01 00:00:00 +0000142 } else {
143 initialize();
144 }
145 }
146
Christian Thalinger16526e02015-10-08 12:49:30 -1000147#if INCLUDE_JVMCI
148 void set_alternative_target(nmethod* target) {
149 _alternative_target = target;
150 }
151
152 nmethod* alternative_target() {
153 return _alternative_target;
154 }
155#endif
156
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100157 // The possible values for _value_state elements.
158 enum {
159 value_state_primitive,
160 value_state_oop,
161 value_state_handle,
162 value_state_jobject,
163 value_state_limit
164 };
J. Duke81537792007-12-01 00:00:00 +0000165
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100166 inline void push_oop(Handle h) {
167 _value_state[_size] = value_state_handle;
168 _size = push_oop_impl(h.raw_value(), _size);
169 }
J. Duke81537792007-12-01 00:00:00 +0000170
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100171 inline void push_jobject(jobject h) {
172 _value_state[_size] = value_state_jobject;
173 _size = push_oop_impl(h, _size);
174 }
J. Duke81537792007-12-01 00:00:00 +0000175
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100176 inline void push_int(int i) {
177 _value_state[_size] = value_state_primitive;
178 JNITypes::put_int(i, _value, _size);
179 }
J. Duke81537792007-12-01 00:00:00 +0000180
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100181 inline void push_double(double d) {
182 _value_state[_size] = value_state_primitive;
183 _value_state[_size + 1] = value_state_primitive;
184 JNITypes::put_double(d, _value, _size);
185 }
186
187 inline void push_long(jlong l) {
188 _value_state[_size] = value_state_primitive;
189 _value_state[_size + 1] = value_state_primitive;
190 JNITypes::put_long(l, _value, _size);
191 }
192
193 inline void push_float(float f) {
194 _value_state[_size] = value_state_primitive;
195 JNITypes::put_float(f, _value, _size);
196 }
J. Duke81537792007-12-01 00:00:00 +0000197
198 // receiver
199 Handle receiver() {
200 assert(_size > 0, "must at least be one argument");
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100201 assert(_value_state[0] == value_state_handle,
202 "first argument must be an oop");
J. Duke81537792007-12-01 00:00:00 +0000203 assert(_value[0] != 0, "receiver must be not-null");
204 return Handle((oop*)_value[0], false);
205 }
206
207 void set_receiver(Handle h) {
208 assert(_start_at_zero == false, "can only be called once");
209 _start_at_zero = true;
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100210 _value_state--;
J. Duke81537792007-12-01 00:00:00 +0000211 _value--;
212 _size++;
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100213 _value_state[0] = value_state_handle;
214 push_oop_impl(h.raw_value(), 0);
J. Duke81537792007-12-01 00:00:00 +0000215 }
216
217 // Converts all Handles to oops, and returns a reference to parameter vector
218 intptr_t* parameters() ;
219 int size_of_parameters() const { return _size; }
220
221 // Verify that pushed arguments fits a given method
Mikael Gerdincc1aaab2017-03-22 15:25:21 +0100222 void verify(const methodHandle& method, BasicType return_type);
J. Duke81537792007-12-01 00:00:00 +0000223};
224
225// All calls to Java have to go via JavaCalls. Sets up the stack frame
226// and makes sure that the last_Java_frame pointers are chained correctly.
227//
228
229class JavaCalls: AllStatic {
Coleen Phillimoref1e89ea2015-10-23 16:48:38 -0400230 static void call_helper(JavaValue* result, const methodHandle& method, JavaCallArguments* args, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000231 public:
J. Duke81537792007-12-01 00:00:00 +0000232 // call_special
233 // ------------
234 // The receiver must be first oop in argument list
Coleen Phillimore4b956222017-03-15 10:25:37 -0400235 static void call_special(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000236
Coleen Phillimore4b956222017-03-15 10:25:37 -0400237 static void call_special(JavaValue* result, Handle receiver, Klass* klass, Symbol* name, Symbol* signature, TRAPS); // No args
238 static void call_special(JavaValue* result, Handle receiver, Klass* klass, Symbol* name, Symbol* signature, Handle arg1, TRAPS);
239 static void call_special(JavaValue* result, Handle receiver, Klass* klass, Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000240
241 // virtual call
242 // ------------
243
244 // The receiver must be first oop in argument list
Coleen Phillimore4b956222017-03-15 10:25:37 -0400245 static void call_virtual(JavaValue* result, Klass* spec_klass, Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000246
Coleen Phillimore4b956222017-03-15 10:25:37 -0400247 static void call_virtual(JavaValue* result, Handle receiver, Klass* spec_klass, Symbol* name, Symbol* signature, TRAPS); // No args
248 static void call_virtual(JavaValue* result, Handle receiver, Klass* spec_klass, Symbol* name, Symbol* signature, Handle arg1, TRAPS);
249 static void call_virtual(JavaValue* result, Handle receiver, Klass* spec_klass, Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000250
251 // Static call
252 // -----------
Coleen Phillimore4b956222017-03-15 10:25:37 -0400253 static void call_static(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000254
Coleen Phillimore4b956222017-03-15 10:25:37 -0400255 static void call_static(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, TRAPS);
256 static void call_static(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, Handle arg1, TRAPS);
257 static void call_static(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, Handle arg1, Handle arg2, TRAPS);
258 static void call_static(JavaValue* result, Klass* klass, Symbol* name, Symbol* signature, Handle arg1, Handle arg2, Handle arg3, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000259
260 // Low-level interface
Coleen Phillimoref1e89ea2015-10-23 16:48:38 -0400261 static void call(JavaValue* result, const methodHandle& method, JavaCallArguments* args, TRAPS);
J. Duke81537792007-12-01 00:00:00 +0000262};
Stefan Karlsson8006fe82010-11-23 13:22:55 -0800263
264#endif // SHARE_VM_RUNTIME_JAVACALLS_HPP