-
Notifications
You must be signed in to change notification settings - Fork 2
/
calculator-backends.js
595 lines (532 loc) · 20.5 KB
/
calculator-backends.js
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
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
// # calculator-backends.js
//
// things to do with a simple calculator parser
//
// [calculator-parser.js](calculator-parser.html) contains a simple parser.
// It contains enough code that you can actually do some basic math with it.
// But what else can you do with a parser?
//
// This file contains seven different applications of
// the calculator parser.
//
// [Try them out.](../calculator.html)
// ## Code as data
// ### 1. Show the JSON
//
// Unmodified, the parser simply builds a tree describing the input
// formula. This is called an abstract syntax tree, or AST.
//
function convertToJSON(code) {
return parse(code);
}
// ### 2. Convert to DOM
//
// Of course one of the nice things about JSON is that there are many ways to
// display it. Here’s code that spits out DOM nodes that show how the program
// would look in Scratch. (!)
// Helper function to create DOM elements.
function span(className, contents) {
var e = document.createElement("span");
e.className = className;
for (var i = 0; i < contents.length; i++) {
var kid = contents[i];
if (typeof kid === "string")
kid = document.createTextNode(kid);
e.appendChild(kid);
}
return e;
}
function convertToDOM(code) {
var fancyOperator = {
"+": "+",
"-": "\u2212", // −
"*": "\u00d7", // ×
"/": "\u00f7" // ÷
};
function convert(obj) {
switch (obj.type) {
case "number":
return span("num", [obj.value]);
case "+": case "-": case "*": case "/":
return span("expr", [convert(obj.left),
fancyOperator[obj.type],
convert(obj.right)]);
case "name":
return span("var", [obj.id]);
}
}
return convert(parse(code));
}
// ### 3. MathML output
//
// One more riff on this theme: How about generating beautiful MathML output?
// (Unfortunately, some browsers still do not support MathML. Firefox does.)
// The hardest part of this was figuring out how to make MathML elements.
// Here’s some code to help with that.
var mathml = "http://www.w3.org/1998/Math/MathML";
function mo(s) {
var e = document.createElementNS(mathml, "mo");
var t = document.createTextNode(s);
e.appendChild(t);
return {prec: 3, element: e};
}
// Create a new MathML DOM element of the specified type and contents.
// `precedence` is used to determine whether or not to add parentheses around
// any of the contents. If it’s `null`, no parentheses are added.
function make(name, precedence, contents) {
var e = document.createElementNS(mathml, name);
for (var i = 0; i < contents.length; i++) {
var kid = contents[i];
var node;
if (typeof kid === "string") {
node = document.createTextNode(kid);
} else {
// If precedence is non-null and higher than this child’s
// precedence, wrap the child in parentheses.
if (precedence !== null
&& (kid.prec < precedence
|| (kid.prec == precedence && i != 0)))
{
kid = make("mrow", null, [mo("("), kid, mo(")")]);
}
node = kid.element;
}
e.appendChild(node);
}
if (precedence === null)
precedence = 3;
return {prec: precedence, element: e};
}
function convertToMathML(code) {
function convert(obj) {
switch (obj.type) {
case "number":
return make("mn", 3, [obj.value]);
case "name":
return make("mi", 3, [obj.id]);
case "+":
return make("mrow", 1, [convert(obj.left),
make("mo", 3, ["+"]),
convert(obj.right)]);
case "-":
return make("mrow", 1, [convert(obj.left),
make("mo", 3, ["-"]),
convert(obj.right)]);
case "*":
return make("mrow", 2, [convert(obj.left),
convert(obj.right)]);
case "/":
return make("mfrac", null, [convert(obj.left), convert(obj.right)]);
}
};
var e = convert(parse(code));
return make("math", null, [e]);
}
// ## Interpreters
// ### 4. Evaluate using floating-point numbers
// Now let’s try actually performing some computation using the program we
// read. This behaves like a stripped-down version of JavaScript `eval()`.
function evaluateAsFloat(code) {
var variables = Object.create(null);
variables.e = Math.E;
variables.pi = Math.PI;
function evaluate(obj) {
switch (obj.type) {
case "number": return parseInt(obj.value);
case "name": return variables[obj.id] || 0;
case "+": return evaluate(obj.left) + evaluate(obj.right);
case "-": return evaluate(obj.left) - evaluate(obj.right);
case "*": return evaluate(obj.left) * evaluate(obj.right);
case "/": return evaluate(obj.left) / evaluate(obj.right);
}
}
return evaluate(parse(code));
}
assert.strictEqual(evaluateAsFloat("2 + 2"), 4);
assert.strictEqual(evaluateAsFloat("3 * 4 * 5"), 60);
assert.strictEqual(evaluateAsFloat("5 * (2 + 2)"), 20);
// ### 5. Evaluate using precise fraction arithmetic
//
// Our little language is a tiny subset of JavaScript. But that doesn’t meant
// it has to behave exactly like JavaScript. This is our language.
// It can behave however we want.
// So how about a calculator that does arbitrary precision arithmetic?
// Let’s start by defining a `Fraction` class...
var BigInteger = require('biginteger').BigInteger;
function gcd(a, b) {
while (!b.isZero()) {
var tmp = a;
a = b;
b = tmp.remainder(b);
}
return a;
}
function Fraction(n, d) {
if (d === undefined)
d = new BigInteger(1);
var x = gcd(n.abs(), d); // Simplify the fraction.
this.n = n.divide(x);
this.d = d.divide(x);
}
// …and some Fraction methods. You learned these techniques in grade school,
// though you may have forgotten some of them.
Fraction.prototype = {
add: function (x) {
return new Fraction(this.n.multiply(x.d).add(x.n.multiply(this.d)),
this.d.multiply(x.d));
},
negate: function (x) {
return new Fraction(this.n.negate(), this.d);
},
sub: function (x) {
return this.add(x.negate());
},
mul: function (x) {
return new Fraction(this.n.multiply(x.n), this.d.multiply(x.d));
},
div: function (x) {
return new Fraction(this.n.multiply(x.d), this.d.multiply(x.n));
},
toString: function () {
var ns = this.n.toString(), ds = this.d.toString();
if (ds === "1")
return ns;
else
return ns + "/" + ds;
}
};
// Now simply write an interpreter that computes the results using `Fraction`
// objects rather than JavaScript numbers. It’s almost too easy.
function evaluateAsFraction(code) {
function evaluate(obj) {
switch (obj.type) {
case "number": return new Fraction(new BigInteger(obj.value));
case "+": return evaluate(obj.left).add(evaluate(obj.right));
case "-": return evaluate(obj.left).sub(evaluate(obj.right));
case "*": return evaluate(obj.left).mul(evaluate(obj.right));
case "/": return evaluate(obj.left).div(evaluate(obj.right));
case "name": throw new SyntaxError("no variables in fraction mode, sorry");
}
}
return evaluate(parse(code));
}
// Our tiny programming language is suddenly doing something JavaScript itself
// doesn’t do: arithmetic with exact (not floating-point) results. Tests:
assert.strictEqual(evaluateAsFraction("1 / 3").toString(), "1/3");
assert.strictEqual(evaluateAsFraction("(2/3) * (3/2)").toString(), "1");
assert.strictEqual(evaluateAsFraction("1/7 + 4/7 + 2/7").toString(), "1");
assert.strictEqual(
evaluateAsFraction("5996788328646786302319492 / 2288327879043508396784319").toString(),
"324298349324/123749732893");
// ## Compilers
// ### 6. JavaScript function output
// This is just to show some very basic code generation.
//
// Code generation for a real compiler will be harder, because the target
// language is typically quite a bit different from the source language. Here
// they are virtually identical, so code generation is very easy.
//
function compileToJSFunction(code) {
function emit(ast) {
switch (ast.type) {
case "number":
return ast.value;
case "name":
if (ast.id !== "x")
throw SyntaxError("only the name 'x' is allowed");
return ast.id;
case "+": case "-": case "*": case "/":
return "(" + emit(ast.left) + " " + ast.type + " " + emit(ast.right) + ")";
}
}
return Function("x", "return " + emit(parse(code)) + ";");
}
assert.strictEqual(compileToJSFunction("x*x - 2*x + 1")(1), 0);
assert.strictEqual(compileToJSFunction("x*x - 2*x + 1")(2), 1);
assert.strictEqual(compileToJSFunction("x*x - 2*x + 1")(3), 4);
assert.strictEqual(compileToJSFunction("x*x - 2*x + 1")(4), 9);
// ### 7. Complex function output
// This one returns a JS function that operates on complex numbers.
//
// This is a more advanced example of a compiler. It has a lowering phase
// and a few simple optimizations.
//
// The input string `code` is a formula, for example, `"(z+1)/(z-1)"`,
// that uses a complex variable `z`.
//
// This returns a JS function that takes two arguments, `z_re` and `z_im`,
// the real and imaginary parts of a complex number *z*,
// and returns an object `{re: some number, im: some number}`,
// representing the complex result of computing the input formula,
// (*z*+1)/(*z*-1). Here is the actual output in that case:
//
// (function anonymous(z_re, z_im) {
// var t0 = (z_re+1);
// var t1 = (z_re-1);
// var t2 = (z_im*z_im);
// var t3 = ((t1*t1)+t2);
// return {re: (((t0*t1)+t2)/t3), im: (((z_im*t1)-(t0*z_im))/t3)};
// })
//
function compileToComplexFunction(code) {
// The first thing here is a lot of code about "values". This takes some
// explanation.
//
// `compileToComplexFunction` works in three steps.
//
// 1. It uses `parse` as its front end. We get an AST that represents
// operations on complex numbers.
//
// 2. Then, it *lowers* the AST to an *intermediate representation*, or
// *IR*, that’s sort of halfway between the AST and JS code.
//
// The IR here is an array of *values*, basic arithmetic operations on
// plain old JS floating-point numbers. If we end up needing JS to
// compute `(z_re + 1) * (z_re - 1)`, then each subexpression there ends
// up being a value, represented by an object in the IR.
//
// Once we have the IR, we can throw away the AST. The IR represents the
// same formula, but in a different form. Lowering breaks down complex
// arithmetic into sequences of JS numeric operations. JS won’t know
// it’s doing complex math.
//
// 3. Lastly, we convert the IR to JS code.
//
// **Why IR?** We don’t have to do it this way; we could define a class
// `Complex`, with methods `.add()`, `.sub()`, etc., and generate JS code
// that uses that. Our back-end would be less than 20 lines. But
// using IR helps us generate faster JS code. In particular, we’ll avoid
// allocating any objects for intermediate results and without any need
// for a `Complex` runtime library.
// #### About the IR
//
// Before we implement `ast_to_ir()`, we need to define objects
// representing values. (We could just use strings of JS code, but it turns
// out to be really complicated, and it’s hard to apply even basic
// optimizations to strings.)
//
// We call these instructions "values". Each value represents a single JS
// expression that evaluates to a number. A value is one of:
//
// 1. `z_re` or `z_im`, the real or imaginary part of the argument z; or
// 2. a numeric constant; or
// 3. an arithmetic operation, one of `+ - * /`, on two previously-
// calculated values.
//
// And each value is represented by a plain old JSON object, as follows:
//
// 1. `{type: "arg", arg0: "z_re"` or `"z_im"}`
// 2. `{type: "number", arg0: (string picture of number)}`
// 3. `{type: (one of "+" "-" "*" "/"), arg0: (int), arg1: (int)}`
//
// All values are stored sequentially in the array `values`. The two ints
// `v.arg0` and `v.arg1` in an arithmetic value are indexes into `values`.
//
// The main reason to store all values in a flat array, rather than a tree,
// is for "common subexpression elimination" (see valueToIndex).
//
var values = [
{type: "arg", arg0: "z_re", arg1: null},
{type: "arg", arg0: "z_im", arg1: null}
];
// **valuesEqual(*n1*, *n2*)** returns true if the two arguments *n1* and
// *n2* are equal IR nodes—that is, if it would be redundant to compute them
// both, because they are certain to have the same result.
//
// (This could be more sophisticated; it does not detect, for example, that
// `a+1` is equal to `1+a`.)
//
function valuesEqual(n1, n2) {
return n1.type === n2.type && n1.arg0 === n2.arg0 && n1.arg1 === n2.arg1;
}
// **valueToIndex(*v*)** ensures that *v* is in *values*. Returns the index
// **of *v* within the array.
function valueToIndex(v) {
// Common subexpression elimination. If we have already computed this
// exact value, instead of computing it again, just reuse the already-
// computed value.
for (var i = 0; i < values.length; i++) {
if (valuesEqual(values[i], v))
return i;
}
// We have not already computed this value, so add the value to
// the list.
values.push(v);
return values.length - 1;
}
function num(s) {
return valueToIndex({type: "number", arg0: s, arg1: null});
}
function op(op, a, b) {
return valueToIndex({type: op, arg0: a, arg1: b });
}
// A few predicates used to implement some simple optimizations below.
function isNumber(i) {
return values[i].type === "number";
}
function isZero(i) {
return isNumber(i) && values[i].arg0 === "0";
}
function isOne(i) {
return isNumber(i) && values[i].arg0 === "1";
}
// Each of these four functions ensures that an IR node for a certain
// computation has been added to `values`, and returns the index of that
// node.
function add(a, b) {
if (isZero(a)) // simplify (0+b) to b
return b;
if (isZero(b)) // simplify (a+0) to a
return a;
if (isNumber(a) && isNumber(b)) // constant-fold (1+2) to 3
return num(String(Number(values[a].arg0) + Number(values[b].arg0)));
return op("+", a, b);
}
function sub(a, b) {
if (isZero(b)) // simplify (a-0) to a
return a;
if (isNumber(a) && isNumber(b)) // constant-fold (3-2) to 1
return num(String(Number(values[a].arg0) - Number(values[b].arg0)));
return op("-", a, b);
}
function mul(a, b) {
if (isZero(a)) // simplify 0*b to 0
return a;
if (isZero(b)) // simplify a*0 to 0
return b;
if (isOne(a)) // simplify 1*b to b
return b;
if (isOne(b)) // simplify a*1 to a
return a;
if (isNumber(a) && isNumber(b)) // constant-fold (2*2) to 4
return num(String(Number(values[a].arg0) * Number(values[b].arg0)));
return op("*", a, b);
}
function div(a, b) {
if (isOne(b)) // simplify a/1 to a
return a;
if (isNumber(a) && isNumber(b) && !isZero(b)) // constant-fold 4/2 to 2
return num(String(Number(values[a].arg0) / Number(values[b].arg0)));
return op("/", a, b);
}
// **ast_to_ir(*obj*)** reduces *obj*, which represents an operation on
// complex numbers, to a sequence of operations on floating-point numbers.
//
// It adds IR nodes representing those operations to *values*.
//
// Returns an object `{im: int, re: int}` giving the index, in *values*, of
// the IR nodes representing the real part and the imaginary part of the
// answer.
//
function ast_to_ir(obj) {
switch (obj.type) {
case "number":
return {re: num(obj.value), im: num("0")};
// Complex arithmetic. Start by calling `ast_to_ir` recursively on the
// operands. The rest is just standard formulas:
//
// Re(*a* ± *b*) = Re(*a*) ± Re(*b*)<br>
// Im(*a* ± *b*) = Im(*a*) ± Im(*b*)
//
case "+": case "-":
var a = ast_to_ir(obj.left), b = ast_to_ir(obj.right);
var f = (obj.type === "+" ? add : sub);
return {
re: f(a.re, b.re),
im: f(a.im, b.im)
};
// Multiplication:
//
// Re(*a* × *b*) = Re(*a*) × Re(*b*) − Im(*a*) × Im(*b*)<br>
// Im(*a* × *b*) = Re(*a*) × Im(*b*) + Im(*a*) × Re(*b*)
//
case "*":
var a = ast_to_ir(obj.left), b = ast_to_ir(obj.right);
return {
re: sub(mul(a.re, b.re), mul(a.im, b.im)),
im: add(mul(a.re, b.im), mul(a.im, b.re))
};
// Division:
//
// Re(*a* ÷ *b*) = (Re(*a*) × Re(*b*) + Im(*a*) × Im(*b*)) ÷ *t*<br>
// Im(*a* ÷ *b*) = (Im(*a*) × Re(*b*) − Re(*a*) × Im(*b*)) ÷ *t*<br>
// where *t* = Re(*b*)² + Im(*b*)²
//
case "/":
var a = ast_to_ir(obj.left), b = ast_to_ir(obj.right);
var t = add(mul(b.re, b.re), mul(b.im, b.im));
return {
re: div(add(mul(a.re, b.re), mul(a.im, b.im)), t),
im: div(sub(mul(a.im, b.re), mul(a.re, b.im)), t)
};
case "name":
if (obj.id === "i")
return {re: num("0"), im: num("1")};
// A little subtle here: `values[0]` is the IR node for `z_re`;
// `values[1]` is `z_im`.
if (obj.id === "z")
return {re: 0, im: 1};
throw SyntaxError("undefined variable: " + obj.id);
}
}
// Figure out how many times each value is used in subsequent calculations.
// This is how we avoid emitting all the code for a value every time it is
// used. As we generate JS, whenever we reach a value that is used multiple
// times, we’ll write out a `var` statement.
function computeUseCounts(values) {
var binaryOps = {"+": 1, "-": 1, "*": 1, "/": 1};
var useCounts = [];
for (var i = 0; i < values.length; i++) {
useCounts[i] = 0;
var node = values[i];
if (node.type in binaryOps) {
useCounts[node.arg0]++;
useCounts[node.arg1]++;
}
}
return useCounts;
}
// Step 3: Convert the IR to JavaScript.
function ir_to_js(values, result) {
var useCounts = computeUseCounts(values);
var code = "";
var next_temp_id = 0;
var js = [];
for (var i = 0; i < values.length; i++) {
var node = values[i];
if (node.type === "number" || node.type === "arg") {
js[i] = node.arg0;
} else {
js[i] = "(" + js[node.arg0] + node.type + js[node.arg1] + ")";
if (useCounts[i] > 1) {
var name = "t" + next_temp_id++;
code += "var " + name + " = " + js[i] + ";\n";
js[i] = name;
}
}
}
code += "return {re: " + js[result.re] + ", im: " + js[result.im] + "};\n";
return code;
}
var ast = parse(code);
var result = ast_to_ir(ast);
var code = ir_to_js(values, result);
console.log(code);
return Function("z_re, z_im", code);
/*
I had planned to have this generate asm.js code for extra speed, but it’s
so fast already that unless I can think of a more computationally
intensive task, there is no need.
*/
}
// The last bit of code here simply stores all seven back ends in one place
// where other code can get to them.
var parseModes = {
json: convertToJSON,
blocks: convertToDOM,
mathml: convertToMathML,
calc: evaluateAsFloat,
fraction: evaluateAsFraction,
graph: compileToJSFunction,
complex: compileToComplexFunction
};