-
Notifications
You must be signed in to change notification settings - Fork 1
/
mettamorph.scm
383 lines (329 loc) · 13.5 KB
/
mettamorph.scm
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
;; Run with CHICKEN Scheme!
(import srfi-1 srfi-69) ;filter, hashtable
(import amb amb-extras) ;amb to implement superpose and amb1 to implements its nesting behavior
(import matchable) ;let/case constructs as match-lambda with deconstruction
(import (chicken flonum)) ;floating point options
(import (chicken type)) ;type system
(import (chicken process) (chicken file posix)) ;fork, pipe, file API for hyperpose
;; COLLAPSE AND SUPERPOSE
;""""""""""""""""""""""""
;collapse: using Scheme's 'amb-collect'
(define-syntax collapse
(syntax-rules ()
((_ args)
(amb-collect (auto-list1 args)))))
;superpose-helper enforces nested superpose compatibility
;by flatting out the nested superpose calls before passing it to amb
(define-syntax superpose-helper
(syntax-rules (superpose)
((_ (superpose (argi ...)))
(amb (superpose-helper argi) ...))
((_ arg)
(auto-list1 arg))))
(define-syntax superpose
(syntax-rules ()
((_ (argi ...))
(amb (superpose-helper argi) ...))
((_ args)
(amb1 (superpose-helper args)))))
(define-syntax amb-parallel
(syntax-rules ()
((_ arg1 arg2)
(receive (pipefd0 pipefd1) (create-pipe)
(let ((pid (process-fork)))
(if (eq? pid 0)
(let ((pipefd1_port (open-output-file* pipefd1)))
(write (amb-collect arg2) pipefd1_port) ;we can't backtrack into parent so we collect all of this level's results
(close-output-port pipefd1_port)
(exit 0)) ;child is done
(let* ((res1 (amb-collect arg1)) ;we cannot backtrack here either, the child counts on the parent, so we collect all of this level's results
(res2 (read (open-input-file* pipefd0)))) ;after we collected our results we can collect the result of the child
(if (and (null? res1) (null? res2)) ;but from here we can backtrack safely yet again
((amb-failure-continuation))
(if (null? res1)
(amb1 res2)
(if (null? res2)
(amb1 res1)
(amb1 (list (amb1 res1) (amb1 res2)))))))))))
((_ arg1 arg2 argi ...)
(amb1 (amb-parallel (amb-collect arg1) (amb-collect (amb-parallel arg2 argi ...)))))))
;hyperpose-helper enforces nested superpose compatibility
;by flatting out the nested hyperpose calls before passing it to amb-parallel
(define-syntax hyperpose-helper
(syntax-rules (hyperpose)
((_ (hyperpose (argi ...)))
(amb ((hyperpose-helper argi) ...)))
((_ arg)
(auto-list1 arg))))
(define-syntax hyperpose
(syntax-rules ()
((_ (argi ...))
(amb-parallel (hyperpose-helper argi) ...))
((_ args)
(amb1 (hyperpose-helper args)))))
;; FUNCTION DEFINITION IN METTA
;""""""""""""""""""""""""""""""
;A hash map where the functions are keyed by their name.
;The values are the lists containing all functions of same name.
(define functions (make-hash-table))
;The function is added to the &self space
;then it is added to the list of functions of same name in the hashmap,
;whereby function call is carried out by executing (with backtracking)
;all registered functions of that name.
(define-syntax =
(syntax-rules ()
((_ (name patterni ...) body)
(begin
(hash-table-set! vars '&self (cons '(=def (name patterni ...) body) (hash-table-ref vars '&self)))
(let ((function (match-lambda* ((patterni ...) (auto-list1 body)))))
(if (hash-table-exists? functions 'name)
(hash-table-set! functions 'name (cons function (hash-table-ref functions 'name)))
(hash-table-set! functions 'name (list function))))
(set! name (lambda args (handle-exceptions exn ((amb-failure-continuation))
(apply (amb1 (hash-table-ref functions 'name)) args))))))))
;; SYNTACTIC CONSTRUCTS FOR DEFINING VARIABLES
;"""""""""""""""""""""""""""""""""""""""""""""
;Using Scheme's 'match-let*', while omiting the need for double-parenthesis
(define-syntax Let
(syntax-rules ()
((_ var val body)
(match-let* ((var (auto-list1 val))) (auto-list1 body)))))
;Simply map to 'match-let*'
(define-syntax Let*
(syntax-rules ()
((_ ((vari vali) ...) body)
(match-let* ((vari (auto-list1 vali)) ...) (auto-list1 body)))
((_ (((vari1 vari2) vali) ...) body)
(match-let* (((vari1 vari2) (auto-list1 vali)) ...) (auto-list1 body)))))
;; SYNTACTIC CONSTRUCTS FOR CHECKING VARIABLES
;"""""""""""""""""""""""""""""""""""""""""""""
;The if-then case causes backtracking in the else branch instead of returning #undefined
(define-syntax If
(syntax-rules ()
((_ condition thenbody elsebody)
(if condition (auto-list1 thenbody) (auto-list1 elsebody)))
((_ condition thenbody)
(if condition (auto-list1 thenbody) ((amb-failure-continuation))))))
;Since case causes an exception when no case is met, we catch it and backtrack instead
;additionally we handle Empty case by considering the amount of matched options
;either returning the voidcase if there was no match, or nondeterministically the matched options
(define-syntax Case
(syntax-rules (Empty)
((_ var ((Empty voidcase)))
(if (eq? 0 (length (amb-collect (auto-list1 var)))) (auto-list1 voidcase) ((amb-failure-continuation))))
((_ var ((pati bodi) ... (Empty voidcase)))
(let ((options (amb-collect (handle-exceptions exn ((amb-failure-continuation))
(match (auto-list1 var) (pati (auto-list1 bodi)) ...)))))
(if (eq? 0 (length options))
(auto-list1 voidcase) (amb1 options))))
((_ var ((pati bodi) ...))
(handle-exceptions exn ((amb-failure-continuation))
(match (auto-list1 var) (pati (auto-list1 bodi)) ...)))))
;; QUERY EXECUTION OPERATOR
;;"""""""""""""""""""""""""
;Printing all solutions in the list
(define (print-helper xs)
(display "[")
(define (print-items xs)
(cond
((null? xs)
(display "]")
(newline))
(else
(display (car xs))
(if (not (null? (cdr xs)))
(display ", "))
(print-items (cdr xs)))))
(print-items xs))
;Print all solutions after collecting them
(define-syntax !
(syntax-rules ()
((_ argi ...)
(print-helper (amb-collect (auto-list argi ...))))))
;; AUTO-LIST TO ELIMINATE THE NEED FOR SYNTACTIC LIST/FUNCTIONCALL DISTINCTION
;"""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""""
;To avoid exponential expansion of a code tree with mostly unreachable options
;we use this specialied macro instead of 'if'. Hereby, list/function call resolving is skipped for
;syntactic constructs as they represent neither a function call nor a list.
(define-syntax metta-macro-if
(syntax-rules (collapse superpose hyperpose Let Let* Match Case If == sequential quote do trace! and or)
((_ collapse then else) then)
((_ superpose then else) then)
((_ hyperpose then else) then)
((_ Let then else) then)
((_ Let* then else) then)
((_ Match then else) then)
((_ Case then else) then)
((_ If then else) then)
((_ == then else) then)
((_ sequential then else) then)
((_ quote then else) then)
((_ do then else) then)
((_ trace! then else) then)
((_ and then else) then)
((_ or then else) then)
((_ arg then else) else)))
;Recursively, returns a list if arg is a list or executes a function if arg is a function
(define-syntax auto-list-helper
(syntax-rules ()
((_ expr1 ()) ;empty list
(list expr1))
((_ (expr1i ...) argi ...) ;a nested expression is not a procedure
(list (auto-list expr1i ...) (auto-list1 argi) ...))
((_ expr1 argi ...)
(if (procedure? expr1)
(apply expr1 (list (auto-list1 argi) ...))
(list (auto-list1 expr1) (auto-list1 argi) ...)))))
;Recursively, resolves whether it is a syntactic construct or whether to apply list/function call distinction
(define-syntax auto-list
(syntax-rules ()
((_ expr)
(if (procedure? expr)
(expr)
(list (auto-list1 expr))))
((_ expr1 expri ...)
(metta-macro-if expr1
(expr1 expri ...)
(auto-list-helper expr1 expri ...)))))
;same as before but on a single-arg nested expression
(define-syntax auto-list1
(syntax-rules ()
((_ (vari ...))
(auto-list vari ...))
((_ var1)
var1)))
;; EQUALITY
;""""""""""
;allow using '==' in the code
(define == equal?)
;; TYPE SYSTEM
;"""""""""""""
;Type Definitions
(define-type Atom *)
(define-type %Undefined% *)
(define-type Symbol symbol)
(define-type Expression list)
(define-type Bool boolean)
(define-type Number number)
(define-type String string)
;Types defined via Metta type specifier for fast get-types query in addition to inclusion in &self space
(define types (make-hash-table))
;Letting compiler know of types via syntactic mapping, for potential increased efficiency
(define-syntax Typedef
(syntax-rules ()
((_ arg (-> A ... B))
(begin (cond-expand (USE_TYPES (: arg (A ... -> B))) (else '()))
(hash-table-set! types 'arg '(-> A ... B))
(hash-table-set! vars '&self (cons '(:def arg (-> A ... B)) (hash-table-ref vars '&self)))))
((_ arg1 arg2)
(begin (if (not (hash-table-exists? types 'arg2))
(cond-expand (USE_TYPES (define-type arg2 *)) (else '())))
(cond-expand (USE_TYPES (define-type arg1 arg2)) (else '()))
(hash-table-set! types 'arg1 'arg2)
(hash-table-set! vars '&self (cons '(:def arg1 arg2) (hash-table-ref vars '&self)))))))
;Get the type by looking up its definition and also checking builtin types
(define (get-type x)
(if (hash-table-exists? types x)
(hash-table-ref types x)
(if (string? x)
'String
(if (number? x)
'Number
(if (boolean? x)
'Bool
(if (symbol? x)
'Symbol
(if (list? x)
'Expression
'%Undefined%)))))))
;; SPACES IMPLEMENTATION
;""""""""""""""""""""""""
;Where all state variables and spaces live, including &self
(define vars (make-hash-table))
(hash-table-set! vars '&self '())
;compatibility wrapper to construct states/spaces and retrieving state
(define (new-space s) s)
(define (new-state s) s)
(define (get-state s) s)
;get-atom returns non-deterministally all items of space s
(define (get-atoms space) (amb1 (hash-table-ref vars space)))
;bind a new space or state to a symbol var
(define (bind! var val)
(begin (hash-table-set! vars var val) '()))
;add an atom to an existing space within the vars structure
(define (add-atom space atom)
(let ((space-list (hash-table-ref vars space)))
(if (not (memv atom space-list))
(hash-table-set! vars space (cons atom space-list))
'())))
;remove an atom from an existing space within the vars structure
(define (remove-atom space atom)
(begin (hash-table-set! vars space (delete atom (hash-table-ref vars space))) '()))
;change an existing state variable within the vars structure
(define (change-state! var val)
(begin (hash-table-set! vars var val) (list 'State val)))
;retrieve the state of an existing state variable within the vars structure
(define (get-state var)
(hash-table-ref vars var))
;match expression: using Scheme's match-let*
(define-syntax Match
(syntax-rules (MatchChain)
((_ space (MatchChain bind1 bind2) result)
(Match space bind1 (Match space bind2 result)))
((_ space (MatchChain bind1 bindi ...) result)
(Match space bind1 (Match space (MatchChain bindi ...) result)))
((_ space binds result)
(handle-exceptions exn ((amb-failure-continuation))
(match-let* ((binds (amb1 (hash-table-ref vars space)))) (auto-list1 result))))))
;; PROCEDURAL CONSTRUCTS
;"""""""""""""""""""""""
;Append expression return value to list unless it is marked to be omitted via 'do'
(define-syntax sequential-helper
(syntax-rules (do)
((_ (do expr))
expr)
((_ expr)
(set! ret (append ret (list expr))))))
;procedural sequential execution can be superpose here as well
(define-syntax sequential
(syntax-rules ()
((_ arg)
(superpose arg))))
;standalone do also can cause side effects but returns no result
(define-syntax do
(syntax-rules ()
((_ arg)
(begin (auto-list1 arg) (If #f 42)))))
;; TRACE
;"""""""
;injected debug output for returned expression
(define-syntax trace!
(syntax-rules ()
((_ x y)
(begin (display (auto-list1 x)) (newline) (auto-list1 y)))))
;; MATH
;"""""""
(define (% a b) (modulo a b))
;; USEFUL ADDITIONS FOR FURTHER SPEEDUP
;""""""""""""""""""""""""""""""""""""""
;Cache hashtable to remember function return values
(define cache (make-hash-table))
;Memoize the results of function f
(define (memoized f)
(lambda (n)
(let ((key (cons f n)))
(if (hash-table-exists? cache key)
(hash-table-ref cache key)
(let ((result (f n)))
(hash-table-set! cache key result)
result)))))
;Deterministic function definition
(define-syntax =deterministic
(syntax-rules ()
((_ (fname args ...) body)
(set! fname (lambda (args ...) body)))))
;Memoized function definition
(define-syntax =memoized
(syntax-rules ()
((_ (fname args ...) body)
(set! fname (memoized (lambda (args ...) body))))))