Coverage Report

Created: 2026-05-30 06:28

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/moddable/xs/sources/xsMath.c
Line
Count
Source
1
/*
2
 * Copyright (c) 2016-2025  Moddable Tech, Inc.
3
 *
4
 *   This file is part of the Moddable SDK Runtime.
5
 * 
6
 *   The Moddable SDK Runtime is free software: you can redistribute it and/or modify
7
 *   it under the terms of the GNU Lesser General Public License as published by
8
 *   the Free Software Foundation, either version 3 of the License, or
9
 *   (at your option) any later version.
10
 * 
11
 *   The Moddable SDK Runtime is distributed in the hope that it will be useful,
12
 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
13
 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
 *   GNU Lesser General Public License for more details.
15
 * 
16
 *   You should have received a copy of the GNU Lesser General Public License
17
 *   along with the Moddable SDK Runtime.  If not, see <http://www.gnu.org/licenses/>.
18
 *
19
 * This file incorporates work covered by the following copyright and  
20
 * permission notice:  
21
 *
22
 *       Copyright (C) 2010-2016 Marvell International Ltd.
23
 *       Copyright (C) 2002-2010 Kinoma, Inc.
24
 *
25
 *       Licensed under the Apache License, Version 2.0 (the "License");
26
 *       you may not use this file except in compliance with the License.
27
 *       You may obtain a copy of the License at
28
 *
29
 *        http://www.apache.org/licenses/LICENSE-2.0
30
 *
31
 *       Unless required by applicable law or agreed to in writing, software
32
 *       distributed under the License is distributed on an "AS IS" BASIS,
33
 *       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
34
 *       See the License for the specific language governing permissions and
35
 *       limitations under the License.
36
 */
37
38
#include "xsAll.h"
39
40
#include "xsum.c"
41
42
void fxBuildMath(txMachine* the)
43
24.5k
{
44
24.5k
  txSlot* slot;
45
24.5k
  mxPush(mxObjectPrototype);
46
24.5k
  slot = fxLastProperty(the, fxNewObjectInstance(the));
47
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_abs), 1, mxID(_abs), XS_DONT_ENUM_FLAG);
48
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_acos), 1, mxID(_acos), XS_DONT_ENUM_FLAG);
49
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_acosh), 1, mxID(_acosh), XS_DONT_ENUM_FLAG);
50
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_asin), 1, mxID(_asin), XS_DONT_ENUM_FLAG);
51
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_asinh), 1, mxID(_asinh), XS_DONT_ENUM_FLAG);
52
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atan), 1, mxID(_atan), XS_DONT_ENUM_FLAG);
53
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atanh), 1, mxID(_atanh), XS_DONT_ENUM_FLAG);
54
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atan2), 2, mxID(_atan2), XS_DONT_ENUM_FLAG);
55
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cbrt), 1, mxID(_cbrt), XS_DONT_ENUM_FLAG);
56
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_ceil), 1, mxID(_ceil), XS_DONT_ENUM_FLAG);
57
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_clz32), 1, mxID(_clz32), XS_DONT_ENUM_FLAG);
58
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cos), 1, mxID(_cos), XS_DONT_ENUM_FLAG);
59
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cosh), 1, mxID(_cosh), XS_DONT_ENUM_FLAG);
60
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_exp), 1, mxID(_exp), XS_DONT_ENUM_FLAG);
61
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_expm1), 1, mxID(_expm1), XS_DONT_ENUM_FLAG);
62
24.5k
#if mxFloat16
63
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_f16round), 1, mxID(_f16round), XS_DONT_ENUM_FLAG);
64
24.5k
#endif
65
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_floor), 1, mxID(_floor), XS_DONT_ENUM_FLAG);
66
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_fround), 1, mxID(_fround), XS_DONT_ENUM_FLAG);
67
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_hypot), 2, mxID(_hypot_), XS_DONT_ENUM_FLAG);
68
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_idiv), 2, mxID(_idiv), XS_DONT_ENUM_FLAG);
69
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_idivmod), 2, mxID(_idivmod), XS_DONT_ENUM_FLAG);
70
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imod), 2, mxID(_imod), XS_DONT_ENUM_FLAG);
71
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imul), 2, mxID(_imul), XS_DONT_ENUM_FLAG);
72
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imuldiv), 2, mxID(_imuldiv), XS_DONT_ENUM_FLAG);
73
24.5k
#if mxECMAScript2023
74
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_irandom), 0, mxID(_irandom), XS_DONT_ENUM_FLAG);
75
24.5k
#endif
76
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_irem), 2, mxID(_irem), XS_DONT_ENUM_FLAG);
77
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log), 1, mxID(_log), XS_DONT_ENUM_FLAG);
78
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log1p), 1, mxID(_log1p), XS_DONT_ENUM_FLAG);
79
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log10), 1, mxID(_log10), XS_DONT_ENUM_FLAG);
80
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log2), 1, mxID(_log2), XS_DONT_ENUM_FLAG);
81
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_max), 2, mxID(_max), XS_DONT_ENUM_FLAG);
82
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_min), 2, mxID(_min), XS_DONT_ENUM_FLAG);
83
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_mod), 2, mxID(_mod), XS_DONT_ENUM_FLAG);
84
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_pow), 2, mxID(_pow), XS_DONT_ENUM_FLAG);
85
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_random), 0, mxID(_random), XS_DONT_ENUM_FLAG);
86
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_round), 1, mxID(_round), XS_DONT_ENUM_FLAG);
87
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sign), 1, mxID(_sign), XS_DONT_ENUM_FLAG);
88
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sin), 1, mxID(_sin), XS_DONT_ENUM_FLAG);
89
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sinh), 1, mxID(_sinh), XS_DONT_ENUM_FLAG);
90
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sqrt), 1, mxID(_sqrt), XS_DONT_ENUM_FLAG);
91
24.5k
#if mxECMAScript2026
92
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sumPrecise), 1, mxID(_sumPrecise), XS_DONT_ENUM_FLAG);
93
24.5k
#endif
94
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_tan), 1, mxID(_tan), XS_DONT_ENUM_FLAG);
95
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_tanh), 1, mxID(_tanh), XS_DONT_ENUM_FLAG);
96
24.5k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_trunc), 1, mxID(_trunc), XS_DONT_ENUM_FLAG);
97
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_E, mxID(_E), XS_GET_ONLY);
98
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_LN10, mxID(_LN10), XS_GET_ONLY);
99
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_LN2, mxID(_LN2), XS_GET_ONLY);
100
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_LOG10E, mxID(_LOG10E), XS_GET_ONLY);
101
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_LOG2E, mxID(_LOG2E), XS_GET_ONLY);
102
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_PI, mxID(_PI), XS_GET_ONLY);
103
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_SQRT1_2, mxID(_SQRT1_2), XS_GET_ONLY);
104
24.5k
  slot = fxNextNumberProperty(the, slot, C_M_SQRT2, mxID(_SQRT2), XS_GET_ONLY);
