1 /*
2 * Copyright (C) 2018 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_LIBDEXFILE_DEX_CLASS_ACCESSOR_INL_H_
18 #define ART_LIBDEXFILE_DEX_CLASS_ACCESSOR_INL_H_
19
20 #include "class_accessor.h"
21
22 #include "base/hiddenapi_flags.h"
23 #include "base/leb128.h"
24 #include "base/utils.h"
25 #include "class_iterator.h"
26 #include "code_item_accessors-inl.h"
27 #include "dex_file.h"
28 #include "method_reference.h"
29
30 namespace art {
31
ClassAccessor(const ClassIteratorData & data)32 inline ClassAccessor::ClassAccessor(const ClassIteratorData& data)
33 : ClassAccessor(data.dex_file_, data.class_def_idx_) {}
34
ClassAccessor(const DexFile & dex_file,const dex::ClassDef & class_def,bool parse_hiddenapi_class_data)35 inline ClassAccessor::ClassAccessor(const DexFile& dex_file,
36 const dex::ClassDef& class_def,
37 bool parse_hiddenapi_class_data)
38 : ClassAccessor(dex_file,
39 dex_file.GetClassData(class_def),
40 dex_file.GetIndexForClassDef(class_def),
41 parse_hiddenapi_class_data) {}
42
ClassAccessor(const DexFile & dex_file,uint32_t class_def_index)43 inline ClassAccessor::ClassAccessor(const DexFile& dex_file, uint32_t class_def_index)
44 : ClassAccessor(dex_file, dex_file.GetClassDef(class_def_index)) {}
45
ClassAccessor(const DexFile & dex_file,const uint8_t * class_data,uint32_t class_def_index,bool parse_hiddenapi_class_data)46 inline ClassAccessor::ClassAccessor(const DexFile& dex_file,
47 const uint8_t* class_data,
48 uint32_t class_def_index,
49 bool parse_hiddenapi_class_data)
50 : dex_file_(dex_file),
51 class_def_index_(class_def_index),
52 ptr_pos_(class_data),
53 hiddenapi_ptr_pos_(nullptr),
54 num_static_fields_(ptr_pos_ != nullptr ? DecodeUnsignedLeb128(&ptr_pos_) : 0u),
55 num_instance_fields_(ptr_pos_ != nullptr ? DecodeUnsignedLeb128(&ptr_pos_) : 0u),
56 num_direct_methods_(ptr_pos_ != nullptr ? DecodeUnsignedLeb128(&ptr_pos_) : 0u),
57 num_virtual_methods_(ptr_pos_ != nullptr ? DecodeUnsignedLeb128(&ptr_pos_) : 0u) {
58 if (parse_hiddenapi_class_data && class_def_index != DexFile::kDexNoIndex32) {
59 const dex::HiddenapiClassData* hiddenapi_class_data = dex_file.GetHiddenapiClassData();
60 if (hiddenapi_class_data != nullptr) {
61 hiddenapi_ptr_pos_ = hiddenapi_class_data->GetFlagsPointer(class_def_index);
62 }
63 }
64 }
65
Read()66 inline void ClassAccessor::Method::Read() {
67 index_ += DecodeUnsignedLeb128(&ptr_pos_);
68 access_flags_ = DecodeUnsignedLeb128(&ptr_pos_);
69 code_off_ = DecodeUnsignedLeb128(&ptr_pos_);
70 if (hiddenapi_ptr_pos_ != nullptr) {
71 hiddenapi_flags_ = DecodeUnsignedLeb128(&hiddenapi_ptr_pos_);
72 DCHECK(hiddenapi::ApiList(hiddenapi_flags_).IsValid());
73 }
74 }
75
GetReference()76 inline MethodReference ClassAccessor::Method::GetReference() const {
77 return MethodReference(&dex_file_, GetIndex());
78 }
79
80
Read()81 inline void ClassAccessor::Field::Read() {
82 index_ += DecodeUnsignedLeb128(&ptr_pos_);
83 access_flags_ = DecodeUnsignedLeb128(&ptr_pos_);
84 if (hiddenapi_ptr_pos_ != nullptr) {
85 hiddenapi_flags_ = DecodeUnsignedLeb128(&hiddenapi_ptr_pos_);
86 DCHECK(hiddenapi::ApiList(hiddenapi_flags_).IsValid());
87 }
88 }
89
90 template <typename DataType, typename Visitor>
VisitMembers(size_t count,const Visitor & visitor,DataType * data)91 inline void ClassAccessor::VisitMembers(size_t count,
92 const Visitor& visitor,
93 DataType* data) const {
94 DCHECK(data != nullptr);
95 for ( ; count != 0; --count) {
96 data->Read();
97 visitor(*data);
98 }
99 }
100
101 template <typename StaticFieldVisitor,
102 typename InstanceFieldVisitor,
103 typename DirectMethodVisitor,
104 typename VirtualMethodVisitor>
VisitFieldsAndMethods(const StaticFieldVisitor & static_field_visitor,const InstanceFieldVisitor & instance_field_visitor,const DirectMethodVisitor & direct_method_visitor,const VirtualMethodVisitor & virtual_method_visitor)105 inline void ClassAccessor::VisitFieldsAndMethods(
106 const StaticFieldVisitor& static_field_visitor,
107 const InstanceFieldVisitor& instance_field_visitor,
108 const DirectMethodVisitor& direct_method_visitor,
109 const VirtualMethodVisitor& virtual_method_visitor) const {
110 Field field(dex_file_, ptr_pos_, hiddenapi_ptr_pos_);
111 VisitMembers(num_static_fields_, static_field_visitor, &field);
112 field.NextSection();
113 VisitMembers(num_instance_fields_, instance_field_visitor, &field);
114
115 Method method(dex_file_, field.ptr_pos_, field.hiddenapi_ptr_pos_, /*is_static_or_direct*/ true);
116 VisitMembers(num_direct_methods_, direct_method_visitor, &method);
117 method.NextSection();
118 VisitMembers(num_virtual_methods_, virtual_method_visitor, &method);
119 }
120
121 template <typename DirectMethodVisitor,
122 typename VirtualMethodVisitor>
VisitMethods(const DirectMethodVisitor & direct_method_visitor,const VirtualMethodVisitor & virtual_method_visitor)123 inline void ClassAccessor::VisitMethods(const DirectMethodVisitor& direct_method_visitor,
124 const VirtualMethodVisitor& virtual_method_visitor) const {
125 VisitFieldsAndMethods(VoidFunctor(),
126 VoidFunctor(),
127 direct_method_visitor,
128 virtual_method_visitor);
129 }
130
131 template <typename StaticFieldVisitor,
132 typename InstanceFieldVisitor>
VisitFields(const StaticFieldVisitor & static_field_visitor,const InstanceFieldVisitor & instance_field_visitor)133 inline void ClassAccessor::VisitFields(const StaticFieldVisitor& static_field_visitor,
134 const InstanceFieldVisitor& instance_field_visitor) const {
135 VisitFieldsAndMethods(static_field_visitor,
136 instance_field_visitor,
137 VoidFunctor(),
138 VoidFunctor());
139 }
140
GetCodeItem(const Method & method)141 inline const dex::CodeItem* ClassAccessor::GetCodeItem(const Method& method) const {
142 return dex_file_.GetCodeItem(method.GetCodeItemOffset());
143 }
144
GetInstructions()145 inline CodeItemInstructionAccessor ClassAccessor::Method::GetInstructions() const {
146 return CodeItemInstructionAccessor(dex_file_, dex_file_.GetCodeItem(GetCodeItemOffset()));
147 }
148
GetInstructionsAndData()149 inline CodeItemDataAccessor ClassAccessor::Method::GetInstructionsAndData() const {
150 return CodeItemDataAccessor(dex_file_, dex_file_.GetCodeItem(GetCodeItemOffset()));
151 }
152
GetDescriptor()153 inline const char* ClassAccessor::GetDescriptor() const {
154 return dex_file_.GetTypeDescriptor(GetClassIdx());
155 }
156
GetDescriptorView()157 inline std::string_view ClassAccessor::GetDescriptorView() const {
158 return dex_file_.GetTypeDescriptorView(GetClassIdx());
159 }
160
GetCodeItem()161 inline const dex::CodeItem* ClassAccessor::Method::GetCodeItem() const {
162 return dex_file_.GetCodeItem(code_off_);
163 }
164
165 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Field>>
GetFieldsInternal(size_t count)166 ClassAccessor::GetFieldsInternal(size_t count) const {
167 return {
168 DataIterator<Field>(dex_file_,
169 0u,
170 num_static_fields_,
171 count,
172 ptr_pos_,
173 hiddenapi_ptr_pos_),
174 DataIterator<Field>(dex_file_,
175 count,
176 num_static_fields_,
177 count,
178 // The following pointers are bogus but unused in the `end` iterator.
179 ptr_pos_,
180 hiddenapi_ptr_pos_) };
181 }
182
183 // Return an iteration range for the first <count> methods.
184 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Method>>
GetMethodsInternal(size_t count)185 ClassAccessor::GetMethodsInternal(size_t count) const {
186 // Skip over the fields.
187 Field field(dex_file_, ptr_pos_, hiddenapi_ptr_pos_);
188 VisitMembers(NumFields(), VoidFunctor(), &field);
189 // Return the iterator pair.
190 return {
191 DataIterator<Method>(dex_file_,
192 0u,
193 num_direct_methods_,
194 count,
195 field.ptr_pos_,
196 field.hiddenapi_ptr_pos_),
197 DataIterator<Method>(dex_file_,
198 count,
199 num_direct_methods_,
200 count,
201 // The following pointers are bogus but unused in the `end` iterator.
202 field.ptr_pos_,
203 field.hiddenapi_ptr_pos_) };
204 }
205
GetFields()206 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Field>> ClassAccessor::GetFields()
207 const {
208 return GetFieldsInternal(num_static_fields_ + num_instance_fields_);
209 }
210
211 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Field>>
GetStaticFields()212 ClassAccessor::GetStaticFields() const {
213 return GetFieldsInternal(num_static_fields_);
214 }
215
216
217 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Field>>
GetInstanceFields()218 ClassAccessor::GetInstanceFields() const {
219 IterationRange<ClassAccessor::DataIterator<ClassAccessor::Field>> fields = GetFields();
220 // Skip the static fields.
221 return { std::next(fields.begin(), NumStaticFields()), fields.end() };
222 }
223
224 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Method>>
GetMethods()225 ClassAccessor::GetMethods() const {
226 return GetMethodsInternal(NumMethods());
227 }
228
229 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Method>>
GetDirectMethods()230 ClassAccessor::GetDirectMethods() const {
231 return GetMethodsInternal(NumDirectMethods());
232 }
233
234 inline IterationRange<ClassAccessor::DataIterator<ClassAccessor::Method>>
GetVirtualMethods()235 ClassAccessor::GetVirtualMethods() const {
236 IterationRange<DataIterator<Method>> methods = GetMethods();
237 // Skip the direct fields.
238 return { std::next(methods.begin(), NumDirectMethods()), methods.end() };
239 }
240
GetClassIdx()241 inline dex::TypeIndex ClassAccessor::GetClassIdx() const {
242 return dex_file_.GetClassDef(class_def_index_).class_idx_;
243 }
244
GetClassDef()245 inline const dex::ClassDef& ClassAccessor::GetClassDef() const {
246 return dex_file_.GetClassDef(GetClassDefIndex());
247 }
248
249 } // namespace art
250
251 #endif // ART_LIBDEXFILE_DEX_CLASS_ACCESSOR_INL_H_
252