Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qv4mathobject.cpp
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant
4
6#include "qv4symbol_p.h"
7
8#include <QtCore/qdatetime.h>
9#include <QtCore/qmath.h>
10#include <QtCore/qrandom.h>
11#include <QtCore/private/qnumeric_p.h>
12#include <QtCore/qthreadstorage.h>
13
14#include <cmath>
15
16using namespace QV4;
17
19
20void Heap::MathObject::init()
21{
22 Object::init();
23 Scope scope(internalClass->engine);
24 ScopedObject m(scope, this);
25
26 m->defineReadonlyProperty(QStringLiteral("E"), Value::fromDouble(M_E));
27 m->defineReadonlyProperty(QStringLiteral("LN2"), Value::fromDouble(M_LN2));
28 m->defineReadonlyProperty(QStringLiteral("LN10"), Value::fromDouble(M_LN10));
29 m->defineReadonlyProperty(QStringLiteral("LOG2E"), Value::fromDouble(M_LOG2E));
30 m->defineReadonlyProperty(QStringLiteral("LOG10E"), Value::fromDouble(M_LOG10E));
31 m->defineReadonlyProperty(QStringLiteral("PI"), Value::fromDouble(M_PI));
32 m->defineReadonlyProperty(QStringLiteral("SQRT1_2"), Value::fromDouble(M_SQRT1_2));
33 m->defineReadonlyProperty(QStringLiteral("SQRT2"), Value::fromDouble(M_SQRT2));
34
35 m->defineDefaultProperty(QStringLiteral("abs"), QV4::MathObject::method_abs, 1);
36 m->defineDefaultProperty(QStringLiteral("acos"), QV4::MathObject::method_acos, 1);
37 m->defineDefaultProperty(QStringLiteral("acosh"), QV4::MathObject::method_acosh, 1);
38 m->defineDefaultProperty(QStringLiteral("asin"), QV4::MathObject::method_asin, 1);
39 m->defineDefaultProperty(QStringLiteral("asinh"), QV4::MathObject::method_asinh, 1);
40 m->defineDefaultProperty(QStringLiteral("atan"), QV4::MathObject::method_atan, 1);
41 m->defineDefaultProperty(QStringLiteral("atanh"), QV4::MathObject::method_atanh, 1);
42 m->defineDefaultProperty(QStringLiteral("atan2"), QV4::MathObject::method_atan2, 2);
43 m->defineDefaultProperty(QStringLiteral("cbrt"), QV4::MathObject::method_cbrt, 1);
44 m->defineDefaultProperty(QStringLiteral("ceil"), QV4::MathObject::method_ceil, 1);
45 m->defineDefaultProperty(QStringLiteral("clz32"), QV4::MathObject::method_clz32, 1);
46 m->defineDefaultProperty(QStringLiteral("cos"), QV4::MathObject::method_cos, 1);
47 m->defineDefaultProperty(QStringLiteral("cosh"), QV4::MathObject::method_cosh, 1);
48 m->defineDefaultProperty(QStringLiteral("exp"), QV4::MathObject::method_exp, 1);
49 m->defineDefaultProperty(QStringLiteral("expm1"), QV4::MathObject::method_expm1, 1);
50 m->defineDefaultProperty(QStringLiteral("floor"), QV4::MathObject::method_floor, 1);
51 m->defineDefaultProperty(QStringLiteral("fround"), QV4::MathObject::method_fround, 1);
52 m->defineDefaultProperty(QStringLiteral("hypot"), QV4::MathObject::method_hypot, 2);
53 m->defineDefaultProperty(QStringLiteral("imul"), QV4::MathObject::method_imul, 2);
54 m->defineDefaultProperty(QStringLiteral("log"), QV4::MathObject::method_log, 1);
55 m->defineDefaultProperty(QStringLiteral("log10"), QV4::MathObject::method_log10, 1);
56 m->defineDefaultProperty(QStringLiteral("log1p"), QV4::MathObject::method_log1p, 1);
57 m->defineDefaultProperty(QStringLiteral("log2"), QV4::MathObject::method_log2, 1);
58 m->defineDefaultProperty(QStringLiteral("max"), QV4::MathObject::method_max, 2);
59 m->defineDefaultProperty(QStringLiteral("min"), QV4::MathObject::method_min, 2);
60 m->defineDefaultProperty(QStringLiteral("pow"), QV4::MathObject::method_pow, 2);
61 m->defineDefaultProperty(QStringLiteral("random"), QV4::MathObject::method_random, 0);
62 m->defineDefaultProperty(QStringLiteral("round"), QV4::MathObject::method_round, 1);
63 m->defineDefaultProperty(QStringLiteral("sign"), QV4::MathObject::method_sign, 1);
64 m->defineDefaultProperty(QStringLiteral("sin"), QV4::MathObject::method_sin, 1);
65 m->defineDefaultProperty(QStringLiteral("sinh"), QV4::MathObject::method_sinh, 1);
66 m->defineDefaultProperty(QStringLiteral("sqrt"), QV4::MathObject::method_sqrt, 1);
67 m->defineDefaultProperty(QStringLiteral("tan"), QV4::MathObject::method_tan, 1);
68 m->defineDefaultProperty(QStringLiteral("tanh"), QV4::MathObject::method_tanh, 1);
69 m->defineDefaultProperty(QStringLiteral("trunc"), QV4::MathObject::method_trunc, 1);
70
71 ScopedString name(scope, scope.engine->newString(QStringLiteral("Math")));
72 m->defineReadonlyConfigurableProperty(scope.engine->symbol_toStringTag(), name);
73}
74
75static Q_ALWAYS_INLINE double copySign(double x, double y)
76{
77 return ::copysign(x, y);
78}
79
80ReturnedValue MathObject::method_abs(const FunctionObject *, const Value *, const Value *argv, int argc)
81{
82 if (!argc)
83 RETURN_RESULT(Encode(qt_qnan()));
84
85 if (argv[0].isInteger()) {
86 int i = argv[0].integerValue();
87 RETURN_RESULT(Encode(i < 0 ? - i : i));
88 }
89
90 double v = argv[0].toNumber();
91 if (v == 0) // 0 | -0
92 RETURN_RESULT(Encode(0));
93
94 RETURN_RESULT(Encode(v < 0 ? -v : v));
95}
96
97ReturnedValue MathObject::method_acos(const FunctionObject *, const Value *, const Value *argv, int argc)
98{
99 double v = argc ? argv[0].toNumber() : 2;
100 if (v > 1)
101 RETURN_RESULT(Encode(qt_qnan()));
102
103 RETURN_RESULT(Encode(std::acos(v)));
104}
105
106ReturnedValue MathObject::method_acosh(const FunctionObject *, const Value *, const Value *argv, int argc)
107{
108 double v = argc ? argv[0].toNumber() : 2;
109 if (v < 1)
110 RETURN_RESULT(Encode(qt_qnan()));
111
112#ifdef Q_CC_MINGW
113 // Mingw has a broken std::acosh(). It returns NaN when passed Infinity.
114 if (std::isinf(v))
115 RETURN_RESULT(Encode(v));
116#endif
117 RETURN_RESULT(Encode(std::acosh(v)));
118}
119
120ReturnedValue MathObject::method_asin(const FunctionObject *, const Value *, const Value *argv, int argc)
121{
122 double v = argc ? argv[0].toNumber() : 2;
123 if (v > 1)
124 RETURN_RESULT(Encode(qt_qnan()));
125 else
126 RETURN_RESULT(Encode(std::asin(v)));
127}
128
129ReturnedValue MathObject::method_asinh(const FunctionObject *, const Value *, const Value *argv, int argc)
130{
131 double v = argc ? argv[0].toNumber() : 2;
132 if (v == 0.0)
133 RETURN_RESULT(Encode(v));
134 RETURN_RESULT(Encode(std::asinh(v)));
135}
136
137ReturnedValue MathObject::method_atan(const FunctionObject *, const Value *, const Value *argv, int argc)
138{
139 double v = argc ? argv[0].toNumber() : qt_qnan();
140 if (v == 0.0)
141 RETURN_RESULT(Encode(v));
142 else
143 RETURN_RESULT(Encode(std::atan(v)));
144}
145
146ReturnedValue MathObject::method_atanh(const FunctionObject *, const Value *, const Value *argv, int argc)
147{
148 double v = argc ? argv[0].toNumber() : qt_qnan();
149 if (v == 0.0)
150 RETURN_RESULT(Encode(v));
151
152 RETURN_RESULT(Encode(std::atanh(v)));
153}
154
155ReturnedValue MathObject::method_atan2(const FunctionObject *, const Value *, const Value *argv, int argc)
156{
157 double v1 = argc ? argv[0].toNumber() : qt_qnan();
158 double v2 = argc > 1 ? argv[1].toNumber() : qt_qnan();
159
160 if ((v1 < 0) && qt_is_finite(v1) && qt_is_inf(v2) && (copySign(1.0, v2) == 1.0))
161 RETURN_RESULT(Encode(copySign(0, -1.0)));
162
163 if ((v1 == 0.0) && (v2 == 0.0)) {
164 if ((copySign(1.0, v1) == 1.0) && (copySign(1.0, v2) == -1.0)) {
165 RETURN_RESULT(Encode(M_PI));
166 } else if ((copySign(1.0, v1) == -1.0) && (copySign(1.0, v2) == -1.0)) {
167 RETURN_RESULT(Encode(-M_PI));
168 }
169 }
170 RETURN_RESULT(Encode(std::atan2(v1, v2)));
171}
172
173ReturnedValue MathObject::method_cbrt(const FunctionObject *, const Value *, const Value *argv, int argc)
174{
175 double v = argc ? argv[0].toNumber() : qt_qnan();
176 RETURN_RESULT(Encode(std::cbrt(v))); // cube root
177}
178
179ReturnedValue MathObject::method_ceil(const FunctionObject *, const Value *, const Value *argv, int argc)
180{
181 double v = argc ? argv[0].toNumber() : qt_qnan();
182 if (v < 0.0 && v > -1.0)
183 RETURN_RESULT(Encode(copySign(0, -1.0)));
184 else
185 RETURN_RESULT(Encode(std::ceil(v)));
186}
187
188ReturnedValue MathObject::method_clz32(const FunctionObject *, const Value *, const Value *argv, int argc)
189{
190 quint32 v = argc ? argv[0].toUInt32() : 0;
191 RETURN_RESULT(Encode(qint32(qCountLeadingZeroBits(v))));
192}
193
194ReturnedValue MathObject::method_cos(const FunctionObject *, const Value *, const Value *argv, int argc)
195{
196 double v = argc ? argv[0].toNumber() : qt_qnan();
197 RETURN_RESULT(Encode(std::cos(v)));
198}
199
200ReturnedValue MathObject::method_cosh(const FunctionObject *, const Value *, const Value *argv, int argc)
201{
202 double v = argc ? argv[0].toNumber() : qt_qnan();
203 RETURN_RESULT(Encode(std::cosh(v)));
204}
205
206ReturnedValue MathObject::method_exp(const FunctionObject *, const Value *, const Value *argv, int argc)
207{
208 double v = argc ? argv[0].toNumber() : qt_qnan();
209 if (qt_is_inf(v)) {
210 if (copySign(1.0, v) == -1.0)
211 RETURN_RESULT(Encode(0));
212 else
213 RETURN_RESULT(Encode(qt_inf()));
214 } else {
215 RETURN_RESULT(Encode(std::exp(v)));
216 }
217}
218
219ReturnedValue MathObject::method_expm1(const FunctionObject *, const Value *, const Value *argv, int argc)
220{
221 double v = argc ? argv[0].toNumber() : qt_qnan();
222 if (std::isnan(v) || qIsNull(v)) {
223 RETURN_RESULT(Encode(v));
224 } else if (qt_is_inf(v)) {
225 if (copySign(1.0, v) == -1.0)
226 RETURN_RESULT(Encode(-1.0));
227 else
228 RETURN_RESULT(Encode(qt_inf()));
229 } else {
230 RETURN_RESULT(Encode(std::expm1(v)));
231 }
232}
233
234ReturnedValue MathObject::method_floor(const FunctionObject *, const Value *, const Value *argv, int argc)
235{
236 double v = argc ? argv[0].toNumber() : qt_qnan();
237 Value result = Value::fromDouble(std::floor(v));
238 result.isInt32();
239 RETURN_RESULT(result);
240}
241
242ReturnedValue MathObject::method_fround(const FunctionObject *, const Value *, const Value *argv, int argc)
243{
244 double v = argc ? argv[0].toNumber() : qt_qnan();
245 if (std::isnan(v) || qt_is_inf(v) || qIsNull(v))
246 RETURN_RESULT(Encode(v));
247 else // convert to 32-bit float using roundTiesToEven, then convert back to 64-bit double
248 RETURN_RESULT(Encode(double(float(v))));
249}
250
251ReturnedValue MathObject::method_hypot(const FunctionObject *, const Value *, const Value *argv, int argc)
252{
253 // ES6 Math.hypot(v1, ..., vn) -> sqrt(sum(vi**2)) but "should take care to
254 // avoid the loss of precision from overflows and underflows" (as std::hypot does).
255 double v = 0;
256 // Spec mandates +0 on no args; and says nothing about what to do if toNumber() signals ...
257 if (argc > 0) {
258 QtPrivate::QHypotHelper<double> h(argv[0].toNumber());
259 for (int i = 1; i < argc; i++)
260 h = h.add(argv[i].toNumber());
261 v = h.result();
262 }
263 RETURN_RESULT(Value::fromDouble(v));
264}
265
266ReturnedValue MathObject::method_imul(const FunctionObject *, const Value *, const Value *argv, int argc)
267{
268 quint32 a = argc ? argv[0].toUInt32() : 0;
269 quint32 b = argc > 0 ? argv[1].toUInt32() : 0;
270 qint32 product = a * b;
271 RETURN_RESULT(Encode(product));
272}
273
274ReturnedValue MathObject::method_log(const FunctionObject *, const Value *, const Value *argv, int argc)
275{
276 double v = argc ? argv[0].toNumber() : qt_qnan();
277 if (v < 0)
278 RETURN_RESULT(Encode(qt_qnan()));
279 else
280 RETURN_RESULT(Encode(std::log(v)));
281}
282
283ReturnedValue MathObject::method_log10(const FunctionObject *, const Value *, const Value *argv, int argc)
284{
285 double v = argc ? argv[0].toNumber() : qt_qnan();
286 if (v < 0)
287 RETURN_RESULT(Encode(qt_qnan()));
288 else
289 RETURN_RESULT(Encode(std::log10(v)));
290}
291
292ReturnedValue MathObject::method_log1p(const FunctionObject *, const Value *, const Value *argv, int argc)
293{
294#if !defined(__ANDROID__)
295 using std::log1p;
296#endif
297 double v = argc ? argv[0].toNumber() : qt_qnan();
298 if (v < -1)
299 RETURN_RESULT(Encode(qt_qnan()));
300 else
301 RETURN_RESULT(Encode(log1p(v)));
302}
303
304ReturnedValue MathObject::method_log2(const FunctionObject *, const Value *, const Value *argv, int argc)
305{
306 double v = argc ? argv[0].toNumber() : qt_qnan();
307 if (v < 0) {
308 RETURN_RESULT(Encode(qt_qnan()));
309 } else {
310 RETURN_RESULT(Encode(std::log2(v)));
311 }
312}
313
314ReturnedValue MathObject::method_max(const FunctionObject *, const Value *, const Value *argv, int argc)
315{
316 double mx = -qt_inf();
317 for (int i = 0, ei = argc; i < ei; ++i) {
318 double x = argv[i].toNumber();
319 if ((x == 0 && mx == x && copySign(1.0, x) == 1.0)
320 || (x > mx) || std::isnan(x)) {
321 mx = x;
322 }
323 }
324 RETURN_RESULT(Encode::smallestNumber(mx));
325}
326
327ReturnedValue MathObject::method_min(const FunctionObject *, const Value *, const Value *argv, int argc)
328{
329 double mx = qt_inf();
330 for (int i = 0, ei = argc; i < ei; ++i) {
331 double x = argv[i].toNumber();
332 if ((x == 0 && mx == x && copySign(1.0, x) == -1.0)
333 || (x < mx) || std::isnan(x)) {
334 mx = x;
335 }
336 }
337 RETURN_RESULT(Encode::smallestNumber(mx));
338}
339
340ReturnedValue MathObject::method_pow(const FunctionObject *, const Value *, const Value *argv, int argc)
341{
342 double x = argc > 0 ? argv[0].toNumber() : qt_qnan();
343 double y = argc > 1 ? argv[1].toNumber() : qt_qnan();
344
345 RETURN_RESULT(Encode(QQmlPrivate::jsExponentiate(x, y)));
346}
347
348ReturnedValue MathObject::method_random(const FunctionObject *, const Value *, const Value *, int)
349{
350 RETURN_RESULT(Encode(QRandomGenerator::global()->generateDouble()));
351}
352
353ReturnedValue MathObject::method_round(const FunctionObject *, const Value *, const Value *argv, int argc)
354{
355 double v = argc ? argv[0].toNumber() : qt_qnan();
356 if (!std::isfinite(v))
357 RETURN_RESULT(Encode(v));
358
359 if (v < 0.5 && v >= -0.5)
360 v = std::copysign(0.0, v);
361 else
362 v = std::floor(v + 0.5);
363 RETURN_RESULT(Encode(v));
364}
365
366ReturnedValue MathObject::method_sign(const FunctionObject *, const Value *, const Value *argv, int argc)
367{
368 double v = argc ? argv[0].toNumber() : qt_qnan();
369
370 if (std::isnan(v))
371 RETURN_RESULT(Encode(qt_qnan()));
372
373 if (qIsNull(v))
374 RETURN_RESULT(Encode(v));
375
376 RETURN_RESULT(Encode(std::signbit(v) ? -1 : 1));
377}
378
379ReturnedValue MathObject::method_sin(const FunctionObject *, const Value *, const Value *argv, int argc)
380{
381 double v = argc ? argv[0].toNumber() : qt_qnan();
382 if (v == 0.0)
383 RETURN_RESULT(Encode(v));
384 else
385 RETURN_RESULT(Encode(std::sin(v)));
386}
387
388ReturnedValue MathObject::method_sinh(const FunctionObject *, const Value *, const Value *argv, int argc)
389{
390 double v = argc ? argv[0].toNumber() : qt_qnan();
391 if (v == 0.0)
392 RETURN_RESULT(Encode(v));
393 else
394 RETURN_RESULT(Encode(std::sinh(v)));
395}
396
397ReturnedValue MathObject::method_sqrt(const FunctionObject *, const Value *, const Value *argv, int argc)
398{
399 double v = argc ? argv[0].toNumber() : qt_qnan();
400 RETURN_RESULT(Encode(std::sqrt(v)));
401}
402
403ReturnedValue MathObject::method_tan(const FunctionObject *, const Value *, const Value *argv, int argc)
404{
405 double v = argc ? argv[0].toNumber() : qt_qnan();
406 if (v == 0.0)
407 RETURN_RESULT(Encode(v));
408 else
409 RETURN_RESULT(Encode(std::tan(v)));
410}
411
412ReturnedValue MathObject::method_tanh(const FunctionObject *, const Value *, const Value *argv, int argc)
413{
414 double v = argc ? argv[0].toNumber() : qt_qnan();
415 if (v == 0.0)
416 RETURN_RESULT(Encode(v));
417 else
418 RETURN_RESULT(Encode(std::tanh(v)));
419}
420
421ReturnedValue MathObject::method_trunc(const FunctionObject *, const Value *, const Value *argv, int argc)
422{
423 double v = argc ? argv[0].toNumber() : qt_qnan();
424 RETURN_RESULT(Encode(std::trunc(v)));
425}
#define M_LN10
Definition qmath.h:196
#define M_SQRT2
Definition qmath.h:224
#define M_LN2
Definition qmath.h:192
#define M_E
Definition qmath.h:180
#define M_LOG10E
Definition qmath.h:188
#define M_LOG2E
Definition qmath.h:184
#define M_SQRT1_2
Definition qmath.h:228
#define M_PI
Definition qmath.h:200
DEFINE_OBJECT_VTABLE(MathObject)
static Q_ALWAYS_INLINE double copySign(double x, double y)
#define RETURN_RESULT(r)