| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2021-06-19 01:34:17 +01:00
										 |  |  |  * Copyright (c) 2020-2021, Linus Groh <linusg@serenityos.org> | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2021-04-22 01:24:48 -07:00
										 |  |  |  * SPDX-License-Identifier: BSD-2-Clause | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-17 18:29:28 +02:00
										 |  |  | #include <AK/Math.h>
 | 
					
						
							| 
									
										
										
										
											2021-06-20 01:09:39 +01:00
										 |  |  | #include <LibJS/Runtime/AbstractOperations.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | #include <LibJS/Runtime/Error.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-18 13:18:06 +02:00
										 |  |  | #include <LibJS/Runtime/GlobalObject.h>
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | #include <LibJS/Runtime/NumberConstructor.h>
 | 
					
						
							|  |  |  | #include <LibJS/Runtime/NumberObject.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:20:55 +00:00
										 |  |  | #ifdef __clang__
 | 
					
						
							| 
									
										
										
										
											2021-07-17 18:29:28 +02:00
										 |  |  | #    define EPSILON_VALUE AK::exp2(-52.)
 | 
					
						
							|  |  |  | #    define MAX_SAFE_INTEGER_VALUE AK::exp2(53.) - 1
 | 
					
						
							|  |  |  | #    define MIN_SAFE_INTEGER_VALUE -(AK::exp2(53.) - 1)
 | 
					
						
							| 
									
										
										
										
											2020-11-07 00:20:55 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2021-07-17 18:29:28 +02:00
										 |  |  | constexpr const double EPSILON_VALUE { __builtin_exp2(-52) }; | 
					
						
							|  |  |  | constexpr const double MAX_SAFE_INTEGER_VALUE { __builtin_exp2(53) - 1 }; | 
					
						
							|  |  |  | constexpr const double MIN_SAFE_INTEGER_VALUE { -(__builtin_exp2(53) - 1) }; | 
					
						
							| 
									
										
										
										
											2020-11-07 00:20:55 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-04-07 18:06:44 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | namespace JS { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-20 15:40:48 +02:00
										 |  |  | NumberConstructor::NumberConstructor(GlobalObject& global_object) | 
					
						
							| 
									
										
										
										
											2021-06-28 03:45:49 +03:00
										 |  |  |     : NativeFunction(vm().names.Number.as_string(), *global_object.function_prototype()) | 
					
						
							| 
									
										
										
										
											2020-06-20 15:40:48 +02:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  | void NumberConstructor::initialize(GlobalObject& global_object) | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-10-13 23:49:19 +02:00
										 |  |  |     auto& vm = this->vm(); | 
					
						
							| 
									
										
										
										
											2020-07-22 17:50:18 +02:00
										 |  |  |     NativeFunction::initialize(global_object); | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // 21.1.2.15 Number.prototype, https://tc39.es/ecma262/#sec-number.prototype
 | 
					
						
							| 
									
										
										
										
											2021-07-06 02:15:08 +03:00
										 |  |  |     define_direct_property(vm.names.prototype, global_object.number_prototype(), 0); | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-27 23:05:02 -07:00
										 |  |  |     u8 attr = Attribute::Writable | Attribute::Configurable; | 
					
						
							| 
									
										
										
										
											2020-10-13 23:49:19 +02:00
										 |  |  |     define_native_function(vm.names.isFinite, is_finite, 1, attr); | 
					
						
							|  |  |  |     define_native_function(vm.names.isInteger, is_integer, 1, attr); | 
					
						
							|  |  |  |     define_native_function(vm.names.isNaN, is_nan, 1, attr); | 
					
						
							|  |  |  |     define_native_function(vm.names.isSafeInteger, is_safe_integer, 1, attr); | 
					
						
							| 
									
										
										
										
											2021-10-02 23:52:27 +01:00
										 |  |  |     define_direct_property(vm.names.parseInt, global_object.get_without_side_effects(vm.names.parseInt), attr); | 
					
						
							|  |  |  |     define_direct_property(vm.names.parseFloat, global_object.get_without_side_effects(vm.names.parseFloat), attr); | 
					
						
							| 
									
										
										
										
											2021-07-06 02:15:08 +03:00
										 |  |  |     define_direct_property(vm.names.EPSILON, Value(EPSILON_VALUE), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.MAX_VALUE, Value(NumericLimits<double>::max()), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.MIN_VALUE, Value(NumericLimits<double>::min()), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.MAX_SAFE_INTEGER, Value(MAX_SAFE_INTEGER_VALUE), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.MIN_SAFE_INTEGER, Value(MIN_SAFE_INTEGER_VALUE), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.NEGATIVE_INFINITY, js_negative_infinity(), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.POSITIVE_INFINITY, js_infinity(), 0); | 
					
						
							|  |  |  |     define_direct_property(vm.names.NaN, js_nan(), 0); | 
					
						
							| 
									
										
										
										
											2021-07-08 02:49:53 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |     define_direct_property(vm.names.length, Value(1), Attribute::Configurable); | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NumberConstructor::~NumberConstructor() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-19 01:34:17 +01:00
										 |  |  | // Most of 21.1.1.1 Number ( value ) factored into a separate function for sharing between call() and construct().
 | 
					
						
							|  |  |  | static Value get_value_from_constructor_argument(GlobalObject& global_object) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     auto& vm = global_object.vm(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Value number; | 
					
						
							|  |  |  |     if (vm.argument_count() > 0) { | 
					
						
							| 
									
										
										
										
											2021-10-16 21:42:32 +03:00
										 |  |  |         auto primitive = TRY_OR_DISCARD(vm.argument(0).to_numeric(global_object)); | 
					
						
							| 
									
										
										
										
											2021-06-19 01:34:17 +01:00
										 |  |  |         if (primitive.is_bigint()) { | 
					
						
							|  |  |  |             // FIXME: How should huge values be handled here?
 | 
					
						
							|  |  |  |             auto& big_integer = primitive.as_bigint().big_integer(); | 
					
						
							|  |  |  |             number = Value(static_cast<double>(big_integer.unsigned_value().to_u64()) * (big_integer.is_negative() ? -1.0 : 1.0)); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             number = primitive; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         number = Value(0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return number; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.1.1 Number ( value ), https://tc39.es/ecma262/#sec-number-constructor-number-value
 | 
					
						
							| 
									
										
										
										
											2020-09-27 17:24:14 +02:00
										 |  |  | Value NumberConstructor::call() | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-19 02:03:37 +01:00
										 |  |  |     auto& vm = this->vm(); | 
					
						
							|  |  |  |     auto& global_object = this->global_object(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto number = get_value_from_constructor_argument(global_object); | 
					
						
							|  |  |  |     if (vm.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							|  |  |  |     return number; | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.1.1 Number ( value ), https://tc39.es/ecma262/#sec-number-constructor-number-value
 | 
					
						
							| 
									
										
										
										
											2021-06-27 21:48:34 +02:00
										 |  |  | Value NumberConstructor::construct(FunctionObject& new_target) | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-19 02:03:37 +01:00
										 |  |  |     auto& vm = this->vm(); | 
					
						
							|  |  |  |     auto& global_object = this->global_object(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto number = get_value_from_constructor_argument(global_object); | 
					
						
							|  |  |  |     if (vm.exception()) | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2021-09-16 00:57:38 +03:00
										 |  |  |     return TRY_OR_DISCARD(ordinary_create_from_constructor<NumberObject>(global_object, new_target, &GlobalObject::number_prototype, number.as_double())); | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.2.2 Number.isFinite ( number ), https://tc39.es/ecma262/#sec-number.isfinite
 | 
					
						
							| 
									
										
										
										
											2020-06-20 13:55:34 +02:00
										 |  |  | JS_DEFINE_NATIVE_FUNCTION(NumberConstructor::is_finite) | 
					
						
							| 
									
										
										
										
											2020-04-26 18:24:58 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     return Value(vm.argument(0).is_finite_number()); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:24:58 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.2.3 Number.isInteger ( number ), https://tc39.es/ecma262/#sec-number.isinteger
 | 
					
						
							| 
									
										
										
										
											2020-06-20 13:55:34 +02:00
										 |  |  | JS_DEFINE_NATIVE_FUNCTION(NumberConstructor::is_integer) | 
					
						
							| 
									
										
										
										
											2020-04-26 18:49:23 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-16 13:17:06 +03:00
										 |  |  |     return Value(vm.argument(0).is_integral_number()); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:49:23 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.2.4 Number.isNaN ( number ), https://tc39.es/ecma262/#sec-number.isnan
 | 
					
						
							| 
									
										
										
										
											2020-06-20 13:55:34 +02:00
										 |  |  | JS_DEFINE_NATIVE_FUNCTION(NumberConstructor::is_nan) | 
					
						
							| 
									
										
										
										
											2020-04-26 18:31:56 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     return Value(vm.argument(0).is_nan()); | 
					
						
							| 
									
										
										
										
											2020-04-26 18:31:56 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 00:22:35 +01:00
										 |  |  | // 21.1.2.5 Number.isSafeInteger ( number ), https://tc39.es/ecma262/#sec-number.issafeinteger
 | 
					
						
							| 
									
										
										
										
											2020-06-20 13:55:34 +02:00
										 |  |  | JS_DEFINE_NATIVE_FUNCTION(NumberConstructor::is_safe_integer) | 
					
						
							| 
									
										
										
										
											2020-04-07 18:06:44 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     if (!vm.argument(0).is_number()) | 
					
						
							| 
									
										
										
										
											2020-04-07 18:06:44 +01:00
										 |  |  |         return Value(false); | 
					
						
							| 
									
										
										
										
											2021-08-06 18:35:33 +02:00
										 |  |  |     if (!vm.argument(0).is_integral_number()) | 
					
						
							|  |  |  |         return Value(false); | 
					
						
							| 
									
										
										
										
											2020-09-27 18:36:49 +02:00
										 |  |  |     auto value = vm.argument(0).as_double(); | 
					
						
							| 
									
										
										
										
											2021-08-06 18:35:33 +02:00
										 |  |  |     return Value(value >= MIN_SAFE_INTEGER_VALUE && value <= MAX_SAFE_INTEGER_VALUE); | 
					
						
							| 
									
										
										
										
											2020-04-07 18:06:44 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-07 16:17:23 +01:00
										 |  |  | } |