Coverage Report

Created: 2026-05-16 07:11

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
23.0k
{
44
23.0k
  txSlot* slot;
45
23.0k
  mxPush(mxObjectPrototype);
46
23.0k
  slot = fxLastProperty(the, fxNewObjectInstance(the));
47
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_abs), 1, mxID(_abs), XS_DONT_ENUM_FLAG);
48
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_acos), 1, mxID(_acos), XS_DONT_ENUM_FLAG);
49
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_acosh), 1, mxID(_acosh), XS_DONT_ENUM_FLAG);
50
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_asin), 1, mxID(_asin), XS_DONT_ENUM_FLAG);
51
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_asinh), 1, mxID(_asinh), XS_DONT_ENUM_FLAG);
52
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atan), 1, mxID(_atan), XS_DONT_ENUM_FLAG);
53
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atanh), 1, mxID(_atanh), XS_DONT_ENUM_FLAG);
54
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_atan2), 2, mxID(_atan2), XS_DONT_ENUM_FLAG);
55
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cbrt), 1, mxID(_cbrt), XS_DONT_ENUM_FLAG);
56
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_ceil), 1, mxID(_ceil), XS_DONT_ENUM_FLAG);
57
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_clz32), 1, mxID(_clz32), XS_DONT_ENUM_FLAG);
58
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cos), 1, mxID(_cos), XS_DONT_ENUM_FLAG);
59
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_cosh), 1, mxID(_cosh), XS_DONT_ENUM_FLAG);
60
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_exp), 1, mxID(_exp), XS_DONT_ENUM_FLAG);
61
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_expm1), 1, mxID(_expm1), XS_DONT_ENUM_FLAG);
62
23.0k
#if mxFloat16
63
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_f16round), 1, mxID(_f16round), XS_DONT_ENUM_FLAG);
64
23.0k
#endif
65
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_floor), 1, mxID(_floor), XS_DONT_ENUM_FLAG);
66
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_fround), 1, mxID(_fround), XS_DONT_ENUM_FLAG);
67
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_hypot), 2, mxID(_hypot_), XS_DONT_ENUM_FLAG);
68
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_idiv), 2, mxID(_idiv), XS_DONT_ENUM_FLAG);
69
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_idivmod), 2, mxID(_idivmod), XS_DONT_ENUM_FLAG);
70
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imod), 2, mxID(_imod), XS_DONT_ENUM_FLAG);
71
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imul), 2, mxID(_imul), XS_DONT_ENUM_FLAG);
72
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_imuldiv), 2, mxID(_imuldiv), XS_DONT_ENUM_FLAG);
73
23.0k
#if mxECMAScript2023
74
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_irandom), 0, mxID(_irandom), XS_DONT_ENUM_FLAG);
75
23.0k
#endif
76
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_irem), 2, mxID(_irem), XS_DONT_ENUM_FLAG);
77
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log), 1, mxID(_log), XS_DONT_ENUM_FLAG);
78
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log1p), 1, mxID(_log1p), XS_DONT_ENUM_FLAG);
79
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log10), 1, mxID(_log10), XS_DONT_ENUM_FLAG);
80
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_log2), 1, mxID(_log2), XS_DONT_ENUM_FLAG);
81
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_max), 2, mxID(_max), XS_DONT_ENUM_FLAG);
82
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_min), 2, mxID(_min), XS_DONT_ENUM_FLAG);
83
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_mod), 2, mxID(_mod), XS_DONT_ENUM_FLAG);
84
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_pow), 2, mxID(_pow), XS_DONT_ENUM_FLAG);
85
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_random), 0, mxID(_random), XS_DONT_ENUM_FLAG);
86
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_round), 1, mxID(_round), XS_DONT_ENUM_FLAG);
87
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sign), 1, mxID(_sign), XS_DONT_ENUM_FLAG);
88
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sin), 1, mxID(_sin), XS_DONT_ENUM_FLAG);
89
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sinh), 1, mxID(_sinh), XS_DONT_ENUM_FLAG);
90
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sqrt), 1, mxID(_sqrt), XS_DONT_ENUM_FLAG);
91
23.0k
#if mxECMAScript2026
92
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_sumPrecise), 1, mxID(_sumPrecise), XS_DONT_ENUM_FLAG);
93
23.0k
#endif
94
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_tan), 1, mxID(_tan), XS_DONT_ENUM_FLAG);
95
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_tanh), 1, mxID(_tanh), XS_DONT_ENUM_FLAG);
96
23.0k
  slot = fxNextHostFunctionProperty(the, slot, mxCallback(fx_Math_trunc), 1, mxID(_trunc), XS_DONT_ENUM_FLAG);