105
24.5k
  slot = fxNextStringXProperty(the, slot, "Math", mxID(_Symbol_toStringTag), XS_DONT_ENUM_FLAG | XS_DONT_SET_FLAG);
106
24.5k
  mxPull(mxMathObject);
107
//@@  c_srand((unsigned)c_time(0));
108
24.5k
}
109
110
#define mxNanResultIfNoArg \
111
2.16M
  if (mxArgc < 1) {  \
112
11.9k
    mxResult->kind = XS_NUMBER_KIND; \
113
11.9k
    mxResult->value.number = C_NAN; \
114
11.9k
    return; \
115
11.9k
  }
116
117
#define mxNanResultIfNoArg2 \
118
775k
  if (mxArgc < 2) {  \
119
59.6k
    mxResult->kind = XS_NUMBER_KIND; \
120
59.6k
    mxResult->value.number = C_NAN; \
121
59.6k
    return; \
122
59.6k
  }
123
124
void fx_Math_abs(txMachine* the)
125
29
{
126
29
  mxNanResultIfNoArg;
127
29
  fxToNumber(the, mxArgv(0));
128
29
  mxResult->kind = XS_NUMBER_KIND;
129
29
  mxResult->value.number = c_fabs(mxArgv(0)->value.number);
130
29
}
131
132
void fx_Math_acos(txMachine* the)
133
10.5k
{
134
10.5k
  mxNanResultIfNoArg;
135
9.00k
  fxToNumber(the, mxArgv(0));
136
9.00k
  mxResult->kind = XS_NUMBER_KIND;
137
9.00k
  mxResult->value.number = c_acos(mxArgv(0)->value.number);
138
9.00k
}
139
140
void fx_Math_acosh(txMachine* the)
141
9.96k
{
142
9.96k
  mxNanResultIfNoArg;
143
8.23k
  fxToNumber(the, mxArgv(0));
144
8.23k
  mxResult->kind = XS_NUMBER_KIND;
145
8.23k
  mxResult->value.number = c_acosh(mxArgv(0)->value.number);
146
8.23k
}
147
148
void fx_Math_asin(txMachine* the)
149
10.5k
{
150
10.5k
  mxNanResultIfNoArg;
151
9.53k
  fxToNumber(the, mxArgv(0));
152
9.53k
  mxResult->kind = XS_NUMBER_KIND;
153
9.53k
  mxResult->value.number = c_asin(mxArgv(0)->value.number);
154
9.53k
}
155
156
void fx_Math_asinh(txMachine* the)
157
60.0k
{
158
60.0k
  mxNanResultIfNoArg;
159
59.1k
  fxToNumber(the, mxArgv(0));
160
59.1k
  mxResult->kind = XS_NUMBER_KIND;
161
59.1k
  mxResult->value.number = c_asinh(mxArgv(0)->value.number);
162
59.1k
}
163
164
void fx_Math_atan(txMachine* the)
165
3.85k
{
166
3.85k
  mxNanResultIfNoArg;
167
2.00k
  fxToNumber(the, mxArgv(0));
168
2.00k
  mxResult->kind = XS_NUMBER_KIND;
169
2.00k
  mxResult->value.number = c_atan(mxArgv(0)->value.number);
170
2.00k
}
171
172
void fx_Math_atanh(txMachine* the)
173
470k
{
174
470k
  mxNanResultIfNoArg;
175
469k
  fxToNumber(the, mxArgv(0));
176
469k
  mxResult->kind = XS_NUMBER_KIND;
177
469k
  mxResult->value.number = c_atanh(mxArgv(0)->value.number);
178
469k
}
179
180
void fx_Math_atan2(txMachine* the)
181
654k
{
182
654k
  mxNanResultIfNoArg2;
183
654k
  fxToNumber(the, mxArgv(0));
184
654k
  fxToNumber(the, mxArgv(1));
185
654k
  mxResult->kind = XS_NUMBER_KIND;
186
654k
  mxResult->value.number = c_atan2(mxArgv(0)->value.number, mxArgv(1)->value.number);
187
654k
}
188
189
void fx_Math_cbrt(txMachine* the)
190
24
{
191
24
  mxNanResultIfNoArg;
192
24
  fxToNumber(the, mxArgv(0));
193
24
  mxResult->kind = XS_NUMBER_KIND;
194
24
  mxResult->value.number = c_cbrt(mxArgv(0)->value.number);
195
24
}
196
197
void fx_Math_ceil(txMachine* the)
198
175k
{
199
175k
  mxNanResultIfNoArg;
200
175k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
201
4
    mxResult->kind = XS_INTEGER_KIND;
202
4
    mxResult->value.integer = mxArgv(0)->value.integer;
203
4
    return;
204
4
  }
