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