diff options
Diffstat (limited to 'compiler/llvm')
-rw-r--r-- | compiler/llvm/compiler_llvm.cc | 68 | ||||
-rw-r--r-- | compiler/llvm/compiler_llvm.h | 13 | ||||
-rw-r--r-- | compiler/llvm/llvm_compiler.cc | 161 | ||||
-rw-r--r-- | compiler/llvm/llvm_compiler.h | 29 |
4 files changed, 234 insertions, 37 deletions
diff --git a/compiler/llvm/compiler_llvm.cc b/compiler/llvm/compiler_llvm.cc index 5990e8caea..3aeecad78e 100644 --- a/compiler/llvm/compiler_llvm.cc +++ b/compiler/llvm/compiler_llvm.cc @@ -141,7 +141,7 @@ CompileDexMethod(DexCompilationUnit* dex_compilation_unit, InvokeType invoke_typ cunit->SetDexCompilationUnit(dex_compilation_unit); cunit->SetCompilerDriver(compiler_driver_); // TODO: consolidate ArtCompileMethods - CompileOneMethod(*compiler_driver_, + CompileOneMethod(compiler_driver_, compiler_driver_->GetCompiler(), dex_compilation_unit->GetCodeItem(), dex_compilation_unit->GetAccessFlags(), @@ -172,68 +172,62 @@ CompileNativeMethod(DexCompilationUnit* dex_compilation_unit) { } -} // namespace llvm -} // namespace art - -static art::llvm::CompilerLLVM* ContextOf(art::CompilerDriver* driver) { +static CompilerLLVM* ContextOf(art::CompilerDriver* driver) { void *compiler_context = driver->GetCompilerContext(); CHECK(compiler_context != NULL); - return reinterpret_cast<art::llvm::CompilerLLVM*>(compiler_context); + return reinterpret_cast<CompilerLLVM*>(compiler_context); } -static art::llvm::CompilerLLVM* ContextOf(const art::CompilerDriver& driver) { +static CompilerLLVM* ContextOf(const art::CompilerDriver& driver) { void *compiler_context = driver.GetCompilerContext(); CHECK(compiler_context != NULL); - return reinterpret_cast<art::llvm::CompilerLLVM*>(compiler_context); + return reinterpret_cast<CompilerLLVM*>(compiler_context); } -extern "C" void ArtInitCompilerContext(art::CompilerDriver* driver) { +void ArtInitCompilerContext(CompilerDriver* driver) { CHECK(driver->GetCompilerContext() == nullptr); - art::llvm::CompilerLLVM* compiler_llvm = new art::llvm::CompilerLLVM(driver, - driver->GetInstructionSet()); + CompilerLLVM* compiler_llvm = new CompilerLLVM(driver, driver->GetInstructionSet()); driver->SetCompilerContext(compiler_llvm); } -extern "C" void ArtUnInitCompilerContext(art::CompilerDriver* driver) { +void ArtUnInitCompilerContext(CompilerDriver* driver) { delete ContextOf(driver); driver->SetCompilerContext(nullptr); } -extern "C" art::CompiledMethod* ArtCompileMethod(art::CompilerDriver* driver, - const art::DexFile::CodeItem* code_item, - uint32_t access_flags, - art::InvokeType invoke_type, - uint16_t class_def_idx, - uint32_t method_idx, - jobject class_loader, - const art::DexFile& dex_file) { + +CompiledMethod* ArtCompileMethod(CompilerDriver* driver, const DexFile::CodeItem* code_item, + uint32_t access_flags, InvokeType invoke_type, + uint16_t class_def_idx, uint32_t method_idx, jobject class_loader, + const DexFile& dex_file) { UNUSED(class_def_idx); // TODO: this is used with Compiler::RequiresConstructorBarrier. - art::ClassLinker *class_linker = art::Runtime::Current()->GetClassLinker(); + ClassLinker *class_linker = Runtime::Current()->GetClassLinker(); - art::DexCompilationUnit dex_compilation_unit( - NULL, class_loader, class_linker, dex_file, code_item, - class_def_idx, method_idx, access_flags, driver->GetVerifiedMethod(&dex_file, method_idx)); - art::llvm::CompilerLLVM* compiler_llvm = ContextOf(driver); - art::CompiledMethod* result = compiler_llvm->CompileDexMethod(&dex_compilation_unit, invoke_type); + DexCompilationUnit dex_compilation_unit(nullptr, class_loader, class_linker, dex_file, code_item, + class_def_idx, method_idx, access_flags, + driver->GetVerifiedMethod(&dex_file, method_idx)); + CompilerLLVM* compiler_llvm = ContextOf(driver); + CompiledMethod* result = compiler_llvm->CompileDexMethod(&dex_compilation_unit, invoke_type); return result; } -extern "C" art::CompiledMethod* ArtLLVMJniCompileMethod(art::CompilerDriver* driver, - uint32_t access_flags, uint32_t method_idx, - const art::DexFile& dex_file) { - art::ClassLinker *class_linker = art::Runtime::Current()->GetClassLinker(); +CompiledMethod* ArtLLVMJniCompileMethod(CompilerDriver* driver, uint32_t access_flags, + uint32_t method_idx, const DexFile& dex_file) { + ClassLinker *class_linker = Runtime::Current()->GetClassLinker(); - art::DexCompilationUnit dex_compilation_unit( - nullptr, nullptr, class_linker, dex_file, nullptr, - 0, method_idx, access_flags, nullptr); + DexCompilationUnit dex_compilation_unit(nullptr, nullptr, class_linker, dex_file, nullptr, + 0, method_idx, access_flags, nullptr); - art::llvm::CompilerLLVM* compiler_llvm = ContextOf(driver); - art::CompiledMethod* result = compiler_llvm->CompileNativeMethod(&dex_compilation_unit); + CompilerLLVM* compiler_llvm = ContextOf(driver); + CompiledMethod* result = compiler_llvm->CompileNativeMethod(&dex_compilation_unit); return result; } -extern "C" void compilerLLVMSetBitcodeFileName(const art::CompilerDriver& driver, - const std::string& filename) { +void compilerLLVMSetBitcodeFileName(const CompilerDriver& driver, const std::string& filename) { ContextOf(driver)->SetBitcodeFileName(filename); } + +} // namespace llvm +} // namespace art + diff --git a/compiler/llvm/compiler_llvm.h b/compiler/llvm/compiler_llvm.h index cc74deb7be..7d29198667 100644 --- a/compiler/llvm/compiler_llvm.h +++ b/compiler/llvm/compiler_llvm.h @@ -95,6 +95,19 @@ class CompilerLLVM { DISALLOW_COPY_AND_ASSIGN(CompilerLLVM); }; +void ArtInitCompilerContext(CompilerDriver* driver); + +void ArtUnInitCompilerContext(CompilerDriver* driver); + +CompiledMethod* ArtCompileMethod(CompilerDriver* driver, const DexFile::CodeItem* code_item, + uint32_t access_flags, InvokeType invoke_type, + uint16_t class_def_idx, uint32_t method_idx, jobject class_loader, + const DexFile& dex_file); + +CompiledMethod* ArtLLVMJniCompileMethod(CompilerDriver* driver, uint32_t access_flags, + uint32_t method_idx, const DexFile& dex_file); + +void compilerLLVMSetBitcodeFileName(const CompilerDriver& driver, const std::string& filename); } // namespace llvm } // namespace art diff --git a/compiler/llvm/llvm_compiler.cc b/compiler/llvm/llvm_compiler.cc new file mode 100644 index 0000000000..55af6145be --- /dev/null +++ b/compiler/llvm/llvm_compiler.cc @@ -0,0 +1,161 @@ +/* + * Copyright (C) 2014 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. + */ + +#include "llvm_compiler.h" + +#ifdef ART_USE_PORTABLE_COMPILER +#include "compiler.h" +#include "compiler_llvm.h" +#include "dex/portable/mir_to_gbc.h" +#include "dex_file.h" +#include "elf_writer_mclinker.h" +#include "mirror/art_method-inl.h" +#endif + +namespace art { + +#ifdef ART_USE_PORTABLE_COMPILER + +namespace llvm { + +// Thread-local storage compiler worker threads +class LLVMCompilerTls : public CompilerTls { + public: + LLVMCompilerTls() : llvm_info_(nullptr) {} + ~LLVMCompilerTls() {} + + void* GetLLVMInfo() { return llvm_info_; } + + void SetLLVMInfo(void* llvm_info) { llvm_info_ = llvm_info; } + + private: + void* llvm_info_; +}; + + + +class LLVMCompiler FINAL : public Compiler { + public: + explicit LLVMCompiler(CompilerDriver* driver) : Compiler(driver, 1000) {} + + CompilerTls* CreateNewCompilerTls() { + return new LLVMCompilerTls(); + } + + void Init() const OVERRIDE { + ArtInitCompilerContext(GetCompilerDriver()); + } + + void UnInit() const OVERRIDE { + ArtUnInitCompilerContext(GetCompilerDriver()); + } + + bool CanCompileMethod(uint32_t method_idx, const DexFile& dex_file, CompilationUnit* cu) const + OVERRIDE { + return true; + } + + CompiledMethod* Compile(const DexFile::CodeItem* code_item, + uint32_t access_flags, + InvokeType invoke_type, + uint16_t class_def_idx, + uint32_t method_idx, + jobject class_loader, + const DexFile& dex_file) const OVERRIDE { + CompiledMethod* method = TryCompileWithSeaIR(code_item, + access_flags, + invoke_type, + class_def_idx, + method_idx, + class_loader, + dex_file); + if (method != nullptr) { + return method; + } + + return ArtCompileMethod(GetCompilerDriver(), + code_item, + access_flags, + invoke_type, + class_def_idx, + method_idx, + class_loader, + dex_file); + } + + CompiledMethod* JniCompile(uint32_t access_flags, + uint32_t method_idx, + const DexFile& dex_file) const OVERRIDE { + return ArtLLVMJniCompileMethod(GetCompilerDriver(), access_flags, method_idx, dex_file); + } + + uintptr_t GetEntryPointOf(mirror::ArtMethod* method) const { + return reinterpret_cast<uintptr_t>(method->GetEntryPointFromPortableCompiledCode()); + } + + bool WriteElf(art::File* file, + OatWriter* oat_writer, + const std::vector<const art::DexFile*>& dex_files, + const std::string& android_root, + bool is_host, const CompilerDriver& driver) const + OVERRIDE + SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) { + return art::ElfWriterMclinker::Create( + file, oat_writer, dex_files, android_root, is_host, driver); + } + + Backend* GetCodeGenerator(CompilationUnit* cu, void* compilation_unit) const { + return PortableCodeGenerator( + cu, cu->mir_graph.get(), &cu->arena, + reinterpret_cast<art::llvm::LlvmCompilationUnit*>(compilation_unit)); + } + + void InitCompilationUnit(CompilationUnit& cu) const { + // Fused long branches not currently useful in bitcode. + cu.disable_opt |= + (1 << kBranchFusing) | + (1 << kSuppressExceptionEdges); + } + + bool IsPortable() const OVERRIDE { + return true; + } + + void SetBitcodeFileName(const CompilerDriver& driver, const std::string& filename) { + typedef void (*SetBitcodeFileNameFn)(const CompilerDriver&, const std::string&); + + SetBitcodeFileNameFn set_bitcode_file_name = + reinterpret_cast<SetBitcodeFileNameFn>(compilerLLVMSetBitcodeFileName); + + set_bitcode_file_name(driver, filename); + } + + private: + DISALLOW_COPY_AND_ASSIGN(LLVMCompiler); +}; + +} // namespace llvm +#endif + +Compiler* CreateLLVMCompiler(CompilerDriver* driver) { +#ifdef ART_USE_PORTABLE_COMPILER + return new llvm::LLVMCompiler(driver); +#else + return nullptr; +#endif +} + +} // namespace art diff --git a/compiler/llvm/llvm_compiler.h b/compiler/llvm/llvm_compiler.h new file mode 100644 index 0000000000..da6d0e9784 --- /dev/null +++ b/compiler/llvm/llvm_compiler.h @@ -0,0 +1,29 @@ +/* + * 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_LLVM_LLVM_COMPILER_H_ +#define ART_COMPILER_LLVM_LLVM_COMPILER_H_ + +namespace art { + +class Compiler; +class CompilerDriver; + +Compiler* CreateLLVMCompiler(CompilerDriver* driver); + +} + +#endif // ART_COMPILER_LLVM_LLVM_COMPILER_H_ |