205
175k
  fxToNumber(the, mxArgv(0));
206
175k
  mxResult->kind = XS_NUMBER_KIND;
207
175k
  mxResult->value.number = c_ceil(mxArgv(0)->value.number);
208
175k
  fx_Math_toInteger(the);
209
175k
}
210
211
void fx_Math_clz32(txMachine* the)
212
60
{
213
60
  txUnsigned x = (mxArgc > 0) ? fxToUnsigned(the, mxArgv(0)) : 0;
214
60
  txInteger r;
215
60
  if (x)
216
#if mxWindows
217
  {
218
    _BitScanReverse(&r, x);
219
    r = 31 - r;
220
  }
221
#else
222
47
    r = __builtin_clz(x);
223
13
#endif  
224
13
  else
225
13
    r = 32;
226
60
  mxResult->kind = XS_INTEGER_KIND;
227
60
  mxResult->value.integer = r;
228
60
}
229
230
void fx_Math_cos(txMachine* the)
231
417k
{
232
417k
  mxNanResultIfNoArg;
233
417k
  fxToNumber(the, mxArgv(0));
234
417k
  mxResult->kind = XS_NUMBER_KIND;
235
417k
  mxResult->value.number = c_cos(mxArgv(0)->value.number);
236
417k
}
237
238
void fx_Math_cosh(txMachine* the)
239
10.5k
{
240
10.5k
  mxNanResultIfNoArg;
241
9.82k
  fxToNumber(the, mxArgv(0));
242
9.82k
  mxResult->kind = XS_NUMBER_KIND;
243
9.82k
  mxResult->value.number = c_cosh(mxArgv(0)->value.number);
244
9.82k
}
245
246
void fx_Math_exp(txMachine* the)
247
395k
{
248
395k
  mxNanResultIfNoArg;
249
394k
  fxToNumber(the, mxArgv(0));
250
394k
  mxResult->kind = XS_NUMBER_KIND;
251
394k
  mxResult->value.number = c_exp(mxArgv(0)->value.number);
252
394k
}
253
254
void fx_Math_expm1(txMachine* the)
255
20
{
256
20
  mxNanResultIfNoArg;
257
20
  fxToNumber(the, mxArgv(0));
258
20
  mxResult->kind = XS_NUMBER_KIND;
259
20
  mxResult->value.number = c_expm1(mxArgv(0)->value.number);
260
20
}
261
262
void fx_Math_floor(txMachine* the)
263
181k
{
264
181k
  mxNanResultIfNoArg;
265
181k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
266
56
    mxResult->kind = XS_INTEGER_KIND;
267
56
    mxResult->value.integer = mxArgv(0)->value.integer;
268
56
    return;
269
56
  }
