diff options
author | Roland Levillain <rpl@google.com> | 2014-10-24 10:48:38 +0000 |
---|---|---|
committer | Gerrit Code Review <noreply-gerritcodereview@google.com> | 2014-10-24 10:48:39 +0000 |
commit | 64727aeef8f4243f2058a19a43a937248a60dae1 (patch) | |
tree | 9bf09ff29bb49d4be382e49822300668913f18e5 /compiler | |
parent | 8bde036ebd74ce94477e65077bed6dea9c29616b (diff) | |
parent | 2e07b4f0a84a7968b4690c2b1be2e2f75cc6fa8e (diff) | |
download | art-64727aeef8f4243f2058a19a43a937248a60dae1.zip art-64727aeef8f4243f2058a19a43a937248a60dae1.tar.gz art-64727aeef8f4243f2058a19a43a937248a60dae1.tar.bz2 |
Merge "Revert "Revert "Implement long negate instruction in the optimizing compiler."""
Diffstat (limited to 'compiler')
-rw-r--r-- | compiler/optimizing/builder.cc | 5 | ||||
-rw-r--r-- | compiler/optimizing/code_generator_arm.cc | 26 | ||||
-rw-r--r-- | compiler/optimizing/code_generator_x86.cc | 13 | ||||
-rw-r--r-- | compiler/optimizing/code_generator_x86_64.cc | 6 | ||||
-rw-r--r-- | compiler/utils/x86_64/assembler_x86_64.cc | 7 | ||||
-rw-r--r-- | compiler/utils/x86_64/assembler_x86_64.h | 1 |
6 files changed, 54 insertions, 4 deletions
diff --git a/compiler/optimizing/builder.cc b/compiler/optimizing/builder.cc index e5e9964..e4dee46 100644 --- a/compiler/optimizing/builder.cc +++ b/compiler/optimizing/builder.cc @@ -748,6 +748,11 @@ bool HGraphBuilder::AnalyzeDexInstruction(const Instruction& instruction, uint32 break; } + case Instruction::NEG_LONG: { + Unop_12x<HNeg>(instruction, Primitive::kPrimLong); + break; + } + case Instruction::NOT_INT: { Unop_12x<HNot>(instruction, Primitive::kPrimInt); break; diff --git a/compiler/optimizing/code_generator_arm.cc b/compiler/optimizing/code_generator_arm.cc index 7b00d2f..a3b31d8 100644 --- a/compiler/optimizing/code_generator_arm.cc +++ b/compiler/optimizing/code_generator_arm.cc @@ -1043,11 +1043,13 @@ void LocationsBuilderARM::VisitNeg(HNeg* neg) { new (GetGraph()->GetArena()) LocationSummary(neg, LocationSummary::kNoCall); switch (neg->GetResultType()) { case Primitive::kPrimInt: + case Primitive::kPrimLong: { + bool output_overlaps = (neg->GetResultType() == Primitive::kPrimLong); locations->SetInAt(0, Location::RequiresRegister()); - locations->SetOut(Location::RequiresRegister()); + locations->SetOut(Location::RequiresRegister(), output_overlaps); break; + } - case Primitive::kPrimLong: case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); @@ -1069,6 +1071,26 @@ void InstructionCodeGeneratorARM::VisitNeg(HNeg* neg) { break; case Primitive::kPrimLong: + DCHECK(in.IsRegisterPair()); + // out.lo = 0 - in.lo (and update the carry/borrow (C) flag) + __ rsbs(out.AsRegisterPairLow<Register>(), + in.AsRegisterPairLow<Register>(), + ShifterOperand(0)); + // We cannot emit an RSC (Reverse Subtract with Carry) + // instruction here, as it does not exist in the Thumb-2 + // instruction set. We use the following approach + // using SBC and SUB instead. + // + // out.hi = -C + __ sbc(out.AsRegisterPairHigh<Register>(), + out.AsRegisterPairHigh<Register>(), + ShifterOperand(out.AsRegisterPairHigh<Register>())); + // out.hi = out.hi - in.hi + __ sub(out.AsRegisterPairHigh<Register>(), + out.AsRegisterPairHigh<Register>(), + ShifterOperand(in.AsRegisterPairHigh<Register>())); + break; + case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); diff --git a/compiler/optimizing/code_generator_x86.cc b/compiler/optimizing/code_generator_x86.cc index 61f0750..aa0f06b 100644 --- a/compiler/optimizing/code_generator_x86.cc +++ b/compiler/optimizing/code_generator_x86.cc @@ -990,11 +990,11 @@ void LocationsBuilderX86::VisitNeg(HNeg* neg) { new (GetGraph()->GetArena()) LocationSummary(neg, LocationSummary::kNoCall); switch (neg->GetResultType()) { case Primitive::kPrimInt: + case Primitive::kPrimLong: locations->SetInAt(0, Location::RequiresRegister()); locations->SetOut(Location::SameAsFirstInput()); break; - case Primitive::kPrimLong: case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); @@ -1016,6 +1016,17 @@ void InstructionCodeGeneratorX86::VisitNeg(HNeg* neg) { break; case Primitive::kPrimLong: + DCHECK(in.IsRegisterPair()); + __ negl(out.AsRegisterPairLow<Register>()); + // Negation is similar to subtraction from zero. The least + // significant byte triggers a borrow when it is different from + // zero; to take it into account, add 1 to the most significant + // byte if the carry flag (CF) is set to 1 after the first NEGL + // operation. + __ adcl(out.AsRegisterPairHigh<Register>(), Immediate(0)); + __ negl(out.AsRegisterPairHigh<Register>()); + break; + case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); diff --git a/compiler/optimizing/code_generator_x86_64.cc b/compiler/optimizing/code_generator_x86_64.cc index 4a05b89..892ca9d 100644 --- a/compiler/optimizing/code_generator_x86_64.cc +++ b/compiler/optimizing/code_generator_x86_64.cc @@ -976,11 +976,11 @@ void LocationsBuilderX86_64::VisitNeg(HNeg* neg) { new (GetGraph()->GetArena()) LocationSummary(neg, LocationSummary::kNoCall); switch (neg->GetResultType()) { case Primitive::kPrimInt: + case Primitive::kPrimLong: locations->SetInAt(0, Location::RequiresRegister()); locations->SetOut(Location::SameAsFirstInput()); break; - case Primitive::kPrimLong: case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); @@ -1002,6 +1002,10 @@ void InstructionCodeGeneratorX86_64::VisitNeg(HNeg* neg) { break; case Primitive::kPrimLong: + DCHECK(in.IsRegister()); + __ negq(out.As<CpuRegister>()); + break; + case Primitive::kPrimFloat: case Primitive::kPrimDouble: LOG(FATAL) << "Not yet implemented neg type " << neg->GetResultType(); diff --git a/compiler/utils/x86_64/assembler_x86_64.cc b/compiler/utils/x86_64/assembler_x86_64.cc index f4c9862..f164138 100644 --- a/compiler/utils/x86_64/assembler_x86_64.cc +++ b/compiler/utils/x86_64/assembler_x86_64.cc @@ -1340,6 +1340,13 @@ void X86_64Assembler::negl(CpuRegister reg) { EmitOperand(3, Operand(reg)); } +void X86_64Assembler::negq(CpuRegister reg) { + AssemblerBuffer::EnsureCapacity ensured(&buffer_); + EmitRex64(reg); + EmitUint8(0xF7); + EmitOperand(3, Operand(reg)); +} + void X86_64Assembler::notl(CpuRegister reg) { AssemblerBuffer::EnsureCapacity ensured(&buffer_); diff --git a/compiler/utils/x86_64/assembler_x86_64.h b/compiler/utils/x86_64/assembler_x86_64.h index 92b81ec..ec29271 100644 --- a/compiler/utils/x86_64/assembler_x86_64.h +++ b/compiler/utils/x86_64/assembler_x86_64.h @@ -450,6 +450,7 @@ class X86_64Assembler FINAL : public Assembler { void shrq(CpuRegister reg, const Immediate& imm); void negl(CpuRegister reg); + void negq(CpuRegister reg); void notl(CpuRegister reg); void enter(const Immediate& imm); |