aboutsummaryrefslogtreecommitdiff
path: root/new-luxc/test
diff options
context:
space:
mode:
Diffstat (limited to 'new-luxc/test')
-rw-r--r--new-luxc/test/test/luxc/synthesizer/function.lux127
1 files changed, 125 insertions, 2 deletions
diff --git a/new-luxc/test/test/luxc/synthesizer/function.lux b/new-luxc/test/test/luxc/synthesizer/function.lux
index 9243294a2..7c4776727 100644
--- a/new-luxc/test/test/luxc/synthesizer/function.lux
+++ b/new-luxc/test/test/luxc/synthesizer/function.lux
@@ -4,15 +4,138 @@
(control monad
pipe)
(data [product]
- (coll [list]))
+ [number]
+ text/format
+ (coll [list "L/" Functor<List> Fold<List>]
+ ["D" dict]
+ ["s" set]))
["r" math/random "r/" Monad<Random>]
test)
(luxc (lang ["la" analysis]
["ls" synthesis])
(analyser [";A" structure])
- [synthesizer])
+ [synthesizer]
+ (synthesizer ["&&;" function]))
(.. common))
+(def: (reference var)
+ (-> ls;Variable Ref)
+ (if (&&function;captured-var? var)
+ (#;Captured (|> var (i.* -1) int-to-nat n.dec))
+ (#;Local (int-to-nat var))))
+
+(def: (make-scope env)
+ (-> (List ls;Variable) Scope)
+ {#;name (list)
+ #;inner +0
+ #;locals {#;counter +0 #;mappings (list)}
+ #;captured {#;counter +0
+ #;mappings (L/map (|>. reference [Void] [""])
+ env)}})
+
+(def: gen-function//constant
+ (r;Random [Nat la;Analysis la;Analysis])
+ (r;rec
+ (function [gen-function//constant]
+ (do r;Monad<Random>
+ [function? r;bool]
+ (if function?
+ (do @
+ [[num-args outputA subA] gen-function//constant]
+ (wrap [(n.inc num-args)
+ outputA
+ (#la;Function (make-scope (list)) subA)]))
+ (do @
+ [outputA gen-primitive]
+ (wrap [+0 outputA outputA])))))))
+
+(def: (pick scope-size)
+ (-> Nat (r;Random Nat))
+ (|> r;nat (:: r;Monad<Random> map (n.% scope-size))))
+
+(def: gen-function//captured
+ (r;Random [Nat Int la;Analysis])
+ (do r;Monad<Random>
+ [num-locals (|> r;nat (:: @ map (|>. (n.% +100) (n.max +10))))
+ #let [indices (list;n.range +0 (n.dec num-locals))
+ absolute-env (L/map &&function;to-local indices)
+ relative-env (L/map &&function;to-captured indices)]
+ [total-args prediction bodyA] (: (r;Random [Nat Int la;Analysis])
+ (loop [num-args +1
+ global-env relative-env]
+ (let [env-size (list;size global-env)
+ resolver (L/fold (function [[idx var] resolver]
+ (D;put idx var resolver))
+ (: (D;Dict Nat Int)
+ (D;new number;Hash<Nat>))
+ (list;zip2 (list;n.range +0 (n.dec env-size))
+ global-env))]
+ (do @
+ [nest? r;bool]
+ (if nest?
+ (do @
+ [num-picks (:: @ map (n.max +1) (pick (n.inc env-size)))
+ picks (|> (r;set number;Hash<Nat> num-picks (pick env-size))
+ (:: @ map s;to-list))
+ [total-args prediction bodyA] (recur (n.inc num-args)
+ (L/map (function [pick] (assume (list;nth pick global-env)))
+ picks))]
+ (wrap [total-args prediction (#la;Function (make-scope (L/map &&function;to-captured picks))
+ bodyA)]))
+ (do @
+ [chosen (pick (list;size global-env))]
+ (wrap [num-args
+ (assume (D;get chosen resolver))
+ (#la;Relative (#;Captured chosen))])))))))]
+ (wrap [total-args prediction (#la;Function (make-scope absolute-env) bodyA)])
+ ))
+
+(def: gen-function//local
+ (r;Random [Nat Int la;Analysis])
+ (loop [num-args +0
+ nest? true]
+ (if nest?
+ (do r;Monad<Random>
+ [nest?' r;bool
+ [total-args prediction bodyA] (recur (n.inc num-args) nest?')]
+ (wrap [total-args prediction (#la;Function (make-scope (list)) bodyA)]))
+ (do r;Monad<Random>
+ [chosen (|> r;nat (:: @ map (|>. (n.% +100) (n.max +2))))]
+ (wrap [num-args
+ (|> chosen (n.+ (n.dec num-args)) nat-to-int)
+ (#la;Relative (#;Local chosen))])))))
+
+(test: "Function definition."
+ [[args1 prediction1 function1] gen-function//constant
+ [args2 prediction2 function2] gen-function//captured
+ [args3 prediction3 function3] gen-function//local]
+ ($_ seq
+ (assert "Nested functions will get folded together."
+ (|> (synthesizer;synthesize function1)
+ (case> (#ls;Function args captured output)
+ (and (n.= args1 args)
+ (corresponds? prediction1 output))
+
+ _
+ (n.= +0 args1))))
+ (assert "Folded functions provide direct access to captured variables."
+ (|> (synthesizer;synthesize function2)
+ (case> (#ls;Function args captured (#ls;Variable output))
+ (and (n.= args2 args)
+ (i.= prediction2 output))
+
+ _
+ false)))
+ (assert "Folded functions properly offset local variables."
+ (|> (synthesizer;synthesize function3)
+ (case> (#ls;Function args captured (#ls;Variable output))
+ (and (n.= args3 args)
+ (i.= prediction3 output))
+
+ _
+ false)))
+ ))
+
(test: "Function application."
[num-args (|> r;nat (:: @ map (|>. (n.% +10) (n.max +1))))
funcA gen-primitive