270
181k
  fxToNumber(the, mxArgv(0));
271
181k
  mxResult->kind = XS_NUMBER_KIND;
272
181k
  mxResult->value.number = c_floor(mxArgv(0)->value.number);
273
181k
  fx_Math_toInteger(the);
274
181k
}
275
276
void fx_Math_fround(txMachine* the)
277
56
{
278
56
  float arg;
279
56
  mxNanResultIfNoArg;
280
53
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
281
6
    mxResult->kind = XS_INTEGER_KIND;
282
6
    mxResult->value.integer = mxArgv(0)->value.integer;
283
6
    return;
284
6
  }
285
47
  arg = (float)fxToNumber(the, mxArgv(0));
286
47
  mxResult->kind = XS_NUMBER_KIND;
287
47
  mxResult->value.number = arg;
288
47
}
289
290
void fx_Math_hypot(txMachine* the)
291
29
{
292
29
  if (mxArgc == 2) {
293
16
    fxToNumber(the, mxArgv(0));
294
16
    fxToNumber(the, mxArgv(1));
295
16
    mxResult->kind = XS_NUMBER_KIND;
296
16
    mxResult->value.number = c_hypot(mxArgv(0)->value.number, mxArgv(1)->value.number);
297
16
  }
298
13
  else {
299
13
    txInteger c = mxArgc, i;
300
13
    txNumber result = 0;
301
29
    for (i = 0; i < c; i++) {
302
16
      txNumber argument = fxToNumber(the, mxArgv(i));
303
16
      result += argument * argument;
304
16
    }
305
13
    mxResult->kind = XS_NUMBER_KIND;
306
13
    mxResult->value.number = c_sqrt(result);
307
13
  }
308
29
}
309
310
void fx_Math_idiv(txMachine* the)
311
26.3k
{
312
26.3k
  txInteger x = (mxArgc > 0) ? fxToInteger(the, mxArgv(0)) : 0;
313
26.3k
  txInteger y = (mxArgc > 1) ? fxToInteger(the, mxArgv(1)) : 0;
314
26.3k
  if (y == 0) {
315
25.7k
    mxResult->kind = XS_NUMBER_KIND;
316
25.7k
    mxResult->value.number = C_NAN;
317
25.7k
  }
318
580
  else {
319
580
    mxResult->kind = XS_INTEGER_KIND;
320
580
#if mxIntegerDivideOverflowException
321
580
    if ((x == (txInteger)0x80000000) && (y == -1))
322
5
      mxResult->value.integer = x;
323
575
    else
324
575
#endif
325
575
      mxResult->value.integer = x / y;
326
580
  }
327
26.3k
}
328
329
void fx_Math_idivmod(txMachine* the)
330
0
{
331
0
  mxTypeError("not available");
332
0
}
333
334
void fx_Math_imod(txMachine* the)
335
0
{
336
0
  txInteger x = (mxArgc > 0) ? fxToInteger(the, mxArgv(0)) : 0;
337
0
  txInteger y = (mxArgc > 1) ? fxToInteger(the, mxArgv(1)) : 0;
338
0
  if (y == 0) {
339
0
    mxResult->kind = XS_NUMBER_KIND;
340
0
    mxResult->value.number = C_NAN;
341
0
  }
342
0
  else {
343
0
    mxResult->kind = XS_INTEGER_KIND;
344
0
#if mxIntegerDivideOverflowException
345
0
    if ((x == (txInteger)0x80000000) && (y == -1))
346
0
      mxResult->value.integer = 0;
347
0
    else
348
0
#endif
349
0
      mxResult->value.integer = (x % y + y) % y;
350
0
  }
351
0
}
352
353
void fx_Math_imul(txMachine* the)
354
26.2k
{
355
26.2k
  txInteger x = (mxArgc > 0) ? fxToInteger(the, mxArgv(0)) : 0;
356
26.2k
  txInteger y = (mxArgc > 1) ? fxToInteger(the, mxArgv(1)) : 0;
357
26.2k
  mxResult->kind = XS_INTEGER_KIND;
358
26.2k
  mxResult->value.integer = x * y;
359
26.2k
}
360
361
void fx_Math_imuldiv(txMachine* the)
362
52.5k
{
363
52.5k
  txS8 x = (mxArgc > 0) ? fxToInteger(the, mxArgv(0)) : 0;
364
52.5k
  txS8 y = (mxArgc > 1) ? fxToInteger(the, mxArgv(1)) : 0;
365
52.5k
  txS8 z = (mxArgc > 2) ? fxToInteger(the, mxArgv(2)) : 0;
366
52.5k
  if (z == 0) {
367
26.2k
    mxResult->kind = XS_NUMBER_KIND;
368
26.2k
    mxResult->value.number = C_NAN;
369
26.2k
  }
370
26.2k
  else {
371
26.2k
    txS8 r = (x * y) / z;
372
26.2k
    mxResult->kind = XS_INTEGER_KIND;
373
26.2k
    mxResult->value.integer = (txInteger)(r & 0x00000000FFFFFFFF);
374
26.2k
  }
375
52.5k
}
376
377
void fx_Math_irandom(txMachine* the)
378
0
{
379
0
  double min = 0;
380
0
  double max = 2147483647;
381
0
  uint32_t result;
382
0
  if (mxArgc > 1) {
383
0
    min = (double)fxToInteger(the, mxArgv(0));
384
0
    max = (double)fxToInteger(the, mxArgv(1));
385
0
  }
386
0
  else if (mxArgc > 0) {
387
0
    max = (double)fxToInteger(the, mxArgv(0));
388
0
  }
389
0
  result = c_rand();
390
0
  while (result == C_RAND_MAX)
391
0
    result = c_rand();
392
0
  if (max < min)
393
0
    mxResult->value.integer = (txInteger)c_ceil(min + (((double)result / (double)C_RAND_MAX) * (max - min)));
394
0
  else
395
0
    mxResult->value.integer = (txInteger)c_floor(min + (((double)result / (double)C_RAND_MAX) * (max - min)));
396
0
  mxResult->kind = XS_INTEGER_KIND;
397
0
}
398
399
void fx_Math_irandom_secure(txMachine* the)
400
0
{
401
0
  mxTypeError("secure mode");
402
0
}
403
404
void fx_Math_irem(txMachine* the)
405
0
{
406
0
  txInteger x = (mxArgc > 0) ? fxToInteger(the, mxArgv(0)) : 0;
407
0
  txInteger y = (mxArgc > 1) ? fxToInteger(the, mxArgv(1)) : 0;
408
0
  if (y == 0) {
409
0
    mxResult->kind = XS_NUMBER_KIND;
410
0
    mxResult->value.number = C_NAN;
411
0
  }
412
0
  else {
413
0
    mxResult->kind = XS_INTEGER_KIND;
414
0
#if mxIntegerDivideOverflowException
415
0
    if ((x == (txInteger)0x80000000) && (y == -1))
416
0
      mxResult->value.integer = 0;
417
0
    else
418
0
#endif
419
0
      mxResult->value.integer = x % y;
420
0
  }
421
0
}
422
423
void fx_Math_log(txMachine* the)
424
31
{
425
31
  mxNanResultIfNoArg;
426
30
  fxToNumber(the, mxArgv(0));
427
30
  mxResult->kind = XS_NUMBER_KIND;
428
30
  mxResult->value.number = c_log(mxArgv(0)->value.number);
429
30
}
430
431
void fx_Math_log1p(txMachine* the)
432
2.06k
{
433
2.06k
  mxNanResultIfNoArg;
434
2.06k
  fxToNumber(the, mxArgv(0));
435
2.06k
  mxResult->kind = XS_NUMBER_KIND;
436
2.06k
  mxResult->value.number = c_log1p(mxArgv(0)->value.number);
437
2.06k
}
438
439
void fx_Math_log10(txMachine* the)
440
177
{
441
177
  mxNanResultIfNoArg;
442
176
  fxToNumber(the, mxArgv(0));
443
176
  mxResult->kind = XS_NUMBER_KIND;
444
176
  mxResult->value.number = c_log10(mxArgv(0)->value.number);
445
176
}
446
447
void fx_Math_log2(txMachine* the)
448
23
{
449
23
  mxNanResultIfNoArg;
450
23
  fxToNumber(the, mxArgv(0));
451
23
  mxResult->kind = XS_NUMBER_KIND;
452
#if mxAndroid
453
  mxResult->value.number = c_log(mxArgv(0)->value.number) / c_log(2);
454
#else
455
23
  mxResult->value.number = c_log2(mxArgv(0)->value.number);
456
23
#endif
457
23
}
458
459
void fx_Math_max(txMachine* the)
460
6.93k
{
461
6.93k
  txInteger c = mxArgc, i = 0;
462
6.93k
  mxResult->kind = XS_NUMBER_KIND;
463
6.93k
  mxResult->value.number = -((txNumber)C_INFINITY);
464
6.93k
  if (0 == c)
465
214
    return;
466
467
6.72k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
468
1.31k
    mxResult->kind = XS_INTEGER_KIND;
469
1.31k
    mxResult->value.integer = mxArgv(0)->value.integer;
470
1.31k
    i = 1;
471
1.31k
  }
