You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
185 lines
5.1 KiB
185 lines
5.1 KiB
/*
|
|
* Copyright (C) 2012 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_COMPILER_DRIVER_DEX_COMPILATION_UNIT_H_
|
|
#define ART_COMPILER_DRIVER_DEX_COMPILATION_UNIT_H_
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "base/arena_object.h"
|
|
#include "dex/code_item_accessors.h"
|
|
#include "dex/dex_file.h"
|
|
#include "handle.h"
|
|
|
|
namespace art {
|
|
namespace mirror {
|
|
class Class;
|
|
class ClassLoader;
|
|
class DexCache;
|
|
} // namespace mirror
|
|
class ClassLinker;
|
|
class VerifiedMethod;
|
|
|
|
class DexCompilationUnit : public DeletableArenaObject<kArenaAllocMisc> {
|
|
public:
|
|
DexCompilationUnit(Handle<mirror::ClassLoader> class_loader,
|
|
ClassLinker* class_linker,
|
|
const DexFile& dex_file,
|
|
const dex::CodeItem* code_item,
|
|
uint16_t class_def_idx,
|
|
uint32_t method_idx,
|
|
uint32_t access_flags,
|
|
const VerifiedMethod* verified_method,
|
|
Handle<mirror::DexCache> dex_cache,
|
|
Handle<mirror::Class> compiling_class = Handle<mirror::Class>());
|
|
|
|
Handle<mirror::ClassLoader> GetClassLoader() const {
|
|
return class_loader_;
|
|
}
|
|
|
|
ClassLinker* GetClassLinker() const {
|
|
return class_linker_;
|
|
}
|
|
|
|
const DexFile* GetDexFile() const {
|
|
return dex_file_;
|
|
}
|
|
|
|
uint16_t GetClassDefIndex() const {
|
|
return class_def_idx_;
|
|
}
|
|
|
|
uint32_t GetDexMethodIndex() const {
|
|
return dex_method_idx_;
|
|
}
|
|
|
|
const dex::CodeItem* GetCodeItem() const {
|
|
return code_item_;
|
|
}
|
|
|
|
const char* GetShorty() const {
|
|
const dex::MethodId& method_id = dex_file_->GetMethodId(dex_method_idx_);
|
|
return dex_file_->GetMethodShorty(method_id);
|
|
}
|
|
|
|
const char* GetShorty(uint32_t* shorty_len) const {
|
|
const dex::MethodId& method_id = dex_file_->GetMethodId(dex_method_idx_);
|
|
return dex_file_->GetMethodShorty(method_id, shorty_len);
|
|
}
|
|
|
|
uint32_t GetAccessFlags() const {
|
|
return access_flags_;
|
|
}
|
|
|
|
bool IsConstructor() const {
|
|
return ((access_flags_ & kAccConstructor) != 0);
|
|
}
|
|
|
|
bool IsNative() const {
|
|
return ((access_flags_ & kAccNative) != 0);
|
|
}
|
|
|
|
bool IsStatic() const {
|
|
return ((access_flags_ & kAccStatic) != 0);
|
|
}
|
|
|
|
bool IsSynchronized() const {
|
|
return ((access_flags_ & kAccSynchronized) != 0);
|
|
}
|
|
|
|
const VerifiedMethod* GetVerifiedMethod() const {
|
|
return verified_method_;
|
|
}
|
|
|
|
void ClearVerifiedMethod() {
|
|
verified_method_ = nullptr;
|
|
}
|
|
|
|
const std::string& GetSymbol();
|
|
|
|
Handle<mirror::DexCache> GetDexCache() const {
|
|
return dex_cache_;
|
|
}
|
|
|
|
const CodeItemDataAccessor& GetCodeItemAccessor() const {
|
|
return code_item_accessor_;
|
|
}
|
|
|
|
Handle<mirror::Class> GetCompilingClass() const {
|
|
return compiling_class_;
|
|
}
|
|
|
|
// Does this <init> method require a constructor barrier (prior to the return)?
|
|
// The answer is "yes", if and only if the class has any instance final fields.
|
|
// (This must not be called for any non-<init> methods; the answer would be "no").
|
|
//
|
|
// ---
|
|
//
|
|
// JLS 17.5.1 "Semantics of final fields" mandates that all final fields are frozen at the end
|
|
// of the invoked constructor. The constructor barrier is a conservative implementation means of
|
|
// enforcing the freezes happen-before the object being constructed is observable by another
|
|
// thread.
|
|
//
|
|
// Note: This question only makes sense for instance constructors;
|
|
// static constructors (despite possibly having finals) never need
|
|
// a barrier.
|
|
//
|
|
// JLS 12.4.2 "Detailed Initialization Procedure" approximately describes
|
|
// class initialization as:
|
|
//
|
|
// lock(class.lock)
|
|
// class.state = initializing
|
|
// unlock(class.lock)
|
|
//
|
|
// invoke <clinit>
|
|
//
|
|
// lock(class.lock)
|
|
// class.state = initialized
|
|
// unlock(class.lock) <-- acts as a release
|
|
//
|
|
// The last operation in the above example acts as an atomic release
|
|
// for any stores in <clinit>, which ends up being stricter
|
|
// than what a constructor barrier needs.
|
|
//
|
|
// See also QuasiAtomic::ThreadFenceForConstructor().
|
|
bool RequiresConstructorBarrier() const;
|
|
|
|
private:
|
|
const Handle<mirror::ClassLoader> class_loader_;
|
|
|
|
ClassLinker* const class_linker_;
|
|
|
|
const DexFile* const dex_file_;
|
|
|
|
const dex::CodeItem* const code_item_;
|
|
const uint16_t class_def_idx_;
|
|
const uint32_t dex_method_idx_;
|
|
const uint32_t access_flags_;
|
|
const VerifiedMethod* verified_method_;
|
|
|
|
const Handle<mirror::DexCache> dex_cache_;
|
|
|
|
const CodeItemDataAccessor code_item_accessor_;
|
|
|
|
Handle<mirror::Class> compiling_class_;
|
|
|
|
std::string symbol_;
|
|
};
|
|
|
|
} // namespace art
|
|
|
|
#endif // ART_COMPILER_DRIVER_DEX_COMPILATION_UNIT_H_
|