2011-07-05 11:54:11 +00:00
|
|
|
// Copyright 2011 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
#ifndef V8_CONVERSIONS_INL_H_
|
|
|
|
#define V8_CONVERSIONS_INL_H_
|
|
|
|
|
2014-12-20 13:17:20 +00:00
|
|
|
#include <float.h> // Required for DBL_MAX and on Win32 for finite()
|
|
|
|
#include <limits.h> // Required for INT_MAX etc.
|
|
|
|
#include <stdarg.h>
|
2013-04-19 13:26:47 +00:00
|
|
|
#include <cmath>
|
2014-12-20 13:17:20 +00:00
|
|
|
#include "src/globals.h" // Required for V8_INFINITY
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Extra POSIX/ANSI functions for Win32/MSVC.
|
|
|
|
|
2014-09-02 13:36:35 +00:00
|
|
|
#include "src/base/bits.h"
|
2014-06-30 13:25:46 +00:00
|
|
|
#include "src/base/platform/platform.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/conversions.h"
|
|
|
|
#include "src/double.h"
|
2015-08-13 14:02:22 +00:00
|
|
|
#include "src/objects-inl.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-03-03 13:44:20 +00:00
|
|
|
// The fast double-to-unsigned-int conversion routine does not guarantee
|
2010-03-16 12:20:52 +00:00
|
|
|
// rounding towards zero, or any reasonable value if the argument is larger
|
|
|
|
// than what fits in an unsigned 32-bit integer.
|
2011-11-29 10:56:11 +00:00
|
|
|
inline unsigned int FastD2UI(double x) {
|
2010-03-03 13:44:20 +00:00
|
|
|
// There is no unsigned version of lrint, so there is no fast path
|
|
|
|
// in this function as there is in FastD2I. Using lrint doesn't work
|
|
|
|
// for values of 2^31 and above.
|
|
|
|
|
|
|
|
// Convert "small enough" doubles to uint32_t by fixing the 32
|
|
|
|
// least significant non-fractional bits in the low 32 bits of the
|
|
|
|
// double, and reading them from there.
|
|
|
|
const double k2Pow52 = 4503599627370496.0;
|
|
|
|
bool negative = x < 0;
|
|
|
|
if (negative) {
|
|
|
|
x = -x;
|
|
|
|
}
|
|
|
|
if (x < k2Pow52) {
|
|
|
|
x += k2Pow52;
|
|
|
|
uint32_t result;
|
2014-04-15 16:39:21 +00:00
|
|
|
#ifndef V8_TARGET_BIG_ENDIAN
|
2010-03-16 12:20:52 +00:00
|
|
|
Address mantissa_ptr = reinterpret_cast<Address>(&x);
|
2014-04-15 16:39:21 +00:00
|
|
|
#else
|
2017-01-17 16:38:49 +00:00
|
|
|
Address mantissa_ptr = reinterpret_cast<Address>(&x) + kInt32Size;
|
2014-04-15 16:39:21 +00:00
|
|
|
#endif
|
2010-03-16 12:20:52 +00:00
|
|
|
// Copy least significant 32 bits of mantissa.
|
2014-05-27 12:02:44 +00:00
|
|
|
memcpy(&result, mantissa_ptr, sizeof(result));
|
2010-03-03 13:44:20 +00:00
|
|
|
return negative ? ~result + 1 : result;
|
|
|
|
}
|
|
|
|
// Large number (outside uint32 range), Infinity or NaN.
|
|
|
|
return 0x80000000u; // Return integer indefinite.
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-22 11:42:10 +00:00
|
|
|
inline float DoubleToFloat32(double x) {
|
2015-10-15 16:03:20 +00:00
|
|
|
// TODO(yangguo): This static_cast is implementation-defined behaviour in C++,
|
2014-09-22 11:42:10 +00:00
|
|
|
// so we may need to do the conversion manually instead to match the spec.
|
|
|
|
volatile float f = static_cast<float>(x);
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-29 10:56:11 +00:00
|
|
|
inline double DoubleToInteger(double x) {
|
2013-04-19 13:26:47 +00:00
|
|
|
if (std::isnan(x)) return 0;
|
|
|
|
if (!std::isfinite(x) || x == 0) return x;
|
2014-01-14 09:57:05 +00:00
|
|
|
return (x >= 0) ? std::floor(x) : std::ceil(x);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int32_t DoubleToInt32(double x) {
|
|
|
|
int32_t i = FastD2I(x);
|
|
|
|
if (FastI2D(i) == x) return i;
|
2011-09-07 12:39:53 +00:00
|
|
|
Double d(x);
|
|
|
|
int exponent = d.Exponent();
|
|
|
|
if (exponent < 0) {
|
|
|
|
if (exponent <= -Double::kSignificandSize) return 0;
|
|
|
|
return d.Sign() * static_cast<int32_t>(d.Significand() >> -exponent);
|
|
|
|
} else {
|
|
|
|
if (exponent > 31) return 0;
|
|
|
|
return d.Sign() * static_cast<int32_t>(d.Significand() << exponent);
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2016-07-21 09:18:47 +00:00
|
|
|
bool DoubleToSmiInteger(double value, int* smi_int_value) {
|
|
|
|
if (IsMinusZero(value)) return false;
|
|
|
|
int i = FastD2IChecked(value);
|
|
|
|
if (value != i || !Smi::IsValid(i)) return false;
|
|
|
|
*smi_int_value = i;
|
|
|
|
return true;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2015-08-13 14:02:22 +00:00
|
|
|
bool IsSmiDouble(double value) {
|
|
|
|
return !IsMinusZero(value) && value >= Smi::kMinValue &&
|
|
|
|
value <= Smi::kMaxValue && value == FastI2D(FastD2I(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IsInt32Double(double value) {
|
|
|
|
return !IsMinusZero(value) && value >= kMinInt && value <= kMaxInt &&
|
|
|
|
value == FastI2D(FastD2I(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool IsUint32Double(double value) {
|
|
|
|
return !IsMinusZero(value) && value >= 0 && value <= kMaxUInt32 &&
|
|
|
|
value == FastUI2D(FastD2UI(value));
|
|
|
|
}
|
|
|
|
|
2017-01-16 11:25:05 +00:00
|
|
|
bool DoubleToUint32IfEqualToSelf(double value, uint32_t* uint32_value) {
|
2017-01-17 16:38:49 +00:00
|
|
|
const double k2Pow52 = 4503599627370496.0;
|
|
|
|
const uint32_t kValidTopBits = 0x43300000;
|
|
|
|
const uint64_t kBottomBitMask = V8_2PART_UINT64_C(0x00000000, FFFFFFFF);
|
|
|
|
|
|
|
|
// Add 2^52 to the double, to place valid uint32 values in the low-significant
|
|
|
|
// bits of the exponent, by effectively setting the (implicit) top bit of the
|
|
|
|
// significand. Note that this addition also normalises 0.0 and -0.0.
|
|
|
|
double shifted_value = value + k2Pow52;
|
|
|
|
|
|
|
|
// At this point, a valid uint32 valued double will be represented as:
|
|
|
|
//
|
|
|
|
// sign = 0
|
|
|
|
// exponent = 52
|
|
|
|
// significand = 1. 00...00 <value>
|
|
|
|
// implicit^ ^^^^^^^ 32 bits
|
|
|
|
// ^^^^^^^^^^^^^^^ 52 bits
|
|
|
|
//
|
|
|
|
// Therefore, we can first check the top 32 bits to make sure that the sign,
|
|
|
|
// exponent and remaining significand bits are valid, and only then check the
|
|
|
|
// value in the bottom 32 bits.
|
|
|
|
|
|
|
|
uint64_t result = bit_cast<uint64_t>(shifted_value);
|
|
|
|
if ((result >> 32) == kValidTopBits) {
|
|
|
|
*uint32_value = result & kBottomBitMask;
|
|
|
|
return FastUI2D(result & kBottomBitMask) == value;
|
2017-01-16 11:25:05 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2015-08-13 14:02:22 +00:00
|
|
|
|
|
|
|
int32_t NumberToInt32(Object* number) {
|
2017-07-10 12:58:27 +00:00
|
|
|
if (number->IsSmi()) return Smi::ToInt(number);
|
2015-08-13 14:02:22 +00:00
|
|
|
return DoubleToInt32(number->Number());
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t NumberToUint32(Object* number) {
|
2017-07-10 12:58:27 +00:00
|
|
|
if (number->IsSmi()) return Smi::ToInt(number);
|
2015-08-13 14:02:22 +00:00
|
|
|
return DoubleToUint32(number->Number());
|
|
|
|
}
|
|
|
|
|
2016-12-19 12:06:58 +00:00
|
|
|
uint32_t PositiveNumberToUint32(Object* number) {
|
|
|
|
if (number->IsSmi()) {
|
2017-07-10 12:58:27 +00:00
|
|
|
int value = Smi::ToInt(number);
|
2016-12-19 12:06:58 +00:00
|
|
|
if (value <= 0) return 0;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
DCHECK(number->IsHeapNumber());
|
|
|
|
double value = number->Number();
|
|
|
|
// Catch all values smaller than 1 and use the double-negation trick for NANs.
|
|
|
|
if (!(value >= 1)) return 0;
|
|
|
|
uint32_t max = std::numeric_limits<uint32_t>::max();
|
|
|
|
if (value < max) return static_cast<uint32_t>(value);
|
|
|
|
return max;
|
|
|
|
}
|
|
|
|
|
2016-02-06 18:10:08 +00:00
|
|
|
int64_t NumberToInt64(Object* number) {
|
2017-07-10 12:58:27 +00:00
|
|
|
if (number->IsSmi()) return Smi::ToInt(number);
|
2017-08-03 15:39:40 +00:00
|
|
|
double d = number->Number();
|
|
|
|
if (std::isnan(d)) return 0;
|
|
|
|
if (d >= static_cast<double>(std::numeric_limits<int64_t>::max())) {
|
|
|
|
return std::numeric_limits<int64_t>::max();
|
|
|
|
}
|
|
|
|
if (d <= static_cast<double>(std::numeric_limits<int64_t>::min())) {
|
|
|
|
return std::numeric_limits<int64_t>::min();
|
|
|
|
}
|
|
|
|
return static_cast<int64_t>(d);
|
2016-02-06 18:10:08 +00:00
|
|
|
}
|
2015-08-13 14:02:22 +00:00
|
|
|
|
2016-08-08 15:54:35 +00:00
|
|
|
bool TryNumberToSize(Object* number, size_t* result) {
|
2016-06-29 10:27:36 +00:00
|
|
|
// Do not create handles in this function! Don't use SealHandleScope because
|
|
|
|
// the function can be used concurrently.
|
2015-08-13 14:02:22 +00:00
|
|
|
if (number->IsSmi()) {
|
2017-07-10 12:58:27 +00:00
|
|
|
int value = Smi::ToInt(number);
|
2015-08-13 14:02:22 +00:00
|
|
|
DCHECK(static_cast<unsigned>(Smi::kMaxValue) <=
|
|
|
|
std::numeric_limits<size_t>::max());
|
|
|
|
if (value >= 0) {
|
|
|
|
*result = static_cast<size_t>(value);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
} else {
|
|
|
|
DCHECK(number->IsHeapNumber());
|
|
|
|
double value = HeapNumber::cast(number)->value();
|
2016-12-08 09:19:52 +00:00
|
|
|
// If value is compared directly to the limit, the limit will be
|
|
|
|
// casted to a double and could end up as limit + 1,
|
|
|
|
// because a double might not have enough mantissa bits for it.
|
|
|
|
// So we might as well cast the limit first, and use < instead of <=.
|
|
|
|
double maxSize = static_cast<double>(std::numeric_limits<size_t>::max());
|
|
|
|
if (value >= 0 && value < maxSize) {
|
2015-08-13 14:02:22 +00:00
|
|
|
*result = static_cast<size_t>(value);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-08 15:54:35 +00:00
|
|
|
size_t NumberToSize(Object* number) {
|
2015-08-13 14:02:22 +00:00
|
|
|
size_t result = 0;
|
2016-08-08 15:54:35 +00:00
|
|
|
bool is_valid = TryNumberToSize(number, &result);
|
2015-08-13 14:02:22 +00:00
|
|
|
CHECK(is_valid);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-08-13 15:30:04 +00:00
|
|
|
uint32_t DoubleToUint32(double x) {
|
|
|
|
return static_cast<uint32_t>(DoubleToInt32(x));
|
|
|
|
}
|
|
|
|
|
2015-09-30 13:46:56 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
#endif // V8_CONVERSIONS_INL_H_
|