-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbuiltin_math.c
99 lines (86 loc) · 2.67 KB
/
builtin_math.c
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
#include <math.h>
#include "builtin_math.h"
#include "cfunc.h"
#include "cons.h"
#include "eval.h"
#include "gc.h"
#include "number.h"
#include "object.h"
#include "symbol.h"
static pobject plus(pobject env, pobject params)
{
double result = 0;
while (is_cons(params)) {
pobject o = eval(env, cons_car(params));
if (is_number(o))
result += number_value(o);
params = cons_cdr(params);
}
return gc_add(number_new(result));
}
static pobject minus(pobject env, pobject params)
{
double result = 0;
pobject o = eval(env, cons_car(params));
if (is_number(o)) {
result = number_value(o);
params = cons_cdr(params);
if (is_cons(params)) {
while (is_cons(params)) {
pobject o = eval(env, cons_car(params));
if (is_number(o))
result -= number_value(o);
params = cons_cdr(params);
}
} else {
result = -result;
}
}
return gc_add(number_new(result));
}
static pobject mult(pobject env, pobject params)
{
double result = 1;
while (is_cons(params)) {
pobject o = eval(env, cons_car(params));
if (is_number(o))
result *= number_value(o);
params = cons_cdr(params);
}
return gc_add(number_new(result));
}
static pobject div(pobject env, pobject params)
{
double result = 0;
pobject o = eval(env, cons_car(params));
if (is_number(o)) {
result = number_value(o);
params = cons_cdr(params);
if (is_cons(params)) {
while (is_cons(params)) {
pobject o = eval(env, cons_car(params));
if (is_number(o))
result /= number_value(o); /* TODO: division by zero error handling */
params = cons_cdr(params);
}
}
}
return gc_add(number_new(result));
}
static pobject mod(pobject env, pobject params)
{
pobject o1 = eval(env, cons_nth(params, 1));
pobject o2 = eval(env, cons_nth(params, 2));
return (is_number(o1) && is_number(o2))
? gc_add((number_new( (int)number_value(o1) % (int)number_value(o2) )))
: NIL;
}
void builtin_math_init(pobject *env)
{
cons_assoc_set(env, symbol_intern("*pi*"), gc_add(number_new(M_PI)), 1);
cons_assoc_set(env, symbol_intern("+"), gc_add(cfunc_new(plus)), 1);
cons_assoc_set(env, symbol_intern("-"), gc_add(cfunc_new(minus)), 1);
cons_assoc_set(env, symbol_intern("*"), gc_add(cfunc_new(mult)), 1);
cons_assoc_set(env, symbol_intern("/"), gc_add(cfunc_new(div)), 1);
cons_assoc_set(env, symbol_intern("mod"), gc_add(cfunc_new(mod)), 1);
}