472
473
22.8k
  for (; i < c; i++) {
474
16.7k
    txSlot *slot = mxArgv(i);
475
16.7k
    if (XS_INTEGER_KIND == mxResult->kind) {
476
1.55k
      if (XS_INTEGER_KIND == slot->kind) {
477
1.22k
        if (mxResult->value.integer < slot->value.integer) {
478
785
          mxResult->value.integer = slot->value.integer;
479
785
        }
480
1.22k
        continue;
481
1.22k
      }
482
324
      mxResult->kind = XS_NUMBER_KIND;
483
324
      mxResult->value.number = mxResult->value.integer;
484
324
    }
485
486
15.5k
    txNumber n = fxToNumber(the, slot);
487
15.5k
    if (c_isnan(n)) {
488
2.01k
      for (; i < c; i++)
489
1.39k
        fxToNumber(the, mxArgv(i));
490
623
      mxResult->value.number = C_NAN;
491
623
      return;
492
623
    }
493
14.9k
    if (mxResult->value.number < n)
494
9.47k
      mxResult->value.number = n;
495
5.44k
    else if ((mxResult->value.number == 0) && (n == 0)) {
496
382
      if (c_signbit(mxResult->value.number) != c_signbit(n))
497
80
        mxResult->value.number = 0;
498
382
    }
499
14.9k
  }
