Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(102)

Unified Diff: src/compiler/x64/code-generator-x64.cc

Issue 1584663007: [turbofan] Implement rounding of floats on x64 and ia32 without sse4.1. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Created 4 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: src/compiler/x64/code-generator-x64.cc
diff --git a/src/compiler/x64/code-generator-x64.cc b/src/compiler/x64/code-generator-x64.cc
index c9ce8d7f193716ebeb676a8315a3927ab3187ddd..c9de5428183a22603c57540b100766ef9276eb6c 100644
--- a/src/compiler/x64/code-generator-x64.cc
+++ b/src/compiler/x64/code-generator-x64.cc
@@ -941,10 +941,53 @@ void CodeGenerator::AssembleArchInstruction(Instruction* instr) {
ASSEMBLE_SSE_UNOP(Cvtss2sd);
break;
case kSSEFloat32Round: {
- CpuFeatureScope sse_scope(masm(), SSE4_1);
RoundingMode const mode =
static_cast<RoundingMode>(MiscField::decode(instr->opcode()));
- __ Roundss(i.OutputDoubleRegister(), i.InputDoubleRegister(0), mode);
+ if (CpuFeatures::IsSupported(SSE4_1)) {
+ CpuFeatureScope sse_scope(masm(), SSE4_1);
+ __ Roundss(i.OutputDoubleRegister(), i.InputDoubleRegister(0), mode);
+ } else {
+ // Set the rounding mode.
+ // We have to store the original rounding mode in memory.
+ __ subq(rsp, Immediate(kPointerSize * 2));
+ __ stmxcsr(Operand(rsp, 0));
+ __ movl(kScratchRegister, Operand(rsp, 0));
+ __ andl(kScratchRegister, Immediate(0xffff9fff));
+ __ orl(kScratchRegister, Immediate(mode << 13));
+ __ movl(Operand(rsp, kPointerSize), kScratchRegister);
+ __ ldmxcsr(Operand(rsp, kPointerSize));
+
+ // Start with the conversion.
+ if (instr->InputAt(0)->IsDoubleRegister()) {
+ __ Cvtss2siq(kScratchRegister, i.InputDoubleRegister(0));
+ } else {
+ __ Cvtss2siq(kScratchRegister, i.InputOperand(0));
+ }
+
+ Label out_of_range;
+ Label done;
+ __ cmpq(kScratchRegister, Immediate(1));
+ // If the conversion results in INT64_MIN, then the input is out of
+ // int64 range.
+ __ j(overflow, &out_of_range);
+ // Rounding is done by converting the value back to float.
+ __ Cvtqsi2ss(i.OutputDoubleRegister(), kScratchRegister);
+ __ jmp(&done);
+
+ __ bind(&out_of_range);
+ // The input is out of int64 range. Fractions are not possible for
+ // inputs out of int64 range, so we don't have to do any rounding.
+ if (instr->InputAt(0)->IsDoubleRegister()) {
+ __ movss(i.OutputDoubleRegister(), i.InputDoubleRegister(0));
+ } else {
+ __ movss(i.OutputDoubleRegister(), i.InputOperand(0));
+ }
+
+ __ bind(&done);
+ // Restore the original rounding mode.
+ __ ldmxcsr(Operand(rsp, 0));
+ __ addq(rsp, Immediate(kPointerSize * 2));
+ }
break;
}
case kSSEFloat64Cmp:
@@ -1021,10 +1064,53 @@ void CodeGenerator::AssembleArchInstruction(Instruction* instr) {
ASSEMBLE_SSE_UNOP(sqrtsd);
break;
case kSSEFloat64Round: {
- CpuFeatureScope sse_scope(masm(), SSE4_1);
RoundingMode const mode =
static_cast<RoundingMode>(MiscField::decode(instr->opcode()));
- __ Roundsd(i.OutputDoubleRegister(), i.InputDoubleRegister(0), mode);
+ if (CpuFeatures::IsSupported(SSE4_1)) {
+ CpuFeatureScope sse_scope(masm(), SSE4_1);
+ __ Roundsd(i.OutputDoubleRegister(), i.InputDoubleRegister(0), mode);
+ } else {
+ // Set the rounding mode.
+ // We have to store the original rounding mode in memory.
+ __ subq(rsp, Immediate(kPointerSize * 2));
+ __ stmxcsr(Operand(rsp, 0));
+ __ movl(kScratchRegister, Operand(rsp, 0));
+ __ andl(kScratchRegister, Immediate(0xffff9fff));
+ __ orl(kScratchRegister, Immediate(mode << 13));
+ __ movl(Operand(rsp, kPointerSize), kScratchRegister);
+ __ ldmxcsr(Operand(rsp, kPointerSize));
+
+ // Start with the conversion.
+ if (instr->InputAt(0)->IsDoubleRegister()) {
+ __ Cvtsd2siq(kScratchRegister, i.InputDoubleRegister(0));
+ } else {
+ __ Cvtsd2siq(kScratchRegister, i.InputOperand(0));
+ }
+
+ Label out_of_range;
+ Label done;
+ __ cmpq(kScratchRegister, Immediate(1));
+ // If the conversion results in INT64_MIN, but the input was not
+ // INT64_MIN, then the conversion fails.
+ __ j(overflow, &out_of_range);
+ __ Cvtqsi2sd(i.OutputDoubleRegister(), kScratchRegister);
+ __ jmp(&done);
+
+ __ bind(&out_of_range);
+ // The input is out of int64 range. Fractions are not possible for
+ // inputs
+ // out of int64 range, so we don't have to do any rounding.
+ if (instr->InputAt(0)->IsDoubleRegister()) {
+ __ movsd(i.OutputDoubleRegister(), i.InputDoubleRegister(0));
+ } else {
+ __ movsd(i.OutputDoubleRegister(), i.InputOperand(0));
+ }
+
+ __ bind(&done);
+ // Restore the original rounding mode.
+ __ ldmxcsr(Operand(rsp, 0));
+ __ addq(rsp, Immediate(kPointerSize * 2));
+ }
break;
}
case kSSEFloat64ToFloat32:

Powered by Google App Engine
This is Rietveld 408576698