6c9fe8ab5b
JerryScript CI / Checks (push) Failing after 22s
JerryScript CI / Linux_x86_cpointer-32bit_Build_Correctness_Tests (push) Failing after 14s
JerryScript CI / Linux_x86-64_Build_Option_Tests (push) Failing after 15s
JerryScript CI / Conformance_Tests_ESNext (push) Failing after 10s
JerryScript CI / ASAN_Tests_Debug (push) Failing after 14s
JerryScript CI / UBSAN_Tests (push) Failing after 16s
JerryScript CI / Linux_ARMv7l_Tests (push) Failing after 14s
JerryScript CI / Linux_ARMv7l_Tests_Debug (push) Failing after 15s
JerryScript CI / Linux_AArch64_Tests (push) Failing after 14s
JerryScript CI / Linux_AArch64_Tests_Debug (push) Failing after 15s
JerryScript CI / MbedOS_K64F_Build_Test (push) Failing after 14s
JerryScript CI / ESP8266_RTOS_SDK_Build_Test (push) Failing after 22s
JerryScript CI / ESP_IDF_Build_Test (push) Failing after 1m26s
JerryScript CI / Notification (push) Has been skipped
JerryScript CI / Linux_x86-64_Build_Correctness_Debugger_Tests (push) Failing after 9s
JerryScript CI / Conformance_Tests_ESNext_Debug_A (push) Failing after 10s
JerryScript CI / Conformance_Tests_ESNext_Debug_B (push) Failing after 10s
JerryScript CI / Unit_Tests (push) Failing after 8s
JerryScript CI / Clang_Unit_Build_Option_Tests (push) Failing after 15s
JerryScript CI / ASAN_Tests (push) Failing after 15s
JerryScript CI / Zephyr_STM32F4_Build_Test (push) Failing after 14s
JerryScript CI / NuttX_STM32F4_Build_Test (push) Failing after 14s
JerryScript CI / RIOT_STM32F4_Build_Test (push) Failing after 6s
JerryScript CI / Win_x86-64_Conformance_Tests_ESNext (push) Has been cancelled
JerryScript CI / Win_x86-64_Conformance_Tests_ESNext_Debug (push) Has been cancelled
JerryScript CI / Win_x86-64_Tests-MINGW (push) Has been cancelled
JerryScript CI / Win_x86-64_Tests (push) Has been cancelled
JerryScript CI / Win_x86-64_Tests_Debug (push) Has been cancelled
JerryScript CI / OSX_x86-64_Build_Correctness_Unit_Tests (push) Has been cancelled
JerryScript CI / OSX_x86-64_Build_Correctness_Unit_Tests_Debug (push) Has been cancelled
991 lines
25 KiB
C
991 lines
25 KiB
C
/* Copyright JS Foundation and other contributors, http://js.foundation
|
|
*
|
|
* 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 <math.h>
|
|
|
|
#include "ecma-globals.h"
|
|
#include "ecma-helpers-number.h"
|
|
#include "ecma-helpers.h"
|
|
|
|
#include "jrt-libc-includes.h"
|
|
#include "lit-char-helpers.h"
|
|
#include "lit-magic-strings.h"
|
|
|
|
/** \addtogroup ecma ECMA
|
|
* @{
|
|
*
|
|
* \addtogroup ecmahelpers Helpers for operations with ECMA data types
|
|
* @{
|
|
*/
|
|
|
|
#if JERRY_NUMBER_TYPE_FLOAT64
|
|
|
|
/**
|
|
* \addtogroup ecmahelpersbigintegers Helpers for operations intermediate 128-bit integers
|
|
* @{
|
|
*/
|
|
|
|
/**
|
|
* 128-bit integer type
|
|
*/
|
|
typedef struct
|
|
{
|
|
uint64_t hi; /**< high 64 bits */
|
|
uint64_t lo; /**< low 64 bits */
|
|
} ecma_uint128_t;
|
|
|
|
/**
|
|
* Round high part of 128-bit integer to uint64_t
|
|
*
|
|
* @return rounded high to uint64_t
|
|
*/
|
|
static uint64_t
|
|
ecma_round_high_to_uint64 (ecma_uint128_t *num_p) /**< 128-bit unsigned integer */
|
|
{
|
|
uint64_t masked_lo = num_p->lo & ~(1ULL << 63u);
|
|
uint64_t masked_hi = num_p->hi & 0x1;
|
|
|
|
if ((num_p->lo >> 63u != 0) && (masked_lo > 0 || masked_hi != 0))
|
|
{
|
|
return (num_p->hi + 1);
|
|
}
|
|
|
|
return num_p->hi;
|
|
} /* ecma_round_high_to_uint64 */
|
|
|
|
/**
|
|
* Left shift 128-bit integer by max 63 bits.
|
|
*/
|
|
static void JERRY_ATTR_ALWAYS_INLINE
|
|
ecma_uint128_shift_left (ecma_uint128_t *num_p, /**< 128-bit unsigned integer */
|
|
int32_t shift) /**< left shift count */
|
|
{
|
|
num_p->hi = (num_p->hi << shift) | (num_p->lo >> (64 - shift));
|
|
num_p->lo <<= shift;
|
|
} /* ecma_uint128_shift_left */
|
|
|
|
/**
|
|
* Right shift 128-bit integer by max 63 bits.
|
|
*/
|
|
static void JERRY_ATTR_ALWAYS_INLINE
|
|
ecma_uint128_shift_right (ecma_uint128_t *num_p, /**< 128-bit unsigned integer */
|
|
int32_t shift) /**< right shift count */
|
|
{
|
|
num_p->lo = (num_p->lo >> shift) | (num_p->hi << (64 - shift));
|
|
num_p->hi >>= shift;
|
|
} /* ecma_uint128_shift_right */
|
|
|
|
/**
|
|
* Add two 128-bit integer values and assign the result to the left one.
|
|
*/
|
|
static void
|
|
ecma_uint128_add (ecma_uint128_t *left_p, /**< left 128-bit unsigned integer */
|
|
ecma_uint128_t *right_p) /**< right 128-bit unsigned integer */
|
|
{
|
|
left_p->hi += right_p->hi;
|
|
left_p->lo += right_p->lo;
|
|
|
|
if (left_p->lo < right_p->lo)
|
|
{
|
|
left_p->hi++;
|
|
}
|
|
} /* ecma_uint128_add */
|
|
|
|
/**
|
|
* Multiply 128-bit integer by 10
|
|
*/
|
|
static void
|
|
ecma_uint128_mul10 (ecma_uint128_t *num_p) /**< 128-bit unsigned integer */
|
|
{
|
|
ecma_uint128_shift_left (num_p, 1u);
|
|
|
|
ecma_uint128_t tmp = { .hi = num_p->hi, .lo = num_p->lo };
|
|
ecma_uint128_shift_left (&tmp, 2u);
|
|
|
|
ecma_uint128_add (num_p, &tmp);
|
|
} /* ecma_uint128_mul10 */
|
|
|
|
/**
|
|
* Divide 128-bit integer by 10
|
|
*
|
|
* N = N3 *2^96 + N2 *2^64 + N1 *2^32 + N0 *2^0 // 128-bit dividend
|
|
* T = T3 *2^-32 + T2 *2^-64 + T1 *2^-96 + T0 *2^-128 // 128-bit divisor reciprocal, 1/10 * 2^-128
|
|
*
|
|
* N * T = N3*T3 *2^64 + N2*T3 *2^32 + N1*T3 *2^0 + N0*T3 *2^-32
|
|
* + N3*T2 *2^32 + N2*T2 *2^0 + N1*T2 *2^-32 + N0*T2 *2^-64
|
|
* + N3*T1 *2^0 + N2*T1 *2^-32 + N1*T1 *2^-64 + N0*T1 *2^-96
|
|
* + N3*T0 *2^-32 + N2*T0 *2^-64 + N1*T0 *2^-96 + N0*T0 *2^-128
|
|
*
|
|
* Q3=carry Q2=^+carry Q1=^+carry Q0=^+carry fraction=^...
|
|
*
|
|
* Q = Q3 *2^96 + Q2 *2^64 + Q1 *2^32 + Q0 *2^0 // 128-bit quotient
|
|
*/
|
|
static void
|
|
ecma_uint128_div10 (ecma_uint128_t *num_p) /**< 128-bit unsigned integer */
|
|
{
|
|
/* estimation of reciprocal of 10, 128 bits right of the binary point (T1 == T2) */
|
|
const uint64_t tenth_l = 0x9999999aul;
|
|
const uint64_t tenth_m = 0x99999999ul;
|
|
const uint64_t tenth_h = 0x19999999ul;
|
|
|
|
const uint64_t l0 = ((uint32_t) num_p->lo) * tenth_l;
|
|
const uint64_t l1 = (num_p->lo >> 32u) * tenth_l;
|
|
const uint64_t l2 = ((uint32_t) num_p->hi) * tenth_l;
|
|
const uint64_t l3 = (num_p->hi >> 32u) * tenth_l;
|
|
const uint64_t m0 = ((uint32_t) num_p->lo) * tenth_m;
|
|
const uint64_t m1 = (num_p->lo >> 32u) * tenth_m;
|
|
const uint64_t m2 = ((uint32_t) num_p->hi) * tenth_m;
|
|
const uint64_t m3 = (num_p->hi >> 32u) * tenth_m;
|
|
const uint64_t h0 = ((uint32_t) num_p->lo) * tenth_h;
|
|
const uint64_t h1 = (num_p->lo >> 32u) * tenth_h;
|
|
const uint64_t h2 = ((uint32_t) num_p->hi) * tenth_h;
|
|
const uint64_t h3 = (num_p->hi >> 32u) * tenth_h;
|
|
|
|
uint64_t q0 = l0 >> 32u;
|
|
q0 += (uint32_t) l1;
|
|
q0 += (uint32_t) m0;
|
|
|
|
q0 >>= 32u;
|
|
q0 += l1 >> 32u;
|
|
q0 += m0 >> 32u;
|
|
q0 += (uint32_t) l2;
|
|
q0 += (uint32_t) m1;
|
|
q0 += (uint32_t) m0;
|
|
|
|
q0 >>= 32u;
|
|
q0 += l2 >> 32u;
|
|
q0 += m1 >> 32u;
|
|
q0 += m0 >> 32u;
|
|
q0 += (uint32_t) l3;
|
|
q0 += (uint32_t) m2;
|
|
q0 += (uint32_t) m1;
|
|
q0 += (uint32_t) h0;
|
|
|
|
q0 >>= 32u;
|
|
q0 += l3 >> 32u;
|
|
q0 += m2 >> 32u;
|
|
q0 += m1 >> 32u;
|
|
q0 += h0 >> 32u;
|
|
q0 += (uint32_t) m3;
|
|
q0 += (uint32_t) m2;
|
|
q0 += (uint32_t) h1;
|
|
|
|
uint64_t q1 = q0 >> 32u;
|
|
q1 += m3 >> 32u;
|
|
q1 += m2 >> 32u;
|
|
q1 += h1 >> 32u;
|
|
q1 += (uint32_t) m3;
|
|
q1 += (uint32_t) h2;
|
|
|
|
uint64_t q32 = q1 >> 32u;
|
|
q32 += m3 >> 32u;
|
|
q32 += h2 >> 32u;
|
|
q32 += h3;
|
|
|
|
num_p->lo = (q1 << 32u) | ((uint32_t) q0);
|
|
num_p->hi = q32;
|
|
} /* ecma_uint128_div10 */
|
|
|
|
/**
|
|
* Count leading zeros in a 64-bit integer. The behaviour is undefined for 0.
|
|
*
|
|
* @return number of leading zeros.
|
|
*/
|
|
inline static int JERRY_ATTR_CONST
|
|
ecma_uint64_clz (uint64_t n) /**< integer to count leading zeros in */
|
|
{
|
|
#if defined(__GNUC__) || defined(__clang__)
|
|
return __builtin_clzll (n);
|
|
#else /* !defined (__GNUC__) && !defined (__clang__) */
|
|
JERRY_ASSERT (n != 0);
|
|
|
|
int cnt = 0;
|
|
uint64_t one = 0x8000000000000000ull;
|
|
|
|
while ((n & one) == 0)
|
|
{
|
|
cnt++;
|
|
one >>= 1;
|
|
}
|
|
|
|
return cnt;
|
|
#endif /* !defined (__GNUC__) && !defined (__clang__) */
|
|
} /* ecma_uint64_clz */
|
|
|
|
/**
|
|
* Count leading zeros in the top 4 bits of a 64-bit integer.
|
|
*
|
|
* @return number of leading zeros in top 4 bits.
|
|
*/
|
|
inline static int JERRY_ATTR_ALWAYS_INLINE JERRY_ATTR_CONST
|
|
ecma_uint64_clz_top4 (uint64_t n) /**< integer to count leading zeros in */
|
|
{
|
|
static const uint8_t ecma_uint4_clz[] = { 4, 3, 2, 2, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0 };
|
|
return ecma_uint4_clz[n >> 60];
|
|
} /* ecma_uint64_clz */
|
|
|
|
/**
|
|
* Shift required to clear 4 bits of a 64-bit integer.
|
|
*
|
|
* @return 0-4
|
|
*/
|
|
inline static int JERRY_ATTR_ALWAYS_INLINE JERRY_ATTR_CONST
|
|
ecma_uint64_normalize_shift (uint64_t n) /**< integer to count leading zeros in */
|
|
{
|
|
static const uint8_t ecma_uint4_shift[] = { 0, 1, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4, 4 };
|
|
|
|
return ecma_uint4_shift[n >> 60];
|
|
} /* ecma_uint64_normalize_shift */
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* Number.MAX_VALUE exponent part when using 64 bit float representation.
|
|
*/
|
|
#define NUMBER_MAX_DECIMAL_EXPONENT 308
|
|
/**
|
|
* Number.MIN_VALUE exponent part when using 64 bit float representation.
|
|
*/
|
|
#define NUMBER_MIN_DECIMAL_EXPONENT -324
|
|
|
|
#elif !JERRY_NUMBER_TYPE_FLOAT64
|
|
|
|
/**
|
|
* Number.MAX_VALUE exponent part when using 32 bit float representation.
|
|
*/
|
|
#define NUMBER_MAX_DECIMAL_EXPONENT 38
|
|
/**
|
|
* Number.MIN_VALUE exponent part when using 32 bit float representation.
|
|
*/
|
|
#define NUMBER_MIN_DECIMAL_EXPONENT -45
|
|
|
|
#endif /* JERRY_NUMBER_TYPE_FLOAT64 */
|
|
|
|
/**
|
|
* Value of epsilon
|
|
*/
|
|
#define EPSILON 0.0000001
|
|
|
|
/**
|
|
* ECMA-defined conversion from string to number for different radixes (2, 8, 16).
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5 9.3.1
|
|
* ECMA-262 v6 7.1.3.1
|
|
*
|
|
* @return NaN - if the conversion fails
|
|
* converted number - otherwise
|
|
*/
|
|
ecma_number_t
|
|
ecma_utf8_string_to_number_by_radix (const lit_utf8_byte_t *str_p, /**< utf-8 string */
|
|
const lit_utf8_size_t string_size, /**< end of utf-8 string */
|
|
uint32_t radix, /**< radix */
|
|
uint32_t options) /**< option flags */
|
|
{
|
|
JERRY_ASSERT (radix == 2 || radix == 8 || radix == 16);
|
|
JERRY_ASSERT (*str_p == LIT_CHAR_0);
|
|
|
|
const lit_utf8_byte_t *end_p = str_p + string_size;
|
|
|
|
/* Skip leading zero */
|
|
str_p++;
|
|
|
|
if (radix != 8 || LEXER_TO_ASCII_LOWERCASE (*str_p) == LIT_CHAR_LOWERCASE_O)
|
|
{
|
|
/* Skip radix specifier */
|
|
str_p++;
|
|
}
|
|
|
|
ecma_number_t num = ECMA_NUMBER_ZERO;
|
|
|
|
while (str_p < end_p)
|
|
{
|
|
lit_utf8_byte_t digit = *str_p++;
|
|
|
|
if (digit == LIT_CHAR_UNDERSCORE && (options & ECMA_CONVERSION_ALLOW_UNDERSCORE))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (!lit_char_is_hex_digit (digit))
|
|
{
|
|
return ecma_number_make_nan ();
|
|
}
|
|
|
|
uint32_t value = lit_char_hex_to_int (digit);
|
|
|
|
if (value >= radix)
|
|
{
|
|
return ecma_number_make_nan ();
|
|
}
|
|
|
|
num = num * (ecma_number_t) radix + (ecma_number_t) value;
|
|
}
|
|
|
|
return num;
|
|
} /* ecma_utf8_string_to_number_by_radix */
|
|
|
|
/**
|
|
* ECMA-defined conversion of string to Number.
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 9.3.1
|
|
*
|
|
* @return NaN - if the conversion fails
|
|
* converted number - otherwise
|
|
*/
|
|
ecma_number_t
|
|
ecma_utf8_string_to_number (const lit_utf8_byte_t *str_p, /**< utf-8 string */
|
|
lit_utf8_size_t str_size, /**< string size */
|
|
uint32_t options) /**< allowing underscore option bit */
|
|
{
|
|
ecma_string_trim_helper (&str_p, &str_size);
|
|
const lit_utf8_byte_t *end_p = str_p + str_size;
|
|
|
|
if (str_size == 0)
|
|
{
|
|
return ECMA_NUMBER_ZERO;
|
|
}
|
|
|
|
bool sign = false;
|
|
|
|
if (str_p + 2 < end_p && str_p[0] == LIT_CHAR_0)
|
|
{
|
|
uint8_t radix = lit_char_to_radix (str_p[1]);
|
|
|
|
if (radix != 10)
|
|
{
|
|
return ecma_utf8_string_to_number_by_radix (str_p, str_size, radix, options);
|
|
}
|
|
}
|
|
|
|
if (*str_p == LIT_CHAR_PLUS)
|
|
{
|
|
str_p++;
|
|
}
|
|
else if (*str_p == LIT_CHAR_MINUS)
|
|
{
|
|
sign = true;
|
|
str_p++;
|
|
}
|
|
|
|
/* Check if string is equal to "Infinity". */
|
|
const lit_utf8_byte_t *infinity_str_p = lit_get_magic_string_utf8 (LIT_MAGIC_STRING_INFINITY_UL);
|
|
const lit_utf8_size_t infinity_length = lit_get_magic_string_size (LIT_MAGIC_STRING_INFINITY_UL);
|
|
|
|
if ((lit_utf8_size_t) (end_p - str_p) == infinity_length && memcmp (infinity_str_p, str_p, infinity_length) == 0)
|
|
{
|
|
return ecma_number_make_infinity (sign);
|
|
}
|
|
|
|
uint64_t significand = 0;
|
|
uint32_t digit_count = 0;
|
|
int32_t decimal_exponent = 0;
|
|
bool has_significand = false;
|
|
|
|
/* Parsing integer part */
|
|
while (str_p < end_p)
|
|
{
|
|
if (*str_p == LIT_CHAR_UNDERSCORE && (options & ECMA_CONVERSION_ALLOW_UNDERSCORE))
|
|
{
|
|
str_p++;
|
|
continue;
|
|
}
|
|
|
|
if (!lit_char_is_decimal_digit (*str_p))
|
|
{
|
|
break;
|
|
}
|
|
|
|
has_significand = true;
|
|
uint32_t digit_value = (uint32_t) (*str_p++ - LIT_CHAR_0);
|
|
|
|
if (digit_count == 0 && digit_value == 0)
|
|
{
|
|
/* Leading zeros are omitted. */
|
|
continue;
|
|
}
|
|
|
|
if (digit_count >= ECMA_NUMBER_MAX_DIGITS)
|
|
{
|
|
decimal_exponent++;
|
|
continue;
|
|
}
|
|
|
|
significand = significand * 10 + digit_value;
|
|
digit_count++;
|
|
}
|
|
|
|
/* Parse fraction part */
|
|
if (str_p < end_p && *str_p == LIT_CHAR_DOT)
|
|
{
|
|
str_p++;
|
|
|
|
while (str_p < end_p)
|
|
{
|
|
if (*str_p == LIT_CHAR_UNDERSCORE && (options & ECMA_CONVERSION_ALLOW_UNDERSCORE))
|
|
{
|
|
str_p++;
|
|
continue;
|
|
}
|
|
|
|
if (!lit_char_is_decimal_digit (*str_p))
|
|
{
|
|
break;
|
|
}
|
|
|
|
has_significand = true;
|
|
uint32_t digit_value = (uint32_t) (*str_p++ - LIT_CHAR_0);
|
|
|
|
if (digit_count == 0 && digit_value == 0)
|
|
{
|
|
/* Leading zeros are omitted. */
|
|
decimal_exponent--;
|
|
continue;
|
|
}
|
|
|
|
if (digit_count < ECMA_NUMBER_MAX_DIGITS)
|
|
{
|
|
significand = significand * 10 + digit_value;
|
|
digit_count++;
|
|
decimal_exponent--;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Parsing exponent */
|
|
if (str_p < end_p && LEXER_TO_ASCII_LOWERCASE (*str_p) == LIT_CHAR_LOWERCASE_E)
|
|
{
|
|
str_p++;
|
|
|
|
int32_t exponent = 0;
|
|
int32_t exponent_sign = 1;
|
|
|
|
if (str_p >= end_p)
|
|
{
|
|
return ecma_number_make_nan ();
|
|
}
|
|
|
|
if (*str_p == LIT_CHAR_PLUS)
|
|
{
|
|
str_p++;
|
|
}
|
|
else if (*str_p == LIT_CHAR_MINUS)
|
|
{
|
|
exponent_sign = -1;
|
|
str_p++;
|
|
}
|
|
|
|
if (str_p >= end_p || !lit_char_is_decimal_digit (*str_p))
|
|
{
|
|
return ecma_number_make_nan ();
|
|
}
|
|
|
|
while (str_p < end_p)
|
|
{
|
|
if (*str_p == LIT_CHAR_UNDERSCORE && (options & ECMA_CONVERSION_ALLOW_UNDERSCORE))
|
|
{
|
|
str_p++;
|
|
continue;
|
|
}
|
|
|
|
if (!lit_char_is_decimal_digit (*str_p))
|
|
{
|
|
break;
|
|
}
|
|
|
|
int32_t digit_value = (*str_p++ - LIT_CHAR_0);
|
|
exponent = exponent * 10 + digit_value;
|
|
|
|
if (exponent_sign * exponent > NUMBER_MAX_DECIMAL_EXPONENT)
|
|
{
|
|
return ecma_number_make_infinity (sign);
|
|
}
|
|
|
|
if (exponent_sign * exponent < NUMBER_MIN_DECIMAL_EXPONENT)
|
|
{
|
|
return sign ? -ECMA_NUMBER_ZERO : ECMA_NUMBER_ZERO;
|
|
}
|
|
}
|
|
|
|
decimal_exponent += exponent_sign * exponent;
|
|
}
|
|
|
|
if (!has_significand || str_p < end_p)
|
|
{
|
|
return ecma_number_make_nan ();
|
|
}
|
|
|
|
if (significand == 0)
|
|
{
|
|
return sign ? -ECMA_NUMBER_ZERO : ECMA_NUMBER_ZERO;
|
|
}
|
|
|
|
#if JERRY_NUMBER_TYPE_FLOAT64
|
|
/*
|
|
* 128-bit mantissa storage
|
|
*
|
|
* Normalized: |4 bits zero|124-bit mantissa with highest bit set to 1|
|
|
*/
|
|
ecma_uint128_t significand_uint128 = { .hi = significand, .lo = 0 };
|
|
|
|
/* Normalizing mantissa */
|
|
int shift = 4 - ecma_uint64_clz (significand_uint128.hi);
|
|
|
|
if (shift < 0)
|
|
{
|
|
ecma_uint128_shift_left (&significand_uint128, -shift);
|
|
}
|
|
else
|
|
{
|
|
ecma_uint128_shift_right (&significand_uint128, shift);
|
|
}
|
|
|
|
int32_t binary_exponent = ECMA_NUMBER_FRACTION_WIDTH + shift;
|
|
|
|
while (decimal_exponent > 0)
|
|
{
|
|
JERRY_ASSERT (ecma_uint64_clz (significand_uint128.hi) == 4);
|
|
|
|
ecma_uint128_mul10 (&significand_uint128);
|
|
decimal_exponent--;
|
|
|
|
/* Re-normalizing mantissa */
|
|
shift = ecma_uint64_normalize_shift (significand_uint128.hi);
|
|
JERRY_ASSERT (shift >= 0 && shift <= 4);
|
|
|
|
ecma_uint128_shift_right (&significand_uint128, shift);
|
|
binary_exponent += shift;
|
|
}
|
|
|
|
while (decimal_exponent < 0)
|
|
{
|
|
/* Denormalizing mantissa, moving highest 1 to bit 127 */
|
|
JERRY_ASSERT (ecma_uint64_clz (significand_uint128.hi) <= 4);
|
|
shift = ecma_uint64_clz_top4 (significand_uint128.hi);
|
|
JERRY_ASSERT (shift >= 0 && shift <= 4);
|
|
|
|
ecma_uint128_shift_left (&significand_uint128, shift);
|
|
binary_exponent -= shift;
|
|
|
|
ecma_uint128_div10 (&significand_uint128);
|
|
decimal_exponent++;
|
|
}
|
|
|
|
/*
|
|
* Preparing mantissa for conversion to 52-bit representation, converting it to:
|
|
*
|
|
* |11 zero bits|1|116 mantissa bits|
|
|
*/
|
|
JERRY_ASSERT (ecma_uint64_clz (significand_uint128.hi) <= 4);
|
|
shift = 11 - ecma_uint64_clz_top4 (significand_uint128.hi);
|
|
ecma_uint128_shift_right (&significand_uint128, shift);
|
|
binary_exponent += shift;
|
|
|
|
JERRY_ASSERT (ecma_uint64_clz (significand_uint128.hi) == 11);
|
|
|
|
binary_exponent += ECMA_NUMBER_EXPONENT_BIAS;
|
|
|
|
/* Handle denormal numbers */
|
|
if (binary_exponent < 1)
|
|
{
|
|
ecma_uint128_shift_right (&significand_uint128, -binary_exponent + 1);
|
|
binary_exponent = 0;
|
|
}
|
|
|
|
significand = ecma_round_high_to_uint64 (&significand_uint128);
|
|
|
|
if (significand >= 1ull << (ECMA_NUMBER_FRACTION_WIDTH + 1))
|
|
{
|
|
/* Rounding carried over to the most significant bit, re-normalize.
|
|
* No need to shift mantissa right, as the low 52 bits will be 0 regardless. */
|
|
binary_exponent++;
|
|
}
|
|
|
|
if (binary_exponent >= ((1 << ECMA_NUMBER_BIASED_EXP_WIDTH) - 1))
|
|
{
|
|
return ecma_number_make_infinity (sign);
|
|
}
|
|
|
|
/* Mask low 52 bits. */
|
|
significand &= ((1ull << ECMA_NUMBER_FRACTION_WIDTH) - 1);
|
|
|
|
JERRY_ASSERT (binary_exponent < (1 << ECMA_NUMBER_BIASED_EXP_WIDTH) - 1);
|
|
JERRY_ASSERT (significand < (1ull << ECMA_NUMBER_FRACTION_WIDTH));
|
|
|
|
return ecma_number_create (sign, (uint32_t) binary_exponent, significand);
|
|
#elif !JERRY_NUMBER_TYPE_FLOAT64
|
|
/* Less precise conversion */
|
|
ecma_number_t num = (ecma_number_t) (uint32_t) significand;
|
|
|
|
bool e_sign = decimal_exponent < 0;
|
|
int32_t e = e_sign ? -decimal_exponent : decimal_exponent;
|
|
|
|
ecma_number_t m = e_sign ? (ecma_number_t) 0.1 : (ecma_number_t) 10.0;
|
|
|
|
while (e)
|
|
{
|
|
if (e % 2)
|
|
{
|
|
num *= m;
|
|
}
|
|
|
|
m *= m;
|
|
e /= 2;
|
|
}
|
|
|
|
return sign ? -num : num;
|
|
#endif /* JERRY_NUMBER_TYPE_FLOAT64 */
|
|
} /* ecma_utf8_string_to_number */
|
|
|
|
/**
|
|
* ECMA-defined conversion of UInt32 to String (zero-terminated).
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 9.8.1
|
|
*
|
|
* @return number of bytes copied to buffer
|
|
*/
|
|
lit_utf8_size_t
|
|
ecma_uint32_to_utf8_string (uint32_t value, /**< value to convert */
|
|
lit_utf8_byte_t *out_buffer_p, /**< buffer for string */
|
|
lit_utf8_size_t buffer_size) /**< size of buffer */
|
|
{
|
|
lit_utf8_byte_t *buf_p = out_buffer_p + buffer_size;
|
|
|
|
do
|
|
{
|
|
JERRY_ASSERT (buf_p >= out_buffer_p);
|
|
|
|
buf_p--;
|
|
*buf_p = (lit_utf8_byte_t) ((value % 10) + LIT_CHAR_0);
|
|
value /= 10;
|
|
} while (value != 0);
|
|
|
|
JERRY_ASSERT (buf_p >= out_buffer_p);
|
|
|
|
lit_utf8_size_t bytes_copied = (lit_utf8_size_t) (out_buffer_p + buffer_size - buf_p);
|
|
|
|
if (JERRY_LIKELY (buf_p != out_buffer_p))
|
|
{
|
|
memmove (out_buffer_p, buf_p, bytes_copied);
|
|
}
|
|
|
|
return bytes_copied;
|
|
} /* ecma_uint32_to_utf8_string */
|
|
|
|
/**
|
|
* ECMA-defined conversion of Number value to UInt32 value
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 9.6
|
|
*
|
|
* @return 32-bit unsigned integer - result of conversion.
|
|
*/
|
|
uint32_t
|
|
ecma_number_to_uint32 (ecma_number_t num) /**< ecma-number */
|
|
{
|
|
if (JERRY_UNLIKELY (ecma_number_is_zero (num) || !ecma_number_is_finite (num)))
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
const bool sign = ecma_number_is_negative (num);
|
|
const ecma_number_t abs_num = sign ? -num : num;
|
|
|
|
/* 2 ^ 32 */
|
|
const uint64_t uint64_2_pow_32 = (1ull << 32);
|
|
|
|
const ecma_number_t num_2_pow_32 = (ecma_number_t) uint64_2_pow_32;
|
|
|
|
ecma_number_t num_in_uint32_range;
|
|
|
|
if (abs_num >= num_2_pow_32)
|
|
{
|
|
num_in_uint32_range = ecma_number_remainder (abs_num, num_2_pow_32);
|
|
}
|
|
else
|
|
{
|
|
num_in_uint32_range = abs_num;
|
|
}
|
|
|
|
/* Check that the floating point value can be represented with uint32_t. */
|
|
JERRY_ASSERT (num_in_uint32_range < uint64_2_pow_32);
|
|
uint32_t uint32_num = (uint32_t) num_in_uint32_range;
|
|
|
|
const uint32_t ret = sign ? -uint32_num : uint32_num;
|
|
|
|
#ifndef JERRY_NDEBUG
|
|
if (sign && uint32_num != 0)
|
|
{
|
|
JERRY_ASSERT (ret == uint64_2_pow_32 - uint32_num);
|
|
}
|
|
else
|
|
{
|
|
JERRY_ASSERT (ret == uint32_num);
|
|
}
|
|
#endif /* !JERRY_NDEBUG */
|
|
|
|
return ret;
|
|
} /* ecma_number_to_uint32 */
|
|
|
|
/**
|
|
* ECMA-defined conversion of Number value to Int32 value
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 9.5
|
|
*
|
|
* @return 32-bit signed integer - result of conversion.
|
|
*/
|
|
int32_t
|
|
ecma_number_to_int32 (ecma_number_t num) /**< ecma-number */
|
|
{
|
|
uint32_t uint32_num = ecma_number_to_uint32 (num);
|
|
|
|
/* 2 ^ 32 */
|
|
const int64_t int64_2_pow_32 = (1ll << 32);
|
|
|
|
/* 2 ^ 31 */
|
|
const uint32_t uint32_2_pow_31 = (1ull << 31);
|
|
|
|
int32_t ret;
|
|
|
|
if (uint32_num >= uint32_2_pow_31)
|
|
{
|
|
ret = (int32_t) (uint32_num - int64_2_pow_32);
|
|
}
|
|
else
|
|
{
|
|
ret = (int32_t) uint32_num;
|
|
}
|
|
|
|
#ifndef JERRY_NDEBUG
|
|
int64_t int64_num = uint32_num;
|
|
|
|
JERRY_ASSERT (int64_num >= 0);
|
|
|
|
if (int64_num >= uint32_2_pow_31)
|
|
{
|
|
JERRY_ASSERT (ret == int64_num - int64_2_pow_32);
|
|
}
|
|
else
|
|
{
|
|
JERRY_ASSERT (ret == int64_num);
|
|
}
|
|
#endif /* !JERRY_NDEBUG */
|
|
|
|
return ret;
|
|
} /* ecma_number_to_int32 */
|
|
|
|
/**
|
|
* Perform conversion of ecma-number to decimal representation with decimal exponent.
|
|
*
|
|
* Note:
|
|
* The calculated values correspond to s, n, k parameters in ECMA-262 v5, 9.8.1, item 5:
|
|
* - parameter out_digits_p corresponds to s, the digits of the number;
|
|
* - parameter out_decimal_exp_p corresponds to n, the decimal exponent;
|
|
* - return value corresponds to k, the number of digits.
|
|
*
|
|
* @return the number of digits
|
|
*/
|
|
lit_utf8_size_t
|
|
ecma_number_to_decimal (ecma_number_t num, /**< ecma-number */
|
|
lit_utf8_byte_t *out_digits_p, /**< [out] buffer to fill with digits */
|
|
int32_t *out_decimal_exp_p) /**< [out] decimal exponent */
|
|
{
|
|
JERRY_ASSERT (!ecma_number_is_nan (num));
|
|
JERRY_ASSERT (!ecma_number_is_zero (num));
|
|
JERRY_ASSERT (!ecma_number_is_infinity (num));
|
|
JERRY_ASSERT (!ecma_number_is_negative (num));
|
|
|
|
#if JERRY_NUMBER_TYPE_FLOAT64
|
|
return ecma_errol0_dtoa ((double) num, out_digits_p, out_decimal_exp_p);
|
|
#else /* !JERRY_NUMBER_TYPE_FLOAT64 */
|
|
/* Float32: snprintf with ECMA_NUMBER_MAX_DIGITS significant digits guarantees round-trip.
|
|
* Using %e gives "D.DDDDDDDDe±EEE" which we parse into digits + decimal exponent. */
|
|
char buf[ECMA_NUMBER_MAX_DIGITS + 16];
|
|
snprintf (buf, sizeof (buf), "%.*e", ECMA_NUMBER_MAX_DIGITS - 1, (double) num);
|
|
|
|
char *p = buf;
|
|
lit_utf8_byte_t *out_p = out_digits_p;
|
|
int32_t k = 0;
|
|
|
|
*out_p++ = (lit_utf8_byte_t) *p++; /* first significant digit */
|
|
k++;
|
|
p++; /* skip '.' */
|
|
|
|
while (*p != 'e' && *p != 'E')
|
|
{
|
|
*out_p++ = (lit_utf8_byte_t) *p++;
|
|
k++;
|
|
}
|
|
|
|
/* strip trailing zeros */
|
|
while (k > 1 && out_digits_p[k - 1] == '0')
|
|
{
|
|
k--;
|
|
}
|
|
|
|
p++; /* skip 'e' */
|
|
int32_t exp_sign = (*p == '-') ? -1 : 1;
|
|
p++; /* skip sign */
|
|
int32_t exponent = 0;
|
|
while (*p)
|
|
{
|
|
exponent = exponent * 10 + (*p++ - '0');
|
|
}
|
|
|
|
/* decimal exponent convention: n=1 means first digit is units, n=0 means "0.d..." */
|
|
*out_decimal_exp_p = exp_sign * exponent + 1;
|
|
|
|
return (lit_utf8_size_t) k;
|
|
#endif /* JERRY_NUMBER_TYPE_FLOAT64 */
|
|
} /* ecma_number_to_decimal */
|
|
|
|
/**
|
|
* Convert ecma-number to zero-terminated string
|
|
*
|
|
* See also:
|
|
* ECMA-262 v5, 9.8.1
|
|
*
|
|
*
|
|
* @return size of utf-8 string
|
|
*/
|
|
lit_utf8_size_t
|
|
ecma_number_to_utf8_string (ecma_number_t num, /**< ecma-number */
|
|
lit_utf8_byte_t *buffer_p, /**< buffer for utf-8 string */
|
|
lit_utf8_size_t buffer_size) /**< size of buffer */
|
|
{
|
|
lit_utf8_byte_t *dst_p;
|
|
|
|
if (ecma_number_is_nan (num))
|
|
{
|
|
/* 1. */
|
|
dst_p = lit_copy_magic_string_to_buffer (LIT_MAGIC_STRING_NAN, buffer_p, buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
if (ecma_number_is_zero (num))
|
|
{
|
|
/* 2. */
|
|
*buffer_p = LIT_CHAR_0;
|
|
JERRY_ASSERT (1 <= buffer_size);
|
|
return 1;
|
|
}
|
|
|
|
dst_p = buffer_p;
|
|
|
|
if (ecma_number_is_negative (num))
|
|
{
|
|
/* 3. */
|
|
*dst_p++ = LIT_CHAR_MINUS;
|
|
num = -num;
|
|
}
|
|
|
|
if (ecma_number_is_infinity (num))
|
|
{
|
|
/* 4. */
|
|
dst_p = lit_copy_magic_string_to_buffer (LIT_MAGIC_STRING_INFINITY_UL,
|
|
dst_p,
|
|
(lit_utf8_size_t) (buffer_p + buffer_size - dst_p));
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
/* 5. */
|
|
uint32_t num_uint32 = ecma_number_to_uint32 (num);
|
|
|
|
if (((ecma_number_t) num_uint32) == num)
|
|
{
|
|
dst_p += ecma_uint32_to_utf8_string (num_uint32, dst_p, (lit_utf8_size_t) (buffer_p + buffer_size - dst_p));
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
/* decimal exponent */
|
|
int32_t n;
|
|
/* number of digits in mantissa */
|
|
int32_t k;
|
|
|
|
k = (int32_t) ecma_number_to_decimal (num, dst_p, &n);
|
|
|
|
if (k <= n && n <= 21)
|
|
{
|
|
/* 6. */
|
|
dst_p += k;
|
|
|
|
memset (dst_p, LIT_CHAR_0, (size_t) (n - k));
|
|
dst_p += n - k;
|
|
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
if (0 < n && n <= 21)
|
|
{
|
|
/* 7. */
|
|
memmove (dst_p + n + 1, dst_p + n, (size_t) (k - n));
|
|
*(dst_p + n) = LIT_CHAR_DOT;
|
|
dst_p += k + 1;
|
|
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
if (-6 < n && n <= 0)
|
|
{
|
|
/* 8. */
|
|
memmove (dst_p + 2 - n, dst_p, (size_t) k);
|
|
memset (dst_p + 2, LIT_CHAR_0, (size_t) -n);
|
|
*dst_p = LIT_CHAR_0;
|
|
*(dst_p + 1) = LIT_CHAR_DOT;
|
|
dst_p += k - n + 2;
|
|
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
}
|
|
|
|
if (k == 1)
|
|
{
|
|
/* 9. */
|
|
dst_p++;
|
|
}
|
|
else
|
|
{
|
|
/* 10. */
|
|
memmove (dst_p + 2, dst_p + 1, (size_t) (k - 1));
|
|
*(dst_p + 1) = LIT_CHAR_DOT;
|
|
dst_p += k + 1;
|
|
}
|
|
|
|
/* 9., 10. */
|
|
*dst_p++ = LIT_CHAR_LOWERCASE_E;
|
|
*dst_p++ = (n >= 1) ? LIT_CHAR_PLUS : LIT_CHAR_MINUS;
|
|
uint32_t t = (uint32_t) (n >= 1 ? (n - 1) : -(n - 1));
|
|
|
|
dst_p += ecma_uint32_to_utf8_string (t, dst_p, (lit_utf8_size_t) (buffer_p + buffer_size - dst_p));
|
|
|
|
JERRY_ASSERT (dst_p <= buffer_p + buffer_size);
|
|
|
|
return (lit_utf8_size_t) (dst_p - buffer_p);
|
|
} /* ecma_number_to_utf8_string */
|
|
|
|
/**
|
|
* @}
|
|
* @}
|
|
*/
|