5 #include "src/builtins/builtins-utils-inl.h" 6 #include "src/builtins/builtins.h" 7 #include "src/code-factory.h" 8 #include "src/conversions.h" 9 #include "src/counters.h" 10 #include "src/objects-inl.h" 11 #ifdef V8_INTL_SUPPORT 12 #include "src/objects/intl-objects.h" 22 BUILTIN(NumberPrototypeToExponential) {
23 HandleScope scope(isolate);
24 Handle<Object> value = args.at(0);
25 Handle<Object> fraction_digits = args.atOrUndefined(isolate, 1);
28 if (value->IsJSValue()) {
29 value = handle(Handle<JSValue>::cast(value)->value(), isolate);
31 if (!value->IsNumber()) {
32 THROW_NEW_ERROR_RETURN_FAILURE(
33 isolate, NewTypeError(MessageTemplate::kNotGeneric,
34 isolate->factory()->NewStringFromAsciiChecked(
35 "Number.prototype.toExponential"),
36 isolate->factory()->Number_string()));
38 double const value_number = value->Number();
41 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
42 isolate, fraction_digits, Object::ToInteger(isolate, fraction_digits));
43 double const fraction_digits_number = fraction_digits->Number();
45 if (std::isnan(value_number))
return ReadOnlyRoots(isolate).NaN_string();
46 if (std::isinf(value_number)) {
47 return (value_number < 0.0) ? ReadOnlyRoots(isolate).minus_Infinity_string()
48 : ReadOnlyRoots(isolate).Infinity_string();
50 if (fraction_digits_number < 0.0 ||
51 fraction_digits_number > kMaxFractionDigits) {
52 THROW_NEW_ERROR_RETURN_FAILURE(
53 isolate, NewRangeError(MessageTemplate::kNumberFormatRange,
54 isolate->factory()->NewStringFromAsciiChecked(
57 int const f = args.atOrUndefined(isolate, 1)->IsUndefined(isolate)
59 :
static_cast<int>(fraction_digits_number);
60 char*
const str = DoubleToExponentialCString(value_number, f);
61 Handle<String> result = isolate->factory()->NewStringFromAsciiChecked(str);
67 BUILTIN(NumberPrototypeToFixed) {
68 HandleScope scope(isolate);
69 Handle<Object> value = args.at(0);
70 Handle<Object> fraction_digits = args.atOrUndefined(isolate, 1);
73 if (value->IsJSValue()) {
74 value = handle(Handle<JSValue>::cast(value)->value(), isolate);
76 if (!value->IsNumber()) {
77 THROW_NEW_ERROR_RETURN_FAILURE(
78 isolate, NewTypeError(MessageTemplate::kNotGeneric,
79 isolate->factory()->NewStringFromAsciiChecked(
80 "Number.prototype.toFixed"),
81 isolate->factory()->Number_string()));
83 double const value_number = value->Number();
86 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(
87 isolate, fraction_digits, Object::ToInteger(isolate, fraction_digits));
88 double const fraction_digits_number = fraction_digits->Number();
91 if (fraction_digits_number < 0.0 ||
92 fraction_digits_number > kMaxFractionDigits) {
93 THROW_NEW_ERROR_RETURN_FAILURE(
94 isolate, NewRangeError(MessageTemplate::kNumberFormatRange,
95 isolate->factory()->NewStringFromAsciiChecked(
96 "toFixed() digits")));
99 if (std::isnan(value_number))
return ReadOnlyRoots(isolate).NaN_string();
100 if (std::isinf(value_number)) {
101 return (value_number < 0.0) ? ReadOnlyRoots(isolate).minus_Infinity_string()
102 : ReadOnlyRoots(isolate).Infinity_string();
104 char*
const str = DoubleToFixedCString(
105 value_number, static_cast<int>(fraction_digits_number));
106 Handle<String> result = isolate->factory()->NewStringFromAsciiChecked(str);
112 BUILTIN(NumberPrototypeToLocaleString) {
113 HandleScope scope(isolate);
115 isolate->CountUsage(v8::Isolate::UseCounterFeature::kNumberToLocaleString);
117 Handle<Object> value = args.at(0);
120 if (value->IsJSValue()) {
121 value = handle(Handle<JSValue>::cast(value)->value(), isolate);
124 if (!value->IsNumber()) {
125 THROW_NEW_ERROR_RETURN_FAILURE(
126 isolate, NewTypeError(MessageTemplate::kNotGeneric,
127 isolate->factory()->NewStringFromAsciiChecked(
128 "Number.prototype.toLocaleString"),
129 isolate->factory()->Number_string()));
132 #ifdef V8_INTL_SUPPORT 133 RETURN_RESULT_OR_FAILURE(
135 Intl::NumberToLocaleString(isolate, value, args.atOrUndefined(isolate, 1),
136 args.atOrUndefined(isolate, 2)));
139 return *isolate->factory()->NumberToString(value);
140 #endif // V8_INTL_SUPPORT 144 BUILTIN(NumberPrototypeToPrecision) {
145 HandleScope scope(isolate);
146 Handle<Object> value = args.at(0);
147 Handle<Object> precision = args.atOrUndefined(isolate, 1);
150 if (value->IsJSValue()) {
151 value = handle(Handle<JSValue>::cast(value)->value(), isolate);
153 if (!value->IsNumber()) {
154 THROW_NEW_ERROR_RETURN_FAILURE(
155 isolate, NewTypeError(MessageTemplate::kNotGeneric,
156 isolate->factory()->NewStringFromAsciiChecked(
157 "Number.prototype.toPrecision"),
158 isolate->factory()->Number_string()));
160 double const value_number = value->Number();
163 if (precision->IsUndefined(isolate)) {
164 return *isolate->factory()->NumberToString(value);
168 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, precision,
169 Object::ToInteger(isolate, precision));
170 double const precision_number = precision->Number();
172 if (std::isnan(value_number))
return ReadOnlyRoots(isolate).NaN_string();
173 if (std::isinf(value_number)) {
174 return (value_number < 0.0) ? ReadOnlyRoots(isolate).minus_Infinity_string()
175 : ReadOnlyRoots(isolate).Infinity_string();
177 if (precision_number < 1.0 || precision_number > kMaxFractionDigits) {
178 THROW_NEW_ERROR_RETURN_FAILURE(
179 isolate, NewRangeError(MessageTemplate::kToPrecisionFormatRange));
181 char*
const str = DoubleToPrecisionCString(
182 value_number, static_cast<int>(precision_number));
183 Handle<String> result = isolate->factory()->NewStringFromAsciiChecked(str);
189 BUILTIN(NumberPrototypeToString) {
190 HandleScope scope(isolate);
191 Handle<Object> value = args.at(0);
192 Handle<Object> radix = args.atOrUndefined(isolate, 1);
195 if (value->IsJSValue()) {
196 value = handle(Handle<JSValue>::cast(value)->value(), isolate);
198 if (!value->IsNumber()) {
199 THROW_NEW_ERROR_RETURN_FAILURE(
200 isolate, NewTypeError(MessageTemplate::kNotGeneric,
201 isolate->factory()->NewStringFromAsciiChecked(
202 "Number.prototype.toString"),
203 isolate->factory()->Number_string()));
205 double const value_number = value->Number();
208 if (radix->IsUndefined(isolate)) {
209 return *isolate->factory()->NumberToString(value);
213 ASSIGN_RETURN_FAILURE_ON_EXCEPTION(isolate, radix,
214 Object::ToInteger(isolate, radix));
215 double const radix_number = radix->Number();
218 if (radix_number == 10.0)
return *isolate->factory()->NumberToString(value);
221 if (radix_number < 2.0 || radix_number > 36.0) {
222 THROW_NEW_ERROR_RETURN_FAILURE(
223 isolate, NewRangeError(MessageTemplate::kToRadixFormatRange));
227 if ((IsUint32Double(value_number) && value_number < radix_number) ||
228 value_number == -0.0) {
230 static const char kCharTable[] =
"0123456789abcdefghijklmnopqrstuvwxyz";
231 return *isolate->factory()->LookupSingleCharacterStringFromCode(
232 kCharTable[static_cast<uint32_t>(value_number)]);
236 if (std::isnan(value_number))
return ReadOnlyRoots(isolate).NaN_string();
237 if (std::isinf(value_number)) {
238 return (value_number < 0.0) ? ReadOnlyRoots(isolate).minus_Infinity_string()
239 : ReadOnlyRoots(isolate).Infinity_string();
242 DoubleToRadixCString(value_number, static_cast<int>(radix_number));
243 Handle<String> result = isolate->factory()->NewStringFromAsciiChecked(str);