Commit 3552f87e authored by paul.lind's avatar paul.lind Committed by Commit bot

MIPS: Use signaling NaN for holes in fixed double arrays.

Port 9eace97b
Port 5d641ec9

BUG=

Review URL: https://codereview.chromium.org/867453002

Cr-Commit-Position: refs/heads/master@{#26208}
parent 76193749
...@@ -771,15 +771,16 @@ void ElementsTransitionGenerator::GenerateSmiToDouble( ...@@ -771,15 +771,16 @@ void ElementsTransitionGenerator::GenerateSmiToDouble(
// Repurpose registers no longer in use. // Repurpose registers no longer in use.
Register hole_lower = elements; Register hole_lower = elements;
Register hole_upper = length; Register hole_upper = length;
__ li(hole_lower, Operand(kHoleNanLower32)); __ li(hole_lower, Operand(kHoleNanLower32));
__ li(hole_upper, Operand(kHoleNanUpper32));
// scratch1: begin of source FixedArray element fields, not tagged // scratch1: begin of source FixedArray element fields, not tagged
// hole_lower: kHoleNanLower32 // hole_lower: kHoleNanLower32
// hole_upper: kHoleNanUpper32 // hole_upper: kHoleNanUpper32
// array_end: end of destination FixedDoubleArray, not tagged // array_end: end of destination FixedDoubleArray, not tagged
// scratch3: begin of FixedDoubleArray element fields, not tagged // scratch3: begin of FixedDoubleArray element fields, not tagged
__ Branch(USE_DELAY_SLOT, &entry);
__ li(hole_upper, Operand(kHoleNanUpper32)); // In delay slot. __ Branch(&entry);
__ bind(&only_change_map); __ bind(&only_change_map);
__ sw(target_map, FieldMemOperand(receiver, HeapObject::kMapOffset)); __ sw(target_map, FieldMemOperand(receiver, HeapObject::kMapOffset));
......
...@@ -1744,6 +1744,20 @@ void LCodeGen::DoConstantS(LConstantS* instr) { ...@@ -1744,6 +1744,20 @@ void LCodeGen::DoConstantS(LConstantS* instr) {
void LCodeGen::DoConstantD(LConstantD* instr) { void LCodeGen::DoConstantD(LConstantD* instr) {
DCHECK(instr->result()->IsDoubleRegister()); DCHECK(instr->result()->IsDoubleRegister());
DoubleRegister result = ToDoubleRegister(instr->result()); DoubleRegister result = ToDoubleRegister(instr->result());
#if V8_HOST_ARCH_IA32
// Need some crappy work-around for x87 sNaN -> qNaN breakage in simulator
// builds.
uint64_t bits = instr->bits();
if ((bits & V8_UINT64_C(0x7FF8000000000000)) ==
V8_UINT64_C(0x7FF0000000000000)) {
uint32_t lo = static_cast<uint32_t>(bits);
uint32_t hi = static_cast<uint32_t>(bits >> 32);
__ li(at, Operand(lo));
__ li(scratch0(), Operand(hi));
__ Move(result, at, scratch0());
return;
}
#endif
double v = instr->value(); double v = instr->value();
__ Move(result, v); __ Move(result, v);
} }
......
...@@ -1323,6 +1323,7 @@ class LConstantD FINAL : public LTemplateInstruction<1, 0, 0> { ...@@ -1323,6 +1323,7 @@ class LConstantD FINAL : public LTemplateInstruction<1, 0, 0> {
DECLARE_HYDROGEN_ACCESSOR(Constant) DECLARE_HYDROGEN_ACCESSOR(Constant)
double value() const { return hydrogen()->DoubleValue(); } double value() const { return hydrogen()->DoubleValue(); }
uint64_t bits() const { return hydrogen()->DoubleValueAsBits(); }
}; };
......
...@@ -3917,7 +3917,7 @@ void MacroAssembler::StoreNumberToDoubleElements(Register value_reg, ...@@ -3917,7 +3917,7 @@ void MacroAssembler::StoreNumberToDoubleElements(Register value_reg,
// Check for nan: all NaN values have a value greater (signed) than 0x7ff00000 // Check for nan: all NaN values have a value greater (signed) than 0x7ff00000
// in the exponent. // in the exponent.
li(scratch1, Operand(kNaNOrInfinityLowerBoundUpper32)); li(scratch1, Operand(kHoleNanUpper32 & HeapNumber::kExponentMask));
lw(exponent_reg, FieldMemOperand(value_reg, HeapNumber::kExponentOffset)); lw(exponent_reg, FieldMemOperand(value_reg, HeapNumber::kExponentOffset));
Branch(&maybe_nan, ge, exponent_reg, Operand(scratch1)); Branch(&maybe_nan, ge, exponent_reg, Operand(scratch1));
......
...@@ -665,13 +665,15 @@ void ElementsTransitionGenerator::GenerateSmiToDouble( ...@@ -665,13 +665,15 @@ void ElementsTransitionGenerator::GenerateSmiToDouble(
Register hole_lower = elements; Register hole_lower = elements;
Register hole_upper = length; Register hole_upper = length;
__ li(hole_lower, Operand(kHoleNanLower32)); __ li(hole_lower, Operand(kHoleNanLower32));
__ li(hole_upper, Operand(kHoleNanUpper32));
// scratch1: begin of source FixedArray element fields, not tagged // scratch1: begin of source FixedArray element fields, not tagged
// hole_lower: kHoleNanLower32 // hole_lower: kHoleNanLower32
// hole_upper: kHoleNanUpper32 // hole_upper: kHoleNanUpper32
// array_end: end of destination FixedDoubleArray, not tagged // array_end: end of destination FixedDoubleArray, not tagged
// scratch3: begin of FixedDoubleArray element fields, not tagged // scratch3: begin of FixedDoubleArray element fields, not tagged
__ Branch(USE_DELAY_SLOT, &entry);
__ li(hole_upper, Operand(kHoleNanUpper32)); // In delay slot. __ Branch(&entry);
__ bind(&only_change_map); __ bind(&only_change_map);
__ sd(target_map, FieldMemOperand(receiver, HeapObject::kMapOffset)); __ sd(target_map, FieldMemOperand(receiver, HeapObject::kMapOffset));
......
...@@ -3884,7 +3884,7 @@ void MacroAssembler::StoreNumberToDoubleElements(Register value_reg, ...@@ -3884,7 +3884,7 @@ void MacroAssembler::StoreNumberToDoubleElements(Register value_reg,
// Check for nan: all NaN values have a value greater (signed) than 0x7ff00000 // Check for nan: all NaN values have a value greater (signed) than 0x7ff00000
// in the exponent. // in the exponent.
li(scratch1, Operand(kNaNOrInfinityLowerBoundUpper32)); li(scratch1, Operand(kHoleNanUpper32 & HeapNumber::kExponentMask));
lw(exponent_reg, FieldMemOperand(value_reg, HeapNumber::kExponentOffset)); lw(exponent_reg, FieldMemOperand(value_reg, HeapNumber::kExponentOffset));
Branch(&maybe_nan, ge, exponent_reg, Operand(scratch1)); Branch(&maybe_nan, ge, exponent_reg, Operand(scratch1));
......
...@@ -150,8 +150,7 @@ static void TestNaN(const char *code) { ...@@ -150,8 +150,7 @@ static void TestNaN(const char *code) {
double value = a->get_scalar(0); double value = a->get_scalar(0);
CHECK(std::isnan(value) && CHECK(std::isnan(value) &&
bit_cast<uint64_t>(value) == bit_cast<uint64_t>(value) ==
bit_cast<uint64_t>( bit_cast<uint64_t>(std::numeric_limits<double>::quiet_NaN()));
i::FixedDoubleArray::canonical_not_the_hole_nan_as_double()));
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment