| /* |
| * 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_METHOD_HANDLE_IMPL_H_ |
| #define ART_RUNTIME_MIRROR_METHOD_HANDLE_IMPL_H_ |
| |
| #include "art_field.h" |
| #include "art_method.h" |
| #include "class.h" |
| #include "gc_root.h" |
| #include "object-inl.h" |
| #include "method_type.h" |
| |
| namespace art { |
| |
| struct MethodHandleOffsets; |
| struct MethodHandleImplOffsets; |
| |
| namespace mirror { |
| |
| // C++ mirror of java.lang.invoke.MethodHandle |
| class MANAGED MethodHandle : public Object { |
| public: |
| // Defines the behaviour of a given method handle. The behaviour |
| // of a handle of a given kind is identical to the dex bytecode behaviour |
| // of the equivalent instruction. |
| // |
| // NOTE: These must be kept in sync with the constants defined in |
| // java.lang.invoke.MethodHandle. |
| enum Kind { |
| kInvokeVirtual = 0, |
| kInvokeSuper, |
| kInvokeDirect, |
| kInvokeStatic, |
| kInvokeInterface, |
| kInvokeTransform, |
| kInvokeCallSiteTransform, |
| kInstanceGet, |
| kInstancePut, |
| kStaticGet, |
| kStaticPut, |
| kLastValidKind = kStaticPut, |
| kFirstAccessorKind = kInstanceGet, |
| kLastAccessorKind = kStaticPut, |
| kLastInvokeKind = kInvokeCallSiteTransform |
| }; |
| |
| Kind GetHandleKind() REQUIRES_SHARED(Locks::mutator_lock_) { |
| const int32_t handle_kind = GetField32(OFFSET_OF_OBJECT_MEMBER(MethodHandle, handle_kind_)); |
| DCHECK(handle_kind >= 0 && |
| handle_kind <= static_cast<int32_t>(Kind::kLastValidKind)); |
| return static_cast<Kind>(handle_kind); |
| } |
| |
| mirror::MethodType* GetMethodType() REQUIRES_SHARED(Locks::mutator_lock_) { |
| return GetFieldObject<mirror::MethodType>(OFFSET_OF_OBJECT_MEMBER(MethodHandle, method_type_)); |
| } |
| |
| mirror::MethodType* GetNominalType() REQUIRES_SHARED(Locks::mutator_lock_) { |
| return GetFieldObject<mirror::MethodType>(OFFSET_OF_OBJECT_MEMBER(MethodHandle, nominal_type_)); |
| } |
| |
| ArtField* GetTargetField() REQUIRES_SHARED(Locks::mutator_lock_) { |
| return reinterpret_cast<ArtField*>( |
| GetField64(OFFSET_OF_OBJECT_MEMBER(MethodHandle, art_field_or_method_))); |
| } |
| |
| ArtMethod* GetTargetMethod() REQUIRES_SHARED(Locks::mutator_lock_) { |
| return reinterpret_cast<ArtMethod*>( |
| GetField64(OFFSET_OF_OBJECT_MEMBER(MethodHandle, art_field_or_method_))); |
| } |
| |
| ObjPtr<mirror::Class> GetTargetClass() REQUIRES_SHARED(Locks::mutator_lock_) { |
| Kind kind = GetHandleKind(); |
| return (kind <= kLastValidKind) ? |
| GetTargetMethod()->GetDeclaringClass() : GetTargetField()->GetDeclaringClass(); |
| } |
| |
| static mirror::Class* StaticClass() REQUIRES_SHARED(Locks::mutator_lock_); |
| |
| protected: |
| void Initialize(uintptr_t art_field_or_method, Kind kind, Handle<MethodType> method_type) |
| REQUIRES_SHARED(Locks::mutator_lock_); |
| |
| private: |
| HeapReference<mirror::MethodHandle> cached_spread_invoker_; |
| HeapReference<mirror::MethodType> nominal_type_; |
| HeapReference<mirror::MethodType> method_type_; |
| uint32_t handle_kind_; |
| uint64_t art_field_or_method_; |
| |
| private: |
| static MemberOffset CachedSpreadInvokerOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandle, cached_spread_invoker_)); |
| } |
| static MemberOffset NominalTypeOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandle, nominal_type_)); |
| } |
| static MemberOffset MethodTypeOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandle, method_type_)); |
| } |
| static MemberOffset ArtFieldOrMethodOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandle, art_field_or_method_)); |
| } |
| static MemberOffset HandleKindOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandle, handle_kind_)); |
| } |
| |
| friend struct art::MethodHandleOffsets; // for verifying offset information |
| DISALLOW_IMPLICIT_CONSTRUCTORS(MethodHandle); |
| }; |
| |
| // C++ mirror of java.lang.invoke.MethodHandleImpl |
| class MANAGED MethodHandleImpl : public MethodHandle { |
| public: |
| static mirror::MethodHandleImpl* Create(Thread* const self, |
| uintptr_t art_field_or_method, |
| MethodHandle::Kind kind, |
| Handle<MethodType> method_type) |
| REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(!Roles::uninterruptible_); |
| |
| static mirror::Class* StaticClass() REQUIRES_SHARED(Locks::mutator_lock_) { |
| return static_class_.Read(); |
| } |
| |
| 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 MemberOffset InfoOffset() { |
| return MemberOffset(OFFSETOF_MEMBER(MethodHandleImpl, info_)); |
| } |
| |
| HeapReference<mirror::Object> info_; // Unused by the runtime. |
| static GcRoot<mirror::Class> static_class_; // java.lang.invoke.MethodHandleImpl.class |
| |
| friend struct art::MethodHandleImplOffsets; // for verifying offset information |
| DISALLOW_IMPLICIT_CONSTRUCTORS(MethodHandleImpl); |
| }; |
| |
| } // namespace mirror |
| } // namespace art |
| |
| #endif // ART_RUNTIME_MIRROR_METHOD_HANDLE_IMPL_H_ |