500
6.72k
}
501
502
void fx_Math_min(txMachine* the)
503
3.88k
{
504
3.88k
  txInteger c = mxArgc, i = 0;
505
3.88k
  mxResult->kind = XS_NUMBER_KIND;
506
3.88k
  mxResult->value.number = (txNumber)C_INFINITY;
507
3.88k
  if (0 == c)
508
1.19k
    return;
509
510
2.68k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
511
778
    mxResult->kind = XS_INTEGER_KIND;
512
778
    mxResult->value.integer = mxArgv(0)->value.integer;
513
778
    i = 1;
514
778
  }
515
516
7.21k
  for (; i < c; i++) {
517
6.63k
    txSlot *slot = mxArgv(i);
518
6.63k
    if (XS_INTEGER_KIND == mxResult->kind) {
519
905
      if (XS_INTEGER_KIND == slot->kind) {
520
449
        if (mxResult->value.integer > slot->value.integer)
521
194
          mxResult->value.integer = slot->value.integer;
522
449
        continue;
523
449
      }
524
456
      mxResult->kind = XS_NUMBER_KIND;
525
456
      mxResult->value.number = mxResult->value.integer;
526
456
    }
527
  
528
6.18k
    txNumber n = fxToNumber(the, slot);
529
6.18k
    if (c_isnan(n)) {
530
4.49k
      for (; i < c; i++)
531
2.39k
        fxToNumber(the, mxArgv(i));
532
2.10k
      mxResult->value.number = C_NAN;
533
2.10k
      return;
534
2.10k
    }
535
4.08k
    if (mxResult->value.number > n)
536
2.21k
      mxResult->value.number = n;
537
1.86k
    else if ((mxResult->value.number == 0) && (n == 0)) {
538
1.59k
      if (c_signbit(mxResult->value.number) != c_signbit(n))
539
1.55k
        mxResult->value.number = -0.0;
540
1.59k
    }
541
4.08k
  }
