/* * Copyright (C) 2015 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 "dex/quick/quick_compiler.h" #include "dex/pass_manager.h" #include "dex/verification_results.h" #include "dex/quick/dex_file_to_method_inliner_map.h" #include "runtime/dex_file.h" #include "driver/compiler_options.h" #include "driver/compiler_driver.h" #include "codegen_x86.h" #include "gtest/gtest.h" #include "utils/assembler_test_base.h" namespace art { class QuickAssembleX86TestBase : public testing::Test { protected: X86Mir2Lir* Prepare(InstructionSet target) { isa_ = target; pool_.reset(new ArenaPool()); compiler_options_.reset(new CompilerOptions( CompilerOptions::kDefaultCompilerFilter, CompilerOptions::kDefaultHugeMethodThreshold, CompilerOptions::kDefaultLargeMethodThreshold, CompilerOptions::kDefaultSmallMethodThreshold, CompilerOptions::kDefaultTinyMethodThreshold, CompilerOptions::kDefaultNumDexMethodsThreshold, CompilerOptions::kDefaultInlineDepthLimit, CompilerOptions::kDefaultInlineMaxCodeUnits, false, CompilerOptions::kDefaultTopKProfileThreshold, false, CompilerOptions::kDefaultGenerateDebugInfo, false, false, false, false, nullptr, new PassManagerOptions(), nullptr, false)); verification_results_.reset(new VerificationResults(compiler_options_.get())); method_inliner_map_.reset(new DexFileToMethodInlinerMap()); compiler_driver_.reset(new CompilerDriver( compiler_options_.get(), verification_results_.get(), method_inliner_map_.get(), Compiler::kQuick, isa_, nullptr, false, nullptr, nullptr, nullptr, 0, false, false, "", 0, -1, "")); cu_.reset(new CompilationUnit(pool_.get(), isa_, compiler_driver_.get(), nullptr)); DexFile::CodeItem* code_item = static_cast<DexFile::CodeItem*>( cu_->arena.Alloc(sizeof(DexFile::CodeItem), kArenaAllocMisc)); memset(code_item, 0, sizeof(DexFile::CodeItem)); cu_->mir_graph.reset(new MIRGraph(cu_.get(), &cu_->arena)); cu_->mir_graph->current_code_item_ = code_item; cu_->cg.reset(QuickCompiler::GetCodeGenerator(cu_.get(), nullptr)); test_helper_.reset(new AssemblerTestInfrastructure( isa_ == kX86 ? "x86" : "x86_64", "as", isa_ == kX86 ? " --32" : "", "objdump", " -h", "objdump", isa_ == kX86 ? " -D -bbinary -mi386 --no-show-raw-insn" : " -D -bbinary -mi386:x86-64 -Mx86-64,addr64,data32 --no-show-raw-insn", nullptr)); X86Mir2Lir* m2l = static_cast<X86Mir2Lir*>(cu_->cg.get()); m2l->CompilerInitializeRegAlloc(); return m2l; } void Release() { cu_.reset(); compiler_driver_.reset(); method_inliner_map_.reset(); verification_results_.reset(); compiler_options_.reset(); pool_.reset(); test_helper_.reset(); } void TearDown() OVERRIDE { Release(); } bool CheckTools(InstructionSet target) { Prepare(target); bool result = test_helper_->CheckTools(); Release(); return result; } std::unique_ptr<CompilationUnit> cu_; std::unique_ptr<AssemblerTestInfrastructure> test_helper_; private: InstructionSet isa_; std::unique_ptr<ArenaPool> pool_; std::unique_ptr<CompilerOptions> compiler_options_; std::unique_ptr<VerificationResults> verification_results_; std::unique_ptr<DexFileToMethodInlinerMap> method_inliner_map_; std::unique_ptr<CompilerDriver> compiler_driver_; }; class QuickAssembleX86LowLevelTest : public QuickAssembleX86TestBase { protected: void Test(InstructionSet target, std::string test_name, std::string gcc_asm, int opcode, int op0 = 0, int op1 = 0, int op2 = 0, int op3 = 0, int op4 = 0) { X86Mir2Lir* m2l = Prepare(target); LIR lir; memset(&lir, 0, sizeof(LIR)); lir.opcode = opcode; lir.operands[0] = op0; lir.operands[1] = op1; lir.operands[2] = op2; lir.operands[3] = op3; lir.operands[4] = op4; lir.flags.size = m2l->GetInsnSize(&lir); AssemblerStatus status = m2l->AssembleInstructions(&lir, 0); // We don't expect a retry. ASSERT_EQ(status, AssemblerStatus::kSuccess); // Need a "base" std::vector. std::vector<uint8_t> buffer(m2l->code_buffer_.begin(), m2l->code_buffer_.end()); test_helper_->Driver(buffer, gcc_asm, test_name); Release(); } }; TEST_F(QuickAssembleX86LowLevelTest, Addpd) { Test(kX86, "Addpd", "addpd %xmm1, %xmm0\n", kX86AddpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); Test(kX86_64, "Addpd", "addpd %xmm1, %xmm0\n", kX86AddpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); } TEST_F(QuickAssembleX86LowLevelTest, Subpd) { Test(kX86, "Subpd", "subpd %xmm1, %xmm0\n", kX86SubpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); Test(kX86_64, "Subpd", "subpd %xmm1, %xmm0\n", kX86SubpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); } TEST_F(QuickAssembleX86LowLevelTest, Mulpd) { Test(kX86, "Mulpd", "mulpd %xmm1, %xmm0\n", kX86MulpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); Test(kX86_64, "Mulpd", "mulpd %xmm1, %xmm0\n", kX86MulpdRR, RegStorage::Solo128(0).GetReg(), RegStorage::Solo128(1).GetReg()); } TEST_F(QuickAssembleX86LowLevelTest, Pextrw) { Test(kX86, "Pextrw", "pextrw $7, %xmm3, 8(%eax)\n", kX86PextrwMRI, RegStorage::Solo32(r0).GetReg(), 8, RegStorage::Solo128(3).GetReg(), 7); Test(kX86_64, "Pextrw", "pextrw $7, %xmm8, 8(%r10)\n", kX86PextrwMRI, RegStorage::Solo64(r10q).GetReg(), 8, RegStorage::Solo128(8).GetReg(), 7); } class QuickAssembleX86MacroTest : public QuickAssembleX86TestBase { protected: typedef void (X86Mir2Lir::*AsmFn)(MIR*); void TestVectorFn(InstructionSet target, Instruction::Code opcode, AsmFn f, std::string inst_string) { X86Mir2Lir *m2l = Prepare(target); // Create a vector MIR. MIR* mir = cu_->mir_graph->NewMIR(); mir->dalvikInsn.opcode = opcode; mir->dalvikInsn.vA = 0; // Destination and source. mir->dalvikInsn.vB = 1; // Source. int vector_size = 128; int vector_type = kDouble; mir->dalvikInsn.vC = (vector_type << 16) | vector_size; // Type size. (m2l->*f)(mir); m2l->AssembleLIR(); std::string gcc_asm = inst_string + " %xmm1, %xmm0\n"; // Need a "base" std::vector. std::vector<uint8_t> buffer(m2l->code_buffer_.begin(), m2l->code_buffer_.end()); test_helper_->Driver(buffer, gcc_asm, inst_string); Release(); } // Tests are member functions as many of the assembler functions are protected or private, // and it would be inelegant to define ART_FRIEND_TEST for all the tests. void TestAddpd() { TestVectorFn(kX86, static_cast<Instruction::Code>(kMirOpPackedAddition), &X86Mir2Lir::GenAddVector, "addpd"); TestVectorFn(kX86_64, static_cast<Instruction::Code>(kMirOpPackedAddition), &X86Mir2Lir::GenAddVector, "addpd"); } void TestSubpd() { TestVectorFn(kX86, static_cast<Instruction::Code>(kMirOpPackedSubtract), &X86Mir2Lir::GenSubtractVector, "subpd"); TestVectorFn(kX86_64, static_cast<Instruction::Code>(kMirOpPackedSubtract), &X86Mir2Lir::GenSubtractVector, "subpd"); } void TestMulpd() { TestVectorFn(kX86, static_cast<Instruction::Code>(kMirOpPackedMultiply), &X86Mir2Lir::GenMultiplyVector, "mulpd"); TestVectorFn(kX86_64, static_cast<Instruction::Code>(kMirOpPackedMultiply), &X86Mir2Lir::GenMultiplyVector, "mulpd"); } }; TEST_F(QuickAssembleX86MacroTest, CheckTools) { ASSERT_TRUE(CheckTools(kX86)) << "x86 tools not found."; ASSERT_TRUE(CheckTools(kX86_64)) << "x86_64 tools not found."; } #define DECLARE_TEST(name) \ TEST_F(QuickAssembleX86MacroTest, name) { \ Test ## name(); \ } DECLARE_TEST(Addpd) DECLARE_TEST(Subpd) DECLARE_TEST(Mulpd) } // namespace art