-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsimple_lazy.ML
66 lines (50 loc) · 1.42 KB
/
simple_lazy.ML
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
(*
* Copyright 2014, NICTA
*
* This software may be distributed and modified according to the terms of
* the BSD 2-Clause license. Note that NO WARRANTY is provided.
* See "LICENSE_BSD2.txt" for details.
*
* @TAG(NICTA_BSD)
*)
(*
* Title: Pure/Concurrent/lazy_sequential.ML
* Author: Florian Haftmann and Makarius, TU Muenchen
* David Greenaway, NICTA
*
* Thread-safe lazy evaluation with memoing, without using futures.
*)
structure SimpleLazy: LAZY =
struct
(* datatype *)
datatype 'a expr =
Expr of unit -> 'a |
Result of 'a Exn.result;
abstype 'a lazy = Lazy of 'a expr Synchronized.var
with
fun peek (Lazy r) =
(case (Synchronized.value r) of
Expr _ => NONE
| Result res => SOME res);
fun lazy e = Lazy (Synchronized.var "simple_lazy" (Expr e));
fun value a = Lazy (Synchronized.var "simple_lazy" (Result (Exn.Res a)));
(* force result *)
fun force_result (Lazy r) =
Synchronized.change_result r (fn x =>
(case x of
Expr e =>
let
val x = Exn.capture e ()
in
(x, Result x)
end
| Result res => (res, Result res)))
fun force r = Exn.release (force_result r);
fun is_finished x = is_some (peek x)
fun map f x = lazy (fn () => f (force x));
fun future params x =
if is_finished x then Future.value_result (force_result x)
else (singleton o Future.forks) params (fn () => force x);
end;
end;
type 'a simple_lazy = 'a SimpleLazy.lazy;