542
2.68k
}
543
544
void fx_Math_mod(txMachine* the)
545
0
{
546
0
  txNumber x = (mxArgc > 0) ? fxToNumber(the, mxArgv(0)) : 0;
547
0
  txNumber y = (mxArgc > 1) ? fxToNumber(the, mxArgv(1)) : 0;
548
0
  mxResult->kind = XS_NUMBER_KIND;
549
0
  mxResult->value.number = c_fmod((c_fmod(x, y) + y), y);
550
0
}
551
552
txNumber fx_pow(txNumber x, txNumber y)
553
3.98M
{
554
3.98M
  if (!c_isfinite(y) && (c_fabs(x) == 1.0))
555
463
    return C_NAN;
556
3.98M
  return c_pow(x, y);
557
3.98M
}
558
559
void fx_Math_pow(txMachine* the)
560
120k
{
561
120k
  txNumber x, y;
562
120k
  mxNanResultIfNoArg2;
563
61.2k
  x = fxToNumber(the, mxArgv(0));
564
61.2k
  y = fxToNumber(the, mxArgv(1));
565
61.2k
  mxResult->kind = XS_NUMBER_KIND;
566
61.2k
  mxResult->value.number = fx_pow(x, y);
567
61.2k
}
568
569
void fx_Math_random(txMachine* the)
570
103
{
571
103
  uint32_t result;
572
103
  do {
573
103
    result = c_rand();
574
103
  } while (result == C_RAND_MAX);
575
103
  mxResult->kind = XS_NUMBER_KIND;
576
103
  mxResult->value.number = (double)result / (double)C_RAND_MAX;
577
103
}
578
579
void fx_Math_random_secure(txMachine* the)
580
0
{
581
0
  mxTypeError("secure mode");
582
0
}
583
584
void fx_Math_round(txMachine* the)
585
6.07k
{
586
6.07k
    txNumber arg;
587
6.07k
  mxNanResultIfNoArg;
588
6.07k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
589
7
    mxResult->kind = XS_INTEGER_KIND;
590
7
    mxResult->value.integer = mxArgv(0)->value.integer;
591
7
    return;
592
7
  }
593
6.07k
  arg = fxToNumber(the, mxArgv(0));
594
6.07k
  if (c_isnormal(arg) && (-4503599627370495 < arg) && (arg < 4503599627370495)) { // 2 ** 52 - 1
595
5.02k
    if ((arg < -0.5) || (0.5 <= arg))
596
4.92k
      arg = c_floor(arg + 0.5);
597
107
    else if (arg < 0)
598
53
      arg = -0.0;
599
54
    else if (arg > 0)
600
54
      arg = 0.0;
601
5.02k
    }
602
6.07k
  mxResult->kind = XS_NUMBER_KIND;
603
6.07k
  mxResult->value.number = arg;
604
6.07k
  fx_Math_toInteger(the);
