blob: ddd84a167d0ee309a7a187b136e9c79902f5fb4e [file] [log] [blame]
/*
* Copyright (C) 2016 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef ART_RUNTIME_MIRROR_EMULATED_STACK_FRAME_H_
#define ART_RUNTIME_MIRROR_EMULATED_STACK_FRAME_H_
#include "dex_instruction.h"
#include "method_type.h"
#include "object.h"
#include "stack.h"
#include "string.h"
#include "utils.h"
namespace art {
struct EmulatedStackFrameOffsets;
namespace mirror {
// C++ mirror of dalvik.system.EmulatedStackFrame
class MANAGED EmulatedStackFrame : public Object {
public:
// Creates an emulated stack frame whose type is |frame_type| from
// a shadow frame.
template <bool is_range>
static mirror::EmulatedStackFrame* CreateFromShadowFrameAndArgs(
Thread* self,
Handle<mirror::MethodType> args_type,
Handle<mirror::MethodType> frame_type,
const ShadowFrame& caller_frame,
const uint32_t first_src_reg,
const uint32_t (&arg)[Instruction::kMaxVarArgRegs]) REQUIRES_SHARED(Locks::mutator_lock_);
// Writes the contents of this emulated stack frame to the |callee_frame|
// whose type is |callee_type|, starting at |first_dest_reg|.
bool WriteToShadowFrame(
Thread* self,
Handle<mirror::MethodType> callee_type,
const uint32_t first_dest_reg,
ShadowFrame* callee_frame) REQUIRES_SHARED(Locks::mutator_lock_);
// Sets |value| to the return value written to this emulated stack frame (if any).
void GetReturnValue(Thread* self, JValue* value) REQUIRES_SHARED(Locks::mutator_lock_);
// Sets the return value slot of this emulated stack frame to |value|.
void SetReturnValue(Thread* self, const JValue& value) REQUIRES_SHARED(Locks::mutator_lock_);
mirror::MethodType* GetType() REQUIRES_SHARED(Locks::mutator_lock_) {
return GetFieldObject<MethodType>(OFFSET_OF_OBJECT_MEMBER(EmulatedStackFrame, type_));
}
static void SetClass(Class* klass) REQUIRES_SHARED(Locks::mutator_lock_);
static void ResetClass() REQUIRES_SHARED(Locks::mutator_lock_);
static void VisitRoots(RootVisitor* visitor) REQUIRES_SHARED(Locks::mutator_lock_);
private:
static mirror::Class* StaticClass() REQUIRES_SHARED(Locks::mutator_lock_) {
return static_class_.Read();
}
mirror::ObjectArray<mirror::Object>* GetReferences() REQUIRES_SHARED(Locks::mutator_lock_) {
return GetFieldObject<mirror::ObjectArray<mirror::Object>>(
OFFSET_OF_OBJECT_MEMBER(EmulatedStackFrame, references_));
}
mirror::ByteArray* GetStackFrame() REQUIRES_SHARED(Locks::mutator_lock_) {
return GetFieldObject<mirror::ByteArray>(
OFFSET_OF_OBJECT_MEMBER(EmulatedStackFrame, stack_frame_));
}
static MemberOffset CallsiteTypeOffset() {
return MemberOffset(OFFSETOF_MEMBER(EmulatedStackFrame, callsite_type_));
}
static MemberOffset TypeOffset() {
return MemberOffset(OFFSETOF_MEMBER(EmulatedStackFrame, type_));
}
static MemberOffset ReferencesOffset() {
return MemberOffset(OFFSETOF_MEMBER(EmulatedStackFrame, references_));
}
static MemberOffset StackFrameOffset() {
return MemberOffset(OFFSETOF_MEMBER(EmulatedStackFrame, stack_frame_));
}
HeapReference<mirror::MethodType> callsite_type_;
HeapReference<mirror::ObjectArray<mirror::Object>> references_;
HeapReference<mirror::ByteArray> stack_frame_;
HeapReference<mirror::MethodType> type_;
static GcRoot<mirror::Class> static_class_; // dalvik.system.EmulatedStackFrame.class
friend struct art::EmulatedStackFrameOffsets; // for verifying offset information
DISALLOW_IMPLICIT_CONSTRUCTORS(EmulatedStackFrame);
};
} // namespace mirror
} // namespace art
#endif // ART_RUNTIME_MIRROR_EMULATED_STACK_FRAME_H_