97
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_E, mxID(_E), XS_GET_ONLY);
98
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_LN10, mxID(_LN10), XS_GET_ONLY);
99
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_LN2, mxID(_LN2), XS_GET_ONLY);
100
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_LOG10E, mxID(_LOG10E), XS_GET_ONLY);
101
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_LOG2E, mxID(_LOG2E), XS_GET_ONLY);
102
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_PI, mxID(_PI), XS_GET_ONLY);
103
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_SQRT1_2, mxID(_SQRT1_2), XS_GET_ONLY);
104
23.0k
  slot = fxNextNumberProperty(the, slot, C_M_SQRT2, mxID(_SQRT2), XS_GET_ONLY);
105
23.0k
  slot = fxNextStringXProperty(the, slot, "Math", mxID(_Symbol_toStringTag), XS_DONT_ENUM_FLAG | XS_DONT_SET_FLAG);
106
23.0k
  mxPull(mxMathObject);
107
//@@  c_srand((unsigned)c_time(0));
108
23.0k
}
109
110
#define mxNanResultIfNoArg \
111
1.89M
  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
722k
  if (mxArgc < 2) {  \
119
31
    mxResult->kind = XS_NUMBER_KIND; \
120
31
    mxResult->value.number = C_NAN; \
121
31
    return; \
122
31
  }
