/* * Copyright (C) 2018 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_LIBDEXFILE_DEX_CLASS_ACCESSOR_H_ #define ART_LIBDEXFILE_DEX_CLASS_ACCESSOR_H_ #include "code_item_accessors.h" #include "dex/dex_file.h" #include "dex_file_types.h" #include "invoke_type.h" #include "modifiers.h" namespace art { namespace dex { struct ClassDef; struct CodeItem; class DexFileVerifier; } // namespace dex class ClassIteratorData; class DexFile; template class IterationRange; class MethodReference; // Classes to access Dex data. class ClassAccessor { public: class BaseItem { public: explicit BaseItem(const DexFile& dex_file, const uint8_t* ptr_pos, const uint8_t* hiddenapi_ptr_pos) : dex_file_(dex_file), ptr_pos_(ptr_pos), hiddenapi_ptr_pos_(hiddenapi_ptr_pos) {} uint32_t GetIndex() const { return index_; } uint32_t GetAccessFlags() const { return access_flags_; } uint32_t GetHiddenapiFlags() const { return hiddenapi_flags_; } bool IsFinal() const { return (GetAccessFlags() & kAccFinal) != 0; } const DexFile& GetDexFile() const { return dex_file_; } const uint8_t* GetDataPointer() const { return ptr_pos_; } bool MemberIsNative() const { return GetAccessFlags() & kAccNative; } bool MemberIsFinal() const { return GetAccessFlags() & kAccFinal; } protected: // Internal data pointer for reading. const DexFile& dex_file_; const uint8_t* ptr_pos_ = nullptr; const uint8_t* hiddenapi_ptr_pos_ = nullptr; uint32_t index_ = 0u; uint32_t access_flags_ = 0u; uint32_t hiddenapi_flags_ = 0u; }; // A decoded version of the method of a class_data_item. class Method : public BaseItem { public: uint32_t GetCodeItemOffset() const { return code_off_; } InvokeType GetInvokeType(uint32_t class_access_flags) const { return is_static_or_direct_ ? GetDirectMethodInvokeType() : GetVirtualMethodInvokeType(class_access_flags); } MethodReference GetReference() const; CodeItemInstructionAccessor GetInstructions() const; CodeItemDataAccessor GetInstructionsAndData() const; const dex::CodeItem* GetCodeItem() const; bool IsStaticOrDirect() const { return is_static_or_direct_; } private: Method(const DexFile& dex_file, const uint8_t* ptr_pos, const uint8_t* hiddenapi_ptr_pos = nullptr, bool is_static_or_direct = true) : BaseItem(dex_file, ptr_pos, hiddenapi_ptr_pos), is_static_or_direct_(is_static_or_direct) {} void Read(); InvokeType GetDirectMethodInvokeType() const { return (GetAccessFlags() & kAccStatic) != 0 ? kStatic : kDirect; } InvokeType GetVirtualMethodInvokeType(uint32_t class_access_flags) const { DCHECK_EQ(GetAccessFlags() & kAccStatic, 0U); if ((class_access_flags & kAccInterface) != 0) { return kInterface; } else if ((GetAccessFlags() & kAccConstructor) != 0) { return kSuper; } else { return kVirtual; } } // Move to virtual method section. void NextSection() { DCHECK(is_static_or_direct_) << "Already in the virtual methods section"; is_static_or_direct_ = false; index_ = 0u; } bool is_static_or_direct_ = true; uint32_t code_off_ = 0u; friend class ClassAccessor; friend class dex::DexFileVerifier; }; // A decoded version of the field of a class_data_item. class Field : public BaseItem { public: Field(const DexFile& dex_file, const uint8_t* ptr_pos, const uint8_t* hiddenapi_ptr_pos = nullptr) : BaseItem(dex_file, ptr_pos, hiddenapi_ptr_pos) {} bool IsStatic() const { return is_static_; } private: void Read(); // Move to instance fields section. void NextSection() { index_ = 0u; is_static_ = false; } bool is_static_ = true; friend class ClassAccessor; friend class dex::DexFileVerifier; }; template class DataIterator { public: using iterator_category = std::forward_iterator_tag; using value_type = DataType; using difference_type = ptrdiff_t; using pointer = value_type*; using reference = value_type&; DataIterator(const DexFile& dex_file, uint32_t position, uint32_t partition_pos, uint32_t iterator_end, const uint8_t* ptr_pos, const uint8_t* hiddenapi_ptr_pos) : data_(dex_file, ptr_pos, hiddenapi_ptr_pos), position_(position), partition_pos_(partition_pos), iterator_end_(iterator_end) { ReadData(); } bool IsValid() const { return position_ < iterator_end_; } // Value after modification. DataIterator& operator++() { ++position_; ReadData(); return *this; } const value_type& operator*() const { return data_; } const value_type* operator->() const { return &data_; } bool operator==(const DataIterator& rhs) const { DCHECK_EQ(&data_.dex_file_, &rhs.data_.dex_file_) << "Comparing different dex files."; return position_ == rhs.position_; } bool operator!=(const DataIterator& rhs) const { return !(*this == rhs); } bool operator<(const DataIterator& rhs) const { DCHECK_EQ(&data_.dex_file_, &rhs.data_.dex_file_) << "Comparing different dex files."; return position_ < rhs.position_; } bool operator>(const DataIterator& rhs) const { return rhs < *this; } bool operator<=(const DataIterator& rhs) const { return !(rhs < *this); } bool operator>=(const DataIterator& rhs) const { return !(*this < rhs); } const uint8_t* GetDataPointer() const { return data_.ptr_pos_; } private: // Read data at current position. void ReadData() { if (IsValid()) { // At the end of the first section, go to the next section. if (position_ == partition_pos_) { data_.NextSection(); } data_.Read(); } } DataType data_; // Iterator position. uint32_t position_; // At partition_pos_, we go to the next section. const uint32_t partition_pos_; // At iterator_end_, the iterator is no longer valid. const uint32_t iterator_end_; friend class dex::DexFileVerifier; }; // Not explicit specifically for range-based loops. ALWAYS_INLINE ClassAccessor(const ClassIteratorData& data); // NOLINT [runtime/explicit] [5] ALWAYS_INLINE ClassAccessor(const DexFile& dex_file, const dex::ClassDef& class_def, bool parse_hiddenapi_class_data = false); ALWAYS_INLINE ClassAccessor(const DexFile& dex_file, uint32_t class_def_index); ClassAccessor(const DexFile& dex_file, const uint8_t* class_data, uint32_t class_def_index = DexFile::kDexNoIndex32, bool parse_hiddenapi_class_data = false); // Return the code item for a method. const dex::CodeItem* GetCodeItem(const Method& method) const; // Iterator data is not very iterator friendly, use visitors to get around this. template void VisitFieldsAndMethods(const StaticFieldVisitor& static_field_visitor, const InstanceFieldVisitor& instance_field_visitor, const DirectMethodVisitor& direct_method_visitor, const VirtualMethodVisitor& virtual_method_visitor) const; template void VisitMethods(const DirectMethodVisitor& direct_method_visitor, const VirtualMethodVisitor& virtual_method_visitor) const; template void VisitFields(const StaticFieldVisitor& static_field_visitor, const InstanceFieldVisitor& instance_field_visitor) const; // Return the iteration range for all the fields. IterationRange> GetFields() const; // Return the iteration range for all the static fields. IterationRange> GetStaticFields() const; // Return the iteration range for all the instance fields. IterationRange> GetInstanceFields() const; // Return the iteration range for all the methods. IterationRange> GetMethods() const; // Return the iteration range for the direct methods. IterationRange> GetDirectMethods() const; // Return the iteration range for the virtual methods. IterationRange> GetVirtualMethods() const; uint32_t NumStaticFields() const { return num_static_fields_; } uint32_t NumInstanceFields() const { return num_instance_fields_; } uint32_t NumFields() const { return NumStaticFields() + NumInstanceFields(); } uint32_t NumDirectMethods() const { return num_direct_methods_; } uint32_t NumVirtualMethods() const { return num_virtual_methods_; } uint32_t NumMethods() const { return NumDirectMethods() + NumVirtualMethods(); } const char* GetDescriptor() const; std::string_view GetDescriptorView() const; dex::TypeIndex GetClassIdx() const; const DexFile& GetDexFile() const { return dex_file_; } bool HasClassData() const { return ptr_pos_ != nullptr; } bool HasHiddenapiClassData() const { return hiddenapi_ptr_pos_ != nullptr; } uint32_t GetClassDefIndex() const { return class_def_index_; } const dex::ClassDef& GetClassDef() const; protected: // Template visitor to reduce copy paste for visiting elements. // No thread safety analysis since the visitor may require capabilities. template void VisitMembers(size_t count, const Visitor& visitor, DataType* data) const NO_THREAD_SAFETY_ANALYSIS; // Return an iteration range for the first fields. IterationRange> GetFieldsInternal(size_t count) const; // Return an iteration range for the first methods. IterationRange> GetMethodsInternal(size_t count) const; const DexFile& dex_file_; const uint32_t class_def_index_; const uint8_t* ptr_pos_ = nullptr; // Pointer into stream of class_data_item. const uint8_t* hiddenapi_ptr_pos_ = nullptr; // Pointer into stream of hiddenapi_metadata. const uint32_t num_static_fields_ = 0u; const uint32_t num_instance_fields_ = 0u; const uint32_t num_direct_methods_ = 0u; const uint32_t num_virtual_methods_ = 0u; friend class dex::DexFileVerifier; }; } // namespace art #endif // ART_LIBDEXFILE_DEX_CLASS_ACCESSOR_H_