605
6.07k
}
606
607
void fx_Math_sqrt(txMachine* the)
608
1.26k
{
609
1.26k
  mxNanResultIfNoArg;
610
1.26k
  fxToNumber(the, mxArgv(0));
611
1.26k
  mxResult->kind = XS_NUMBER_KIND;
612
1.26k
  mxResult->value.number = c_sqrt(mxArgv(0)->value.number);
613
1.26k
}
614
615
void fx_Math_sumPrecise(txMachine* the)
616
738
{
617
738
  txSlot *iterable, *iterator, *next, *value;
618
738
  xsum_small_accumulator accumulator;
619
738
  txInteger count = 0;
620
738
  txBoolean flag = 1;
621
738
  if (mxArgc < 1)
622
1
    mxTypeError("no items");
623
737
  iterable = mxArgv(0);
624
737
  fxToInstance(the, iterable);
625
737
  mxTemporary(iterator);
626
737
  mxTemporary(next);
627
737
  fxGetIterator(the, iterable, iterator, next, 0);  
628
737
  xsum_small_init(&accumulator);
629
737
  mxTemporary(value);
630
3.49k
  while (fxIteratorNext(the, iterator, next, value)) {
631
2.78k
    mxTry(the) {
632
2.78k
      if (value->kind == XS_INTEGER_KIND) {
633
312
        flag = 0;
634
312
        xsum_small_add1(&accumulator, value->value.integer);
635
312
      }
636
2.46k
      else if (value->kind == XS_NUMBER_KIND) {
637
2.45k
        if (value->value.number != -0.0) {
638
2.44k
          flag = 0;
639
2.44k
          xsum_small_add1(&accumulator, value->value.number);
640
2.44k
        }
641
2.45k
      }
642
13
      else
643
13
        mxTypeError("items[%d]: not a number", count);
644
2.76k
      count++;
645
2.76k
    }
646
2.76k
    mxCatch(the) {
647
13
      fxIteratorReturn(the, iterator, 1);
648
13
      fxJump(the);
649
13
    }
650
2.78k
  }
651
711
  if (flag)
652
8
    mxResult->value.number = -0.0;
653
703
  else
654
703
    mxResult->value.number = xsum_small_round(&accumulator);
655
711
  mxResult->kind = XS_NUMBER_KIND;
656
711
}
657
658
void fx_Math_sign(txMachine* the)
659
11.5k
{
660
11.5k
  txNumber arg;
661
11.5k
  mxNanResultIfNoArg;
662
10.9k
  arg = fxToNumber(the, mxArgv(0));
663
10.9k
  mxResult->kind = XS_NUMBER_KIND;
664
10.9k
  if (c_isnan(arg))
665
5.54k
    mxResult->value.number = C_NAN;
666
5.45k
  else if (arg < 0)
667
1.27k
    mxResult->value.number = -1;
668
4.18k
  else if (arg > 0)
669
1.15k
    mxResult->value.number = 1;
670
3.02k
  else
671
3.02k
    mxResult->value.number = arg;
672
10.9k
  fx_Math_toInteger(the);
673
10.9k
}
674
675
void fx_Math_sin(txMachine* the)
676
22.3k
{
677
22.3k
  mxNanResultIfNoArg;
678
21.5k
  fxToNumber(the, mxArgv(0));
679
21.5k
  mxResult->kind = XS_NUMBER_KIND;
680
21.5k
  mxResult->value.number = c_sin(mxArgv(0)->value.number);
681
21.5k
}
682
683
void fx_Math_sinh(txMachine* the)
684
279k
{
685
279k
  mxNanResultIfNoArg;
686
279k
  fxToNumber(the, mxArgv(0));
687
279k
  mxResult->kind = XS_NUMBER_KIND;
688
279k
  mxResult->value.number = c_sinh(mxArgv(0)->value.number);
689
279k
}
690
691
void fx_Math_tan(txMachine* the)
692
25.1k
{
693
25.1k
  mxNanResultIfNoArg;
694
24.8k
  fxToNumber(the, mxArgv(0));
695
24.8k
  mxResult->kind = XS_NUMBER_KIND;
696
24.8k
  mxResult->value.number = c_tan(mxArgv(0)->value.number);
697
24.8k
}
698
699
void fx_Math_tanh(txMachine* the)
700
67.7k
{
701
67.7k
  mxNanResultIfNoArg;
702
67.6k
  fxToNumber(the, mxArgv(0));
703
67.6k
  mxResult->kind = XS_NUMBER_KIND;
704
67.6k
  mxResult->value.number = c_tanh(mxArgv(0)->value.number);
705
67.6k
}
706
707
void fx_Math_trunc(txMachine* the)
708
152
{
709
152
  mxNanResultIfNoArg;
710
149
  fxToNumber(the, mxArgv(0));
711
149
  mxResult->kind = XS_NUMBER_KIND;
712
149
  mxResult->value.number = c_trunc(mxArgv(0)->value.number);
713
149
  fx_Math_toInteger(the);
714
149
}
715
716
void fx_Math_toInteger(txMachine* the)
717
744k
{
718
744k
  txNumber number = mxResult->value.number;
719
744k
  txInteger integer = (txInteger)number;
720
744k
  txNumber check = integer;
721
744k
  if ((number == check) && (number || !c_signbit(number))) {
722
295k
    mxResult->value.integer = integer;
723
295k
    mxResult->kind = XS_INTEGER_KIND;
724
295k
  }
725
744k
}