123
124
void fx_Math_abs(txMachine* the)
125
21
{
126
21
  mxNanResultIfNoArg;
127
21
  fxToNumber(the, mxArgv(0));
128
21
  mxResult->kind = XS_NUMBER_KIND;
129
21
  mxResult->value.number = c_fabs(mxArgv(0)->value.number);
130
21
}
131
132
void fx_Math_acos(txMachine* the)
133
9.30k
{
134
9.30k
  mxNanResultIfNoArg;
135
7.80k
  fxToNumber(the, mxArgv(0));
136
7.80k
  mxResult->kind = XS_NUMBER_KIND;
137
7.80k
  mxResult->value.number = c_acos(mxArgv(0)->value.number);
138
7.80k
}
139
140
void fx_Math_acosh(txMachine* the)
141
9.59k
{
142
9.59k
  mxNanResultIfNoArg;
143
7.86k
  fxToNumber(the, mxArgv(0));
144
7.86k
  mxResult->kind = XS_NUMBER_KIND;
145
7.86k
  mxResult->value.number = c_acosh(mxArgv(0)->value.number);
146
7.86k
}
147
148
void fx_Math_asin(txMachine* the)
149
9.54k
{
150
9.54k
  mxNanResultIfNoArg;
151
8.52k
  fxToNumber(the, mxArgv(0));
152
8.52k
  mxResult->kind = XS_NUMBER_KIND;
153
8.52k
  mxResult->value.number = c_asin(mxArgv(0)->value.number);
154
8.52k
}
155
156
void fx_Math_asinh(txMachine* the)
157
60.3k
{
158
60.3k
  mxNanResultIfNoArg;
159
59.4k
  fxToNumber(the, mxArgv(0));
160
59.4k
  mxResult->kind = XS_NUMBER_KIND;
161
59.4k
  mxResult->value.number = c_asinh(mxArgv(0)->value.number);
162
59.4k
}
163
164
void fx_Math_atan(txMachine* the)
165
3.52k
{
166
3.52k
  mxNanResultIfNoArg;
167
1.67k
  fxToNumber(the, mxArgv(0));
168
1.67k
  mxResult->kind = XS_NUMBER_KIND;
169
1.67k
  mxResult->value.number = c_atan(mxArgv(0)->value.number);
170
1.67k
}
171
172
void fx_Math_atanh(txMachine* the)
173
433k
{
174
433k
  mxNanResultIfNoArg;
175
431k
  fxToNumber(the, mxArgv(0));
176
431k
  mxResult->kind = XS_NUMBER_KIND;
177
431k
  mxResult->value.number = c_atanh(mxArgv(0)->value.number);
178
431k
}
179
180
void fx_Math_atan2(txMachine* the)
181
661k
{
182
661k
  mxNanResultIfNoArg2;
183
661k
  fxToNumber(the, mxArgv(0));
184
661k
  fxToNumber(the, mxArgv(1));
185
661k
  mxResult->kind = XS_NUMBER_KIND;
186
661k
  mxResult->value.number = c_atan2(mxArgv(0)->value.number, mxArgv(1)->value.number);
187
661k
}
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
174k
{
199
174k
  mxNanResultIfNoArg;
200
174k
  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
174k
  fxToNumber(the, mxArgv(0));
206
174k
  mxResult->kind = XS_NUMBER_KIND;
207
174k
  mxResult->value.number = c_ceil(mxArgv(0)->value.number);
208
174k
  fx_Math_toInteger(the);
209
174k
}
210
211
void fx_Math_clz32(txMachine* the)
212
59
{
213
59
  txUnsigned x = (mxArgc > 0) ? fxToUnsigned(the, mxArgv(0)) : 0;
214
59
  txInteger r;
215
59
  if (x)
216
#if mxWindows
217
  {
218
    _BitScanReverse(&r, x);
219
    r = 31 - r;
220
  }
221
#else
222
46
    r = __builtin_clz(x);
223
13
#endif  
224
13
  else
225
13
    r = 32;
226
59
  mxResult->kind = XS_INTEGER_KIND;
227
59
  mxResult->value.integer = r;
228
59
}
229
230
void fx_Math_cos(txMachine* the)
231
416k
{
232
416k
  mxNanResultIfNoArg;
233
416k
  fxToNumber(the, mxArgv(0));
234
416k
  mxResult->kind = XS_NUMBER_KIND;
235
416k
  mxResult->value.number = c_cos(mxArgv(0)->value.number);
236
416k
}
237
238
void fx_Math_cosh(txMachine* the)
239
8.96k
{
240
8.96k
  mxNanResultIfNoArg;
241
8.24k
  fxToNumber(the, mxArgv(0));
242
8.24k
  mxResult->kind = XS_NUMBER_KIND;
243
8.24k
  mxResult->value.number = c_cosh(mxArgv(0)->value.number);
244
8.24k
}
245
246
void fx_Math_exp(txMachine* the)
247
284k
{
248
284k
  mxNanResultIfNoArg;
249
283k
  fxToNumber(the, mxArgv(0));
250
283k
  mxResult->kind = XS_NUMBER_KIND;
251
283k
  mxResult->value.number = c_exp(mxArgv(0)->value.number);
252
283k
}
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
180k
{
264
180k
  mxNanResultIfNoArg;
265
180k
  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
180k
  fxToNumber(the, mxArgv(0));
271
180k
  mxResult->kind = XS_NUMBER_KIND;
272
180k
  mxResult->value.number = c_floor(mxArgv(0)->value.number);
273
180k
  fx_Math_toInteger(the);
274
180k
}
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
25
{
292
25
  if (mxArgc == 2) {
293
14
    fxToNumber(the, mxArgv(0));
294
14
    fxToNumber(the, mxArgv(1));
295
14
    mxResult->kind = XS_NUMBER_KIND;
296
14
    mxResult->value.number = c_hypot(mxArgv(0)->value.number, mxArgv(1)->value.number);
297
14
  }
298
11
  else {
299
11
    txInteger c = mxArgc, i;
300
11
    txNumber result = 0;
301
25
    for (i = 0; i < c; i++) {
302
14
      txNumber argument = fxToNumber(the, mxArgv(i));
303
14
      result += argument * argument;
304
14
    }
305
11
    mxResult->kind = XS_NUMBER_KIND;
306
11
    mxResult->value.number = c_sqrt(result);
307
11
  }
308
25
}
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
588
  else {
319
588
    mxResult->kind = XS_INTEGER_KIND;
320
588
#if mxIntegerDivideOverflowException
321
588
    if ((x == (txInteger)0x80000000) && (y == -1))
322
4
      mxResult->value.integer = x;
323
584
    else
324
584
#endif
325
584
      mxResult->value.integer = x / y;
326
588
  }
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
27
{
425
27
  mxNanResultIfNoArg;
426
26
  fxToNumber(the, mxArgv(0));
427
26
  mxResult->kind = XS_NUMBER_KIND;
428
26
  mxResult->value.number = c_log(mxArgv(0)->value.number);
429
26
}
430
431
void fx_Math_log1p(txMachine* the)
432
2.07k
{
433
2.07k
  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
153
{
441
153
  mxNanResultIfNoArg;
442
152
  fxToNumber(the, mxArgv(0));
443
152
  mxResult->kind = XS_NUMBER_KIND;
444
152
  mxResult->value.number = c_log10(mxArgv(0)->value.number);
445
152
}
446
447
void fx_Math_log2(txMachine* the)
448
26
{
449
26
  mxNanResultIfNoArg;
450
26
  fxToNumber(the, mxArgv(0));
451
26
  mxResult->kind = XS_NUMBER_KIND;
452
#if mxAndroid
453
  mxResult->value.number = c_log(mxArgv(0)->value.number) / c_log(2);
454
#else
455
26
  mxResult->value.number = c_log2(mxArgv(0)->value.number);
456
26
#endif
457
26
}
458
459
void fx_Math_max(txMachine* the)
460
6.59k
{
461
6.59k
  txInteger c = mxArgc, i = 0;
462
6.59k
  mxResult->kind = XS_NUMBER_KIND;
463
6.59k
  mxResult->value.number = -((txNumber)C_INFINITY);
464
6.59k
  if (0 == c)
465
214
    return;
466
467
6.37k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
468
968
    mxResult->kind = XS_INTEGER_KIND;
469
968
    mxResult->value.integer = mxArgv(0)->value.integer;
470
968
    i = 1;
471
968
  }
472
473
22.1k
  for (; i < c; i++) {
474
16.4k
    txSlot *slot = mxArgv(i);
475
16.4k
    if (XS_INTEGER_KIND == mxResult->kind) {
476
1.20k
      if (XS_INTEGER_KIND == slot->kind) {
477
878
        if (mxResult->value.integer < slot->value.integer) {
478
489
          mxResult->value.integer = slot->value.integer;
479
489
        }
480
878
        continue;
481
878
      }
482
323
      mxResult->kind = XS_NUMBER_KIND;
483
323
      mxResult->value.number = mxResult->value.integer;
484
323
    }
485
486
15.5k
    txNumber n = fxToNumber(the, slot);
487
15.5k
    if (c_isnan(n)) {
488
2.02k
      for (; i < c; i++)
489
1.40k
        fxToNumber(the, mxArgv(i));
490
628
      mxResult->value.number = C_NAN;
491
628
      return;
492
628
    }
493
14.9k
    if (mxResult->value.number < n)
494
9.48k
      mxResult->value.number = n;
495
5.43k
    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.37k
}
501
502
void fx_Math_min(txMachine* the)
503
2.37k
{
504
2.37k
  txInteger c = mxArgc, i = 0;
505
2.37k
  mxResult->kind = XS_NUMBER_KIND;
506
2.37k
  mxResult->value.number = (txNumber)C_INFINITY;
507
2.37k
  if (0 == c)
508
1.19k
    return;
509
510
1.17k
  if (XS_INTEGER_KIND == mxArgv(0)->kind) {
511
768
    mxResult->kind = XS_INTEGER_KIND;
512
768
    mxResult->value.integer = mxArgv(0)->value.integer;
513
768
    i = 1;
514
768
  }
515
516
2.74k
  for (; i < c; i++) {
517
2.19k
    txSlot *slot = mxArgv(i);
518
2.19k
    if (XS_INTEGER_KIND == mxResult->kind) {
519
921
      if (XS_INTEGER_KIND == slot->kind) {
520
457
        if (mxResult->value.integer > slot->value.integer)
521
194
          mxResult->value.integer = slot->value.integer;
522
457
        continue;
523
457
      }
524
464
      mxResult->kind = XS_NUMBER_KIND;
525
464
      mxResult->value.number = mxResult->value.integer;
526
464
    }
527
  
528
1.73k
    txNumber n = fxToNumber(the, slot);
529
1.73k
    if (c_isnan(n)) {
530
1.53k
      for (; i < c; i++)
531
910
        fxToNumber(the, mxArgv(i));
532
622
      mxResult->value.number = C_NAN;
533
622
      return;
534
622
    }
535
1.11k
    if (mxResult->value.number > n)
536
725
      mxResult->value.number = n;
537
388
    else if ((mxResult->value.number == 0) && (n == 0)) {
538
92
      if (c_signbit(mxResult->value.number) != c_signbit(n))
539
50
        mxResult->value.number = -0.0;
540
92
    }
541
1.11k
  }
542
1.17k
}
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.59M
{
554
3.59M
  if (!c_isfinite(y) && (c_fabs(x) == 1.0))
555
444
    return C_NAN;
556
3.59M
  return c_pow(x, y);
557
3.59M
}
558
559
void fx_Math_pow(txMachine* the)
560
61.1k
{
561
61.1k
  txNumber x, y;
562
61.1k
  mxNanResultIfNoArg2;
563
61.1k
  x = fxToNumber(the, mxArgv(0));
564
61.1k
  y = fxToNumber(the, mxArgv(1));
565
61.1k
  mxResult->kind = XS_NUMBER_KIND;
566
61.1k
  mxResult->value.number = fx_pow(x, y);
567
61.1k
}
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.06k
  arg = fxToNumber(the, mxArgv(0));
594
6.06k
  if (c_isnormal(arg) && (-4503599627370495 < arg) && (arg < 4503599627370495)) { // 2 ** 52 - 1
595
5.02k
    if ((arg < -0.5) || (0.5 <= arg))
596
4.91k
      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.06k
  mxResult->kind = XS_NUMBER_KIND;
603
6.06k
  mxResult->value.number = arg;
604
6.06k
  fx_Math_toInteger(the);
605
6.06k
}
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
598
{
617
598
  txSlot *iterable, *iterator, *next, *value;
618
598
  xsum_small_accumulator accumulator;
619
598
  txInteger count = 0;
620
598
  txBoolean flag = 1;
621
598
  if (mxArgc < 1)
622
1
    mxTypeError("no items");
623
597
  iterable = mxArgv(0);
624
597
  fxToInstance(the, iterable);
625
597
  mxTemporary(iterator);
626
597
  mxTemporary(next);
627
597
  fxGetIterator(the, iterable, iterator, next, 0);  
628
597
  xsum_small_init(&accumulator);
629
597
  mxTemporary(value);
630
2.79k
  while (fxIteratorNext(the, iterator, next, value)) {
631
2.22k
    mxTry(the) {
632
2.22k
      if (value->kind == XS_INTEGER_KIND) {
633
219
        flag = 0;
634
219
        xsum_small_add1(&accumulator, value->value.integer);
635
219
      }
636
2.00k
      else if (value->kind == XS_NUMBER_KIND) {
637
1.99k
        if (value->value.number != -0.0) {
638
1.98k
          flag = 0;
639
1.98k
          xsum_small_add1(&accumulator, value->value.number);
640
1.98k
        }
641
1.99k
      }
642
13
      else
643
13
        mxTypeError("items[%d]: not a number", count);
644
2.21k
      count++;
645
2.21k
    }
646
2.21k
    mxCatch(the) {
647
13
      fxIteratorReturn(the, iterator, 1);
648
13
      fxJump(the);
649
13
    }
650
2.22k
  }
651
571
  if (flag)
652
8
    mxResult->value.number = -0.0;
653
563
  else
654
563
    mxResult->value.number = xsum_small_round(&accumulator);
655
571
  mxResult->kind = XS_NUMBER_KIND;
656
571
}
657
658
void fx_Math_sign(txMachine* the)
659
11.1k
{
660
11.1k
  txNumber arg;
661
11.1k
  mxNanResultIfNoArg;
662
10.5k
  arg = fxToNumber(the, mxArgv(0));
663
10.5k
  mxResult->kind = XS_NUMBER_KIND;
664
10.5k
  if (c_isnan(arg))
665
5.51k
    mxResult->value.number = C_NAN;
666
5.07k
  else if (arg < 0)
667
761
    mxResult->value.number = -1;
668
4.31k
  else if (arg > 0)
669
1.15k
    mxResult->value.number = 1;
670
3.15k
  else
671
3.15k
    mxResult->value.number = arg;
672
10.5k
  fx_Math_toInteger(the);
673
10.5k
}
674
675
void fx_Math_sin(txMachine* the)
676
20.1k
{
677
20.1k
  mxNanResultIfNoArg;
678
19.3k
  fxToNumber(the, mxArgv(0));
679
19.3k
  mxResult->kind = XS_NUMBER_KIND;
680
19.3k
  mxResult->value.number = c_sin(mxArgv(0)->value.number);
681
19.3k
}
682
683
void fx_Math_sinh(txMachine* the)
684
166k
{
685
166k
  mxNanResultIfNoArg;
686
166k
  fxToNumber(the, mxArgv(0));
687
166k
  mxResult->kind = XS_NUMBER_KIND;
688
166k
  mxResult->value.number = c_sinh(mxArgv(0)->value.number);
689
166k
}
690
691
void fx_Math_tan(txMachine* the)
692
23.8k
{
693
23.8k
  mxNanResultIfNoArg;
694
23.5k
  fxToNumber(the, mxArgv(0));
695
23.5k
  mxResult->kind = XS_NUMBER_KIND;
696
23.5k
  mxResult->value.number = c_tan(mxArgv(0)->value.number);
697
23.5k
}
698
699
void fx_Math_tanh(txMachine* the)
700
68.4k
{
701
68.4k
  mxNanResultIfNoArg;
702
68.3k
  fxToNumber(the, mxArgv(0));
703
68.3k
  mxResult->kind = XS_NUMBER_KIND;
704
68.3k
  mxResult->value.number = c_tanh(mxArgv(0)->value.number);
705
68.3k
}
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
700k
{
718
700k
  txNumber number = mxResult->value.number;
719
700k
  txInteger integer = (txInteger)number;
720
700k
  txNumber check = integer;
721
700k
  if ((number == check) && (number || !c_signbit(number))) {
722
292k
    mxResult->value.integer = integer;
723
292k
    mxResult->kind = XS_INTEGER_KIND;
724
292k
  }
725
700k
}