1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
|
// Lowering provides a specialisation when the base operand is a constant which
// is a power of two.
load(libdir + "math.js");
function test(x, y, z) {
function pow(x, y) { return `Math.pow(${x}, ${y})` };
function exp(x, y) { return `((${x}) ** ${y})` };
function make(fn, x, y, z) {
return Function(`
// Load from array to prevent constant-folding.
// (Ion is currently not smart enough to realise that both array
// values are the same.)
var ys = [${y}, ${y}];
var zs = [${z}, ${z}];
for (var i = 0; i < 1000; ++i) {
assertNear(${fn(x, "ys[i & 1]")}, zs[i & 1]);
}
`);
}
function double(v) {
// NB: numberToDouble() always returns a double value.
return `numberToDouble(${v})`;
}
function addTests(fn) {
tests.push(make(fn, x, y, z));
tests.push(make(fn, x, double(y), z));
tests.push(make(fn, double(x), y, z));
tests.push(make(fn, double(x), double(y), z));
}
var tests = [];
addTests(pow);
addTests(exp);
for (var i = 0; i < tests.length; ++i) {
for (var j = 0; j < 2; ++j) {
tests[i]();
}
}
}
function* range(a, b, fn) {
for (var i = a; i <= b; ++i) {
yield fn(i);
}
}
// Only 2^i with |i| ∈ {1..8} currently triggers the optimisation, but also test
// the next power-of-two values, 1 and 0, and negative base-of-two values.
var values = [
...range(1, 10, i => 2 ** i),
1,
0,
...range(1, 4, i => -(2 ** i)),
];
for (var x of values) {
test(x, 0, 1);
test(x, 1, x);
test(x, 2, x * x);
// 0**(negative) is Infinity, 1**(negative) is 1.
if (Math.abs(x) > 1) {
test(x, -1076, 0);
}
// (negative)**(odd-negative) is -0.
if (x > 1) {
test(x, -1075, 0);
}
}
|