1 /* 2 * Copyright (C) 2019 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_REFLECTIVE_VALUE_VISITOR_H_ 18 #define ART_RUNTIME_REFLECTIVE_VALUE_VISITOR_H_ 19 20 #include <android-base/logging.h> 21 22 #include <array> 23 #include <compare> 24 #include <functional> 25 #include <stack> 26 27 #include "android-base/macros.h" 28 #include "base/globals.h" 29 #include "base/locks.h" 30 #include "base/macros.h" 31 #include "base/pointer_size.h" 32 #include "base/value_object.h" 33 #include "dex/dex_file.h" 34 #include "jni.h" 35 #include "mirror/dex_cache.h" 36 #include "obj_ptr.h" 37 38 namespace art HIDDEN { 39 40 class ArtField; 41 class ArtMethod; 42 class BaseReflectiveHandleScope; 43 class Thread; 44 45 class ReflectionSourceInfo; 46 47 class ReflectiveValueVisitor : public ValueObject { 48 public: ~ReflectiveValueVisitor()49 virtual ~ReflectiveValueVisitor() {} 50 51 virtual ArtMethod* VisitMethod(ArtMethod* in, const ReflectionSourceInfo& info) 52 REQUIRES_SHARED(Locks::mutator_lock_) = 0; 53 virtual ArtField* VisitField(ArtField* in, const ReflectionSourceInfo& info) 54 REQUIRES_SHARED(Locks::mutator_lock_) = 0; 55 56 // Give it an entrypoint through operator() to interact with things that expect lambda-like things 57 template <typename T, 58 typename = typename std::enable_if<std::is_same_v<T, ArtField> || 59 std::is_same_v<T, ArtMethod>>> operator()60 T* operator()(T* t, const ReflectionSourceInfo& info) REQUIRES_SHARED(Locks::mutator_lock_) { 61 if constexpr (std::is_same_v<T, ArtField>) { 62 return VisitField(t, info); 63 } else { 64 static_assert(std::is_same_v<T, ArtMethod>, "Expected ArtField or ArtMethod"); 65 return VisitMethod(t, info); 66 } 67 } 68 }; 69 70 template <typename FieldVis, typename MethodVis> 71 class FunctionReflectiveValueVisitor : public ReflectiveValueVisitor { 72 public: FunctionReflectiveValueVisitor(FieldVis fv,MethodVis mv)73 FunctionReflectiveValueVisitor(FieldVis fv, MethodVis mv) : fv_(fv), mv_(mv) {} VisitField(ArtField * in,const ReflectionSourceInfo & info)74 ArtField* VisitField(ArtField* in, const ReflectionSourceInfo& info) override 75 REQUIRES(Locks::mutator_lock_) { 76 return fv_(in, info); 77 } VisitMethod(ArtMethod * in,const ReflectionSourceInfo & info)78 ArtMethod* VisitMethod(ArtMethod* in, const ReflectionSourceInfo& info) override 79 REQUIRES(Locks::mutator_lock_) { 80 return mv_(in, info); 81 } 82 83 private: 84 FieldVis fv_; 85 MethodVis mv_; 86 }; 87 88 enum ReflectionSourceType { 89 kSourceUnknown = 0, 90 kSourceJavaLangReflectExecutable, 91 kSourceJavaLangReflectField, 92 kSourceJavaLangInvokeMethodHandle, 93 kSourceJavaLangInvokeFieldVarHandle, 94 kSourceThreadHandleScope, 95 kSourceJniFieldId, 96 kSourceJniMethodId, 97 kSourceDexCacheResolvedMethod, 98 kSourceDexCacheResolvedField, 99 kSourceMiscInternal, 100 }; 101 EXPORT std::ostream& operator<<(std::ostream& os, ReflectionSourceType type); 102 103 class ReflectionSourceInfo : public ValueObject { 104 public: ~ReflectionSourceInfo()105 virtual ~ReflectionSourceInfo() {} 106 // Thread id 0 is for non thread roots. ReflectionSourceInfo(ReflectionSourceType type)107 explicit ReflectionSourceInfo(ReflectionSourceType type) : type_(type) {} Describe(std::ostream & os)108 virtual void Describe(std::ostream& os) const { 109 os << "Type=" << type_; 110 } 111 GetType()112 ReflectionSourceType GetType() const { 113 return type_; 114 } 115 116 private: 117 const ReflectionSourceType type_; 118 119 DISALLOW_COPY_AND_ASSIGN(ReflectionSourceInfo); 120 }; 121 inline std::ostream& operator<<(std::ostream& os, const ReflectionSourceInfo& info) { 122 info.Describe(os); 123 return os; 124 } 125 126 class EXPORT ReflectiveHandleScopeSourceInfo : public ReflectionSourceInfo { 127 public: ReflectiveHandleScopeSourceInfo(BaseReflectiveHandleScope * source)128 explicit ReflectiveHandleScopeSourceInfo(BaseReflectiveHandleScope* source) 129 : ReflectionSourceInfo(kSourceThreadHandleScope), source_(source) {} 130 131 void Describe(std::ostream& os) const override; 132 133 private: 134 BaseReflectiveHandleScope* source_; 135 }; 136 137 // TODO Maybe give this the ability to retrieve the type and ref, if it's useful. 138 class HeapReflectiveSourceInfo : public ReflectionSourceInfo { 139 public: HeapReflectiveSourceInfo(ReflectionSourceType t,mirror::Object * src)140 HeapReflectiveSourceInfo(ReflectionSourceType t, mirror::Object* src) 141 : ReflectionSourceInfo(t), src_(src) {} 142 void Describe(std::ostream& os) const override; 143 144 private: 145 ObjPtr<mirror::Object> src_; 146 }; 147 148 // TODO Maybe give this the ability to retrieve the id if it's useful. 149 template <typename T, 150 typename = typename std::enable_if_t<std::is_same_v<T, jmethodID> || 151 std::is_same_v<T, jfieldID>>> 152 class JniIdReflectiveSourceInfo : public ReflectionSourceInfo { 153 public: JniIdReflectiveSourceInfo(T id)154 explicit JniIdReflectiveSourceInfo(T id) 155 : ReflectionSourceInfo(std::is_same_v<T, jmethodID> ? kSourceJniMethodId : kSourceJniFieldId), 156 id_(id) {} 157 void Describe(std::ostream& os) const override; 158 159 private: 160 T id_; 161 }; 162 163 class DexCacheSourceInfo : public ReflectionSourceInfo { 164 public: DexCacheSourceInfo(ReflectionSourceType type,size_t index,ObjPtr<mirror::DexCache> cache)165 explicit DexCacheSourceInfo(ReflectionSourceType type, 166 size_t index, 167 ObjPtr<mirror::DexCache> cache) 168 : ReflectionSourceInfo(type), index_(index), cache_(cache) {} 169 Describe(std::ostream & os)170 void Describe(std::ostream& os) const override REQUIRES(Locks::mutator_lock_) { 171 ReflectionSourceInfo::Describe(os); 172 os << " index=" << index_ << " cache_=" << cache_.PtrUnchecked() 173 << " files=" << *cache_->GetDexFile(); 174 } 175 176 private: 177 size_t index_; 178 ObjPtr<mirror::DexCache> cache_; 179 }; 180 } // namespace art 181 182 #endif // ART_RUNTIME_REFLECTIVE_VALUE_VISITOR_H_ 183