/* * Copyright (C) 2017 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_OAT_AOT_CLASS_LINKER_H_ #define ART_RUNTIME_OAT_AOT_CLASS_LINKER_H_ #include <forward_list> #include "base/macros.h" #include "sdk_checker.h" #include "class_linker.h" namespace art HIDDEN { class Transaction; namespace gc { class Heap; } // namespace gc // TODO: move to dex2oat/. // AotClassLinker is only used for AOT compiler, which includes some logic for class initialization // which will only be used in pre-compilation. class AotClassLinker : public ClassLinker { public: explicit AotClassLinker(InternTable *intern_table); ~AotClassLinker(); EXPORT static void SetAppImageDexFiles(const std::vector<const DexFile*>* app_image_dex_files); EXPORT static bool CanReferenceInBootImageExtensionOrAppImage( ObjPtr<mirror::Class> klass, gc::Heap* heap) REQUIRES_SHARED(Locks::mutator_lock_); EXPORT void SetSdkChecker(std::unique_ptr<SdkChecker>&& sdk_checker_); const SdkChecker* GetSdkChecker() const; // Verifies if the method is accessible according to the SdkChecker (if installed). bool DenyAccessBasedOnPublicSdk(ArtMethod* art_method) const override REQUIRES_SHARED(Locks::mutator_lock_); // Verifies if the field is accessible according to the SdkChecker (if installed). bool DenyAccessBasedOnPublicSdk(ArtField* art_field) const override REQUIRES_SHARED(Locks::mutator_lock_); // Verifies if the descriptor is accessible according to the SdkChecker (if installed). bool DenyAccessBasedOnPublicSdk(std::string_view type_descriptor) const override; // Enable or disable public sdk checks. void SetEnablePublicSdkChecks(bool enabled) override; // Transaction support. EXPORT bool IsActiveTransaction() const; // EnterTransactionMode may suspend. EXPORT void EnterTransactionMode(bool strict, mirror::Class* root) REQUIRES_SHARED(Locks::mutator_lock_); EXPORT void ExitTransactionMode(); EXPORT void RollbackAllTransactions() REQUIRES_SHARED(Locks::mutator_lock_); // Transaction rollback and exit transaction are always done together, it's convenience to // do them in one function. void RollbackAndExitTransactionMode() REQUIRES_SHARED(Locks::mutator_lock_); const Transaction* GetTransaction() const; Transaction* GetTransaction(); bool IsActiveStrictTransactionMode() const; // Transaction constraint checks for AOT compilation. bool TransactionWriteConstraint(Thread* self, ObjPtr<mirror::Object> obj) override REQUIRES_SHARED(Locks::mutator_lock_); bool TransactionWriteValueConstraint(Thread* self, ObjPtr<mirror::Object> value) override REQUIRES_SHARED(Locks::mutator_lock_); // Note: The read constraint check is non-virtual because it's not needed by `UnstartedRuntime`. bool TransactionReadConstraint(Thread* self, ObjPtr<mirror::Object> obj) REQUIRES_SHARED(Locks::mutator_lock_); bool TransactionAllocationConstraint(Thread* self, ObjPtr<mirror::Class> klass) override REQUIRES_SHARED(Locks::mutator_lock_); // Transaction bookkeeping for AOT compilation. void RecordWriteFieldBoolean(mirror::Object* obj, MemberOffset field_offset, uint8_t value, bool is_volatile) override; void RecordWriteFieldByte(mirror::Object* obj, MemberOffset field_offset, int8_t value, bool is_volatile) override; void RecordWriteFieldChar(mirror::Object* obj, MemberOffset field_offset, uint16_t value, bool is_volatile) override; void RecordWriteFieldShort(mirror::Object* obj, MemberOffset field_offset, int16_t value, bool is_volatile) override; void RecordWriteField32(mirror::Object* obj, MemberOffset field_offset, uint32_t value, bool is_volatile) override; void RecordWriteField64(mirror::Object* obj, MemberOffset field_offset, uint64_t value, bool is_volatile) override; void RecordWriteFieldReference(mirror::Object* obj, MemberOffset field_offset, ObjPtr<mirror::Object> value, bool is_volatile) override REQUIRES_SHARED(Locks::mutator_lock_); void RecordWriteArray(mirror::Array* array, size_t index, uint64_t value) override REQUIRES_SHARED(Locks::mutator_lock_); void RecordStrongStringInsertion(ObjPtr<mirror::String> s) override REQUIRES(Locks::intern_table_lock_); void RecordWeakStringInsertion(ObjPtr<mirror::String> s) override REQUIRES(Locks::intern_table_lock_); void RecordStrongStringRemoval(ObjPtr<mirror::String> s) override REQUIRES(Locks::intern_table_lock_); void RecordWeakStringRemoval(ObjPtr<mirror::String> s) override REQUIRES(Locks::intern_table_lock_); void RecordResolveString(ObjPtr<mirror::DexCache> dex_cache, dex::StringIndex string_idx) override REQUIRES_SHARED(Locks::mutator_lock_); void RecordResolveMethodType(ObjPtr<mirror::DexCache> dex_cache, dex::ProtoIndex proto_idx) override REQUIRES_SHARED(Locks::mutator_lock_); // Aborting transactions for AOT compilation. void ThrowTransactionAbortError(Thread* self) override REQUIRES_SHARED(Locks::mutator_lock_); void AbortTransactionF(Thread* self, const char* fmt, ...) override __attribute__((__format__(__printf__, 3, 4))) REQUIRES_SHARED(Locks::mutator_lock_); void AbortTransactionV(Thread* self, const char* fmt, va_list args) override REQUIRES_SHARED(Locks::mutator_lock_); bool IsTransactionAborted() const override; void VisitTransactionRoots(RootVisitor* visitor) override REQUIRES_SHARED(Locks::mutator_lock_); protected: // Overridden version of PerformClassVerification allows skipping verification if the class was // previously verified but unloaded. verifier::FailureKind PerformClassVerification(Thread* self, verifier::VerifierDeps* verifier_deps, Handle<mirror::Class> klass, verifier::HardFailLogMode log_level, std::string* error_msg) override REQUIRES_SHARED(Locks::mutator_lock_); // Override AllocClass because aot compiler will need to perform a transaction check to determine // can we allocate class from heap. bool CanAllocClass() override REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(!Roles::uninterruptible_); bool InitializeClass(Thread *self, Handle<mirror::Class> klass, bool can_run_clinit, bool can_init_parents) override REQUIRES_SHARED(Locks::mutator_lock_) REQUIRES(!Locks::dex_lock_); private: std::unique_ptr<SdkChecker> sdk_checker_; // Transactions used for pre-initializing classes at compilation time. // Support nested transactions, maintain a list containing all transactions. Transactions are // handled under a stack discipline. Because GC needs to go over all transactions, we choose list // as substantial data structure instead of stack. std::forward_list<Transaction> preinitialization_transactions_; }; } // namespace art #endif // ART_RUNTIME_OAT_AOT_CLASS_LINKER_H_