blob: 22f066fb0a08fc78f2b7226ea76ff86dc2c24b26 [file] [log] [blame]
The Android Open Source Projectf6c38712009-03-03 19:28:47 -08001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16/*
17 * Stack frames, and uses thereof.
18 */
19#ifndef _DALVIK_INTERP_STACK
20#define _DALVIK_INTERP_STACK
21
22#include "jni.h"
23#include <stdarg.h>
24
25
26/*
27Stack layout
28
29In what follows, the "top" of the stack is at a low position in memory,
30and the "bottom" of the stack is in a high position (put more simply,
31they grow downward). They may be merged with the native stack at a
32later date. The interpreter assumes that they have a fixed size,
33determined when the thread is created.
34
35Dalvik's registers (of which there can be up to 64K) map to the "ins"
36(method arguments) and "locals" (local variables). The "outs" (arguments
37to called methods) are specified by the "invoke" operand. The return
38value, which is passed through the interpreter rather than on the stack,
39is retrieved with a "move-result" instruction.
40
41 Low addresses (0x00000000)
42
43 +- - - - - - - - -+
44 - out0 -
45 +-----------------+ <-- stack ptr (top of stack)
46 + VM-specific +
47 + internal goop +
48 +-----------------+ <-- curFrame: FP for cur function
49 + v0 == local0 +
50+-----------------+ +-----------------+
51+ out0 + + v1 == in0 +
52+-----------------+ +-----------------+
53+ out1 + + v2 == in1 +
54+-----------------+ +-----------------+
55+ VM-specific +
56+ internal goop +
57+-----------------+ <-- frame ptr (FP) for previous function
58+ v0 == local0 +
59+-----------------+
60+ v1 == local1 +
61+-----------------+
62+ v2 == in0 +
63+-----------------+
64+ v3 == in1 +
65+-----------------+
66+ v4 == in2 +
67+-----------------+
68- -
69- -
70- -
71+-----------------+ <-- interpStackStart
72
73 High addresses (0xffffffff)
74
75Note the "ins" and "outs" overlap -- values pushed into the "outs" area
76become the parameters to the called method. The VM guarantees that there
77will be enough room for all possible "outs" on the stack before calling
78into a method.
79
80All "V registers" are 32 bits, and all stack entries are 32-bit aligned.
81Registers are accessed as a positive offset from the frame pointer,
82e.g. register v2 is fp[2]. 64-bit quantities are stored in two adjacent
83registers, addressed by the lower-numbered register, and are in host order.
8464-bit quantities do not need to start in an even-numbered register.
85
86We push two stack frames on when calling an interpreted or native method
87directly from the VM (e.g. invoking <clinit> or via reflection "invoke()").
88The first is a "break" frame, which allows us to tell when a call return or
89exception unroll has reached the VM call site. Without the break frame the
90stack might look like an uninterrupted series of interpreted method calls.
91The second frame is for the method itself.
92
93The "break" frame is used as an alternative to adding additional fields
94to the StackSaveArea struct itself. They are recognized by having a
95NULL method pointer.
96
97When calling a native method from interpreted code, the stack setup is
98essentially identical to calling an interpreted method. Because it's a
99native method, though, there are never any "locals" or "outs".
100
101For native calls into JNI, we want to store a table of local references
102on the stack. The GC needs to scan them while the native code is running,
103and we want to trivially discard them when the method returns. See JNI.c
104for a discussion of how this is managed. In particular note that it is
105possible to push additional call frames on without calling a method.
106*/
107
108
109struct StackSaveArea;
110typedef struct StackSaveArea StackSaveArea;
111
112//#define PAD_SAVE_AREA /* help debug stack trampling */
113
114/*
115 * The VM-specific internal goop.
116 *
117 * The idea is to mimic a typical native stack frame, with copies of the
118 * saved PC and FP. At some point we'd like to have interpreted and
119 * native code share the same stack, though this makes portability harder.
120 */
121struct StackSaveArea {
122#ifdef PAD_SAVE_AREA
123 u4 pad0, pad1, pad2;
124#endif
125
126#ifdef EASY_GDB
127 /* make it easier to trek through stack frames in GDB */
128 StackSaveArea* prevSave;
129#endif
130
131 /* saved frame pointer for previous frame, or NULL if this is at bottom */
132 void* prevFrame;
133
134 /* saved program counter (from method in caller's frame) */
135 const u2* savedPc;
136
137 /* pointer to method we're *currently* executing; handy for exceptions */
138 const Method* method;
139
140 union {
Andy McFaddend5ab7262009-08-25 07:19:34 -0700141 /* for JNI native methods: bottom of local reference segment */
142#ifdef USE_INDIRECT_REF
143 u4 localRefCookie;
144#else
145 Object** localRefCookie;
146#endif
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800147
148 /* for interpreted methods: saved current PC, for exception stack
149 * traces and debugger traces */
150 const u2* currentPc;
151 } xtra;
152
Ben Chengba4fc8b2009-06-01 13:00:29 -0700153 /* Native return pointer for JIT, or 0 if interpreted */
154 const u2* returnAddr;
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800155#ifdef PAD_SAVE_AREA
156 u4 pad3, pad4, pad5;
157#endif
158};
159
160/* move between the stack save area and the frame pointer */
161#define SAVEAREA_FROM_FP(_fp) ((StackSaveArea*)(_fp) -1)
162#define FP_FROM_SAVEAREA(_save) ((void*) ((StackSaveArea*)(_save) +1))
163
164/* when calling a function, get a pointer to outs[0] */
165#define OUTS_FROM_FP(_fp, _argCount) \
166 ((u4*) ((u1*)SAVEAREA_FROM_FP(_fp) - sizeof(u4) * (_argCount)))
167
168/* reserve this many bytes for handling StackOverflowError */
169#define STACK_OVERFLOW_RESERVE 512
170
171/*
172 * Determine if the frame pointer points to a "break frame".
173 */
174INLINE bool dvmIsBreakFrame(const u4* fp)
175{
176 return SAVEAREA_FROM_FP(fp)->method == NULL;
177}
178
179/*
180 * Initialize the interp stack (call this after allocating storage and
181 * setting thread->interpStackStart).
182 */
183bool dvmInitInterpStack(Thread* thread, int stackSize);
184
185/*
186 * Push a native method frame directly onto the stack. Used to push the
187 * "fake" native frames at the top of each thread stack.
188 */
189bool dvmPushJNIFrame(Thread* thread, const Method* method);
190
191/*
192 * JNI local frame management.
193 */
194bool dvmPushLocalFrame(Thread* thread, const Method* method);
195bool dvmPopLocalFrame(Thread* thread);
196
197/*
Andy McFaddend5ab7262009-08-25 07:19:34 -0700198 * Call an interpreted method from native code. If this is being called
199 * from a JNI function, references in the argument list will be converted
200 * back to pointers.
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800201 *
202 * "obj" should be NULL for "direct" methods.
203 */
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800204void dvmCallMethod(Thread* self, const Method* method, Object* obj,
205 JValue* pResult, ...);
Andy McFaddend5ab7262009-08-25 07:19:34 -0700206void dvmCallMethodV(Thread* self, const Method* method, Object* obj,
207 bool fromJni, JValue* pResult, va_list args);
208void dvmCallMethodA(Thread* self, const Method* method, Object* obj,
209 bool fromJni, JValue* pResult, const jvalue* args);
The Android Open Source Projectf6c38712009-03-03 19:28:47 -0800210
211/*
212 * Invoke a method, using the specified arguments and return type, through
213 * a reflection interface.
214 *
215 * Deals with boxing/unboxing primitives and performs widening conversions.
216 *
217 * "obj" should be null for a static method.
218 *
219 * "params" and "returnType" come from the Method object, so we don't have
220 * to re-generate them from the method signature. "returnType" should be
221 * NULL if we're invoking a constructor.
222 */
223Object* dvmInvokeMethod(Object* invokeObj, const Method* meth,
224 ArrayObject* argList, ArrayObject* params, ClassObject* returnType,
225 bool noAccessCheck);
226
227/*
228 * Determine the source file line number, given the program counter offset
229 * into the specified method. Returns -2 for native methods, -1 if no
230 * match was found.
231 */
232int dvmLineNumFromPC(const Method* method, u4 relPc);
233
234/*
235 * Given a frame pointer, compute the current call depth. The value can be
236 * "exact" (a count of non-break frames) or "vague" (just subtracting
237 * pointers to give relative values).
238 */
239int dvmComputeExactFrameDepth(const void* fp);
240int dvmComputeVagueFrameDepth(Thread* thread, const void* fp);
241
242/*
243 * Get the frame pointer for the caller's stack frame.
244 */
245void* dvmGetCallerFP(const void* curFrame);
246
247/*
248 * Get the class of the method that called us.
249 */
250ClassObject* dvmGetCallerClass(const void* curFrame);
251
252/*
253 * Get the caller's caller's class. Pass in the current fp.
254 *
255 * This is used by e.g. java.lang.Class, which wants to know about the
256 * class loader of the method that called it.
257 */
258ClassObject* dvmGetCaller2Class(const void* curFrame);
259
260/*
261 * Get the caller's caller's caller's class. Pass in the current fp.
262 *
263 * This is used by e.g. java.lang.Class, which wants to know about the
264 * class loader of the method that called it.
265 */
266ClassObject* dvmGetCaller3Class(const void* curFrame);
267
268/*
269 * Allocate and fill an array of method pointers representing the current
270 * stack trace (element 0 is current frame).
271 */
272bool dvmCreateStackTraceArray(const void* fp, const Method*** pArray,
273 int* pLength);
274
275/*
276 * Common handling for stack overflow.
277 */
278void dvmHandleStackOverflow(Thread* self);
279void dvmCleanupStackOverflow(Thread* self);
280
281/* debugging; dvmDumpThread() is probably a better starting point */
282void dvmDumpThreadStack(const DebugOutputTarget* target, Thread* thread);
283void dvmDumpRunningThreadStack(const DebugOutputTarget* target, Thread* thread);
284
285#endif /*_DALVIK_INTERP_STACK*/