blob: abe999a5ac03ab03267298cd4123c48bb0c2f04f [file] [log] [blame]
Narayan Kamathafa48272016-08-03 12:46:58 +01001/*
2 * Copyright (C) 2016 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#ifndef ART_RUNTIME_MIRROR_METHOD_HANDLE_IMPL_H_
18#define ART_RUNTIME_MIRROR_METHOD_HANDLE_IMPL_H_
19
20#include "class.h"
21#include "gc_root.h"
22#include "object.h"
Narayan Kamath9823e782016-08-03 12:46:58 +010023#include "method_handles.h"
Narayan Kamathafa48272016-08-03 12:46:58 +010024#include "method_type.h"
25
26namespace art {
27
28struct MethodHandleImplOffsets;
29
30namespace mirror {
31
32// C++ mirror of java.lang.invoke.MethodHandle
33class MANAGED MethodHandle : public Object {
34 public:
Orion Hodson811bd5f2016-12-07 11:35:37 +000035 // Defines the behaviour of a given method handle. The behaviour
36 // of a handle of a given kind is identical to the dex bytecode behaviour
37 // of the equivalent instruction.
38 //
39 // NOTE: These must be kept in sync with the constants defined in
40 // java.lang.invoke.MethodHandle.
41 enum Kind {
42 kInvokeVirtual = 0,
43 kInvokeSuper,
44 kInvokeDirect,
45 kInvokeStatic,
46 kInvokeInterface,
47 kInvokeTransform,
48 kInvokeCallSiteTransform,
49 kInstanceGet,
50 kInstancePut,
51 kStaticGet,
52 kStaticPut,
53 kLastValidKind = kStaticPut,
54 kFirstAccessorKind = kInstanceGet,
55 kLastAccessorKind = kStaticPut,
56 kLastInvokeKind = kInvokeCallSiteTransform
57 };
58
59 Kind GetHandleKind() REQUIRES_SHARED(Locks::mutator_lock_) {
60 const int32_t handle_kind = GetField32(OFFSET_OF_OBJECT_MEMBER(MethodHandle, handle_kind_));
61 DCHECK(handle_kind >= 0 &&
62 handle_kind <= static_cast<int32_t>(Kind::kLastValidKind));
63 return static_cast<Kind>(handle_kind);
64 }
65
Narayan Kamathafa48272016-08-03 12:46:58 +010066 mirror::MethodType* GetMethodType() REQUIRES_SHARED(Locks::mutator_lock_) {
67 return GetFieldObject<mirror::MethodType>(OFFSET_OF_OBJECT_MEMBER(MethodHandle, method_type_));
68 }
69
Narayan Kamath0a8485e2016-11-02 18:47:11 +000070 mirror::MethodType* GetNominalType() REQUIRES_SHARED(Locks::mutator_lock_) {
71 return GetFieldObject<mirror::MethodType>(OFFSET_OF_OBJECT_MEMBER(MethodHandle, nominal_type_));
72 }
73
Orion Hodson3d617ac2016-10-19 14:00:46 +010074 ArtField* GetTargetField() REQUIRES_SHARED(Locks::mutator_lock_) {
75 return reinterpret_cast<ArtField*>(
76 GetField64(OFFSET_OF_OBJECT_MEMBER(MethodHandle, art_field_or_method_)));
77 }
78
Narayan Kamathafa48272016-08-03 12:46:58 +010079 ArtMethod* GetTargetMethod() REQUIRES_SHARED(Locks::mutator_lock_) {
80 return reinterpret_cast<ArtMethod*>(
81 GetField64(OFFSET_OF_OBJECT_MEMBER(MethodHandle, art_field_or_method_)));
82 }
83
Orion Hodsoncfa325e2016-10-13 10:25:54 +010084 static mirror::Class* StaticClass() REQUIRES_SHARED(Locks::mutator_lock_);
85
Narayan Kamathafa48272016-08-03 12:46:58 +010086 private:
Narayan Kamath0a8485e2016-11-02 18:47:11 +000087 HeapReference<mirror::MethodType> nominal_type_;
Narayan Kamathafa48272016-08-03 12:46:58 +010088 HeapReference<mirror::MethodType> method_type_;
89 uint64_t art_field_or_method_;
90 uint32_t handle_kind_;
91
92 private:
Narayan Kamath0a8485e2016-11-02 18:47:11 +000093 static MemberOffset NominalTypeOffset() {
94 return MemberOffset(OFFSETOF_MEMBER(MethodHandle, nominal_type_));
Narayan Kamathafa48272016-08-03 12:46:58 +010095 }
96 static MemberOffset MethodTypeOffset() {
97 return MemberOffset(OFFSETOF_MEMBER(MethodHandle, method_type_));
98 }
99 static MemberOffset ArtFieldOrMethodOffset() {
100 return MemberOffset(OFFSETOF_MEMBER(MethodHandle, art_field_or_method_));
101 }
102 static MemberOffset HandleKindOffset() {
103 return MemberOffset(OFFSETOF_MEMBER(MethodHandle, handle_kind_));
104 }
105
106 friend struct art::MethodHandleImplOffsets; // for verifying offset information
107 DISALLOW_IMPLICIT_CONSTRUCTORS(MethodHandle);
108};
109
110// C++ mirror of java.lang.invoke.MethodHandleImpl
111class MANAGED MethodHandleImpl : public MethodHandle {
112 public:
113 static mirror::Class* StaticClass() REQUIRES_SHARED(Locks::mutator_lock_) {
114 return static_class_.Read();
115 }
116
117 static void SetClass(Class* klass) REQUIRES_SHARED(Locks::mutator_lock_);
118 static void ResetClass() REQUIRES_SHARED(Locks::mutator_lock_);
119 static void VisitRoots(RootVisitor* visitor) REQUIRES_SHARED(Locks::mutator_lock_);
120
121 private:
122 static GcRoot<mirror::Class> static_class_; // java.lang.invoke.MethodHandleImpl.class
123
124 friend struct art::MethodHandleImplOffsets; // for verifying offset information
125 DISALLOW_IMPLICIT_CONSTRUCTORS(MethodHandleImpl);
126};
127
128} // namespace mirror
129} // namespace art
130
131#endif // ART_RUNTIME_MIRROR_METHOD_HANDLE_IMPL_H_