@@ -30,13 +30,20 @@ import std/num/int32
30
30
import std/num/ddouble // for C# backend
31
31
import std/time/duration
32
32
import std/debug
33
+ pub import std/os/uv
34
+ // We utilize some timer functions from it's header file.
35
+ pub import std/time/timer
33
36
37
+ extern import
38
+ c file "async/async-inline.h"
39
+ js file "async/async-inline.js"
40
+ cs file "async/async-inline.cs"
34
41
35
42
// A type alias for asynchronous operations that can raise exceptions non-deterministically.
36
43
// This is common for almost all `:async` operations since `cancel` and `timeout` can
37
44
// cancel operations non-deterministically which raises the `Cancel` exception and cancels
38
45
// outstanding asynchronous requests.
39
- pub alias asyncx = <async,exn,ndet >
46
+ pub alias asyncx = <async,io >
40
47
41
48
42
49
// ----------------------------------------------------------------------------
@@ -51,7 +58,7 @@ abstract struct promise<a>
51
58
state : ref<global,promise-state<a>>
52
59
53
60
54
- abstract type promise-state<a>
61
+ abstract value type promise-state<a>
55
62
Resolved( value : a )
56
63
Awaiting( listeners : list<a -> io ()> )
57
64
@@ -61,7 +68,7 @@ pub fun promise() : async promise<a>
61
68
62
69
// Await a promise; returns immediately if the promise was already resolved and otherwise
63
70
// waits asynchronously.
64
- pub fun await( p : promise<a> ) : asyncx a
71
+ pub fun promise/ await( p : promise<a> ) : asyncx a
65
72
fun setup(cb : _ -> io-noexn ())
66
73
val r = p.state
67
74
match (!r)
@@ -96,7 +103,7 @@ pub fun resolve( p : promise<a>, value : a ) : asyncx ()
96
103
97
104
// A _channel_ of values of type `:a`. Values can be asynchronously `emit`ed into
98
105
// a channel, and asynchronously `receive`d.
99
- abstract struct channel<a>(
106
+ abstract value struct channel<a>(
100
107
chid : int,
101
108
state : ref<global,channel-state<a>>
102
109
)
@@ -191,7 +198,7 @@ fun trace-anyx( s : string, x : a ) : async ()
191
198
// abstraction can reliably time out over any composition of asynchronous operations
192
199
// and is therefore quite expressive.
193
200
194
- pub fun timeout( secs : duration, action : () -> <async,exn |e> a ) : <async,exn |e> maybe<a>
201
+ pub fun timeout( secs : duration, action : () -> <async,io |e> a ) : <async,io |e> maybe<a>
195
202
firstof { wait(secs); Nothing} { Just(action()) }
196
203
197
204
// Execute `a` and `b` interleaved. As soon as one of them finishes,
@@ -208,20 +215,20 @@ pub fun firstof( a : () -> <async,exn|e> a, b : () -> <async,exn|e> a ) : <asyn
208
215
209
216
// Wait (asynchronously) for `secs` seconds as a `:double`.
210
217
// Use `yield()` to yield to other asynchronous operations.
211
- pub fun wait( secs : double ) : <async,exn > ()
218
+ pub fun float/ wait( secs : float64 ) : <async,io > ()
212
219
wait(secs.duration)
213
220
214
221
// Wait (asynchronously) for optional `secs` seconds `:duration` (`= 0.seconds`).
215
222
// Use `yield()` to yield generally to other asynchronous operations.
216
- pub fun wait( secs : duration = zero ) : <async,exn > ()
223
+ pub fun wait( secs : duration = zero ) : <async,io > ()
217
224
if secs <= zero then return yield()
218
225
val msecs = max(zero:int32,secs.milli-seconds.int32)
219
226
await fn(cb)
220
227
val tid = set-timeout( fn(){ cb(Ok(())) }, msecs )
221
228
Just( { clear-timeout(tid) } )
222
229
223
230
// Yield to other asynchronous operations. Same as `wait(0)`.
224
- pub fun yield() : <async,exn > ()
231
+ pub fun yield() : <async,io > ()
225
232
await0 fn(cb)
226
233
set-timeout( cb, zero )
227
234
()
@@ -237,13 +244,15 @@ fun set-timeout( cb : () -> io-noexn (), ms : int32 ) : io-noexn timeout-id
237
244
extern set-timeoutx( cb : () -> io-noexn (), ms : int32 ) : io-noexn any
238
245
cs "_Async.SetTimeout"
239
246
js "setTimeout"
247
+ c "kk_set_timeout"
240
248
241
249
fun clear-timeout( tid : timeout-id ) : io-noexn ()
242
250
clear-timeoutx(tid.timer)
243
251
244
252
extern clear-timeoutx( tid : any) : io-noexn ()
245
253
cs "_Async.ClearTimeout"
246
254
js "clearTimeout"
255
+ c "kk_clear_timeout"
247
256
248
257
249
258
// ----------------------------------------------------------------------------
@@ -257,7 +266,7 @@ pub fun interleaved( action1 : () -> <async,exn|e> a, action2 : () -> <async,exn
257
266
(ra.untry,rb.untry)
258
267
259
268
// Interleave a list of actions around their asynchronous operations.
260
- pub fun interleaved( xs : list<() -> <async,exn|e> a> ) : <async,exn|e> list<a>
269
+ pub fun list/ interleaved( xs : list<() -> <async,exn|e> a> ) : <async,exn|e> list<a>
261
270
val ress = xs.map( fn(f) { return { mask behind<exn>(f) } } ).interleavedx
262
271
//ress.map(maybe).ordered_throw
263
272
ress.map(untry)
@@ -331,7 +340,7 @@ fun insert( xs : list<(int,a)>, idx : int, value : a, n : int = 0 ) : list<(int
331
340
332
341
// Interleave a list actions around their asynchronous operations and explicitly returning either
333
342
// either their result or their exception.
334
- pub fun interleavedx( xs : list<() -> <async,exn|e> a> ) : <async|e> list<error<a>>
343
+ pub fun list/ interleavedx( xs : list<() -> <async,exn|e> a> ) : <async|e> list<error<a>>
335
344
val n = xs.length
336
345
if n==0 then []
337
346
elif n==1 then xs.map(unsafe-try-all)
@@ -359,12 +368,12 @@ fun unsafe-no-ndet-div( action : () -> <ndet,div|e> a ) : e a
359
368
inline extern inject-effects : forall<a,h,e> (() -> e a) -> total (() -> <strands<a>,ndet,div|e> a)
360
369
inline "#1"
361
370
362
- fun is-finalize( t : error<a> ) : bool
371
+ fun error/ is-finalize( t : error<a> ) : bool
363
372
match t
364
373
Error(exn) -> exn.is-finalize
365
374
_ -> False
366
375
367
- fun is-cancel( t : error<a> ) : bool
376
+ fun error/ is-cancel( t : error<a> ) : bool
368
377
match t
369
378
Error(exn) -> exn.is-cancel
370
379
_ -> False
@@ -408,13 +417,13 @@ fun interleaved-div( xs : list<() -> <async,exn|e> a> ) : <async,ndet,div,strand
408
417
// ----------------------------------------------------------------------------
409
418
410
419
// Convenience function for awaiting a NodeJS style callback where the first argument is a possible exception.
411
- pub fun await-exn0( setup : (cb : (null<exception>) -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,exn > ()
420
+ pub fun await-exn0( setup : (cb : (null<exception>) -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,io > ()
412
421
await fn(cb)
413
422
setup( fn(nexn) cb(nexn.unnull(())) )
414
423
415
424
// Convenience function for awaiting a NodeJS style callback where the first argument is a possible exception
416
425
// and the second argument the possible result value.
417
- pub fun await-exn1( setup : (cb : (null<exception>,a) -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,exn > a
426
+ pub fun await-exn1( setup : (cb : (null<exception>,a) -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,io > a
418
427
await fn(cb)
419
428
setup( fn(nexn,x) cb(nexn.unnull(x)) )
420
429
@@ -424,13 +433,13 @@ fun unnull( nexn : null<exception>, x : a ) : error<a>
424
433
Just(exn) -> Error(exn)
425
434
426
435
// Convenience function for awaiting a zero argument callback.
427
- pub fun await0( setup : (cb : () -> io-noexn () ) -> io () ) : <async,exn > ()
436
+ pub fun await0( setup : (cb : () -> io-noexn () ) -> io () ) : <async,io > ()
428
437
await fn(cb)
429
438
setup( fn() cb(Ok(())) )
430
439
Nothing
431
440
432
441
// Convenience function for awaiting a single argument callback.
433
- pub fun await1( setup : (cb : (a) -> io-noexn () ) -> io () ) : <async,exn > a
442
+ pub fun await1( setup : (cb : (a) -> io-noexn () ) -> io () ) : <async,io > a
434
443
await fn(cb)
435
444
setup( fn(x) cb(Ok(x)) )
436
445
Nothing
@@ -440,7 +449,7 @@ pub fun await1( setup : (cb : (a) -> io-noexn () ) -> io () ) : <async,exn> a
440
449
// value where the `cleanup` functions is invoked on cancellation to dispose of any resources (see the implementation of `wait`).
441
450
// The callback should be invoked exactly once -- when that happens `await` is resumed with the result using `untry`
442
451
// either raise an exception or return the plain result.
443
- pub fun await( setup : (cb : error<a> -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,exn > a
452
+ pub fun setup/ await( setup : (cb : error<a> -> io-noexn () ) -> io maybe<() -> io-noexn ()> ) : <async,io > a
444
453
await-exn(setup).untry
445
454
446
455
@@ -463,7 +472,7 @@ pub effect async
463
472
// The `cancel` operations cancels any outstanding asynchronous operation under the innermost
464
473
// `cancelable` handler by returning the `Cancel` exception. The `cancel` operation itself returns normally
465
474
// without raising a `Cancel` exception.
466
- pub fun cancel() : async ()
475
+ pub fun noscope/ cancel() : async ()
467
476
cancel(empty-scope)
468
477
469
478
// Primitive: Execute `setup` to set up an asynchronous callback with the host platform. Invoke `cb` as the callback:
@@ -525,25 +534,26 @@ pub fun cancelable( action : () -> <async|e> a ) : <async|e> a
525
534
// this might be needed for `no-await` operations.
526
535
cancel(empty-scope.extend)
527
536
x
528
- ctl do-await(setup,scope,c) -> resume( do-await(setup,scope.extend,c) )
529
- ctl no-await(setup,scope,c,f) -> resume( no-await(setup,scope.extend,c,f) )
530
- ctl cancel(scope) -> resume( cancel(scope.extend) )
531
- ctl async-iox(f) -> resume( async-iox(f) )
537
+ fun do-await(setup,scope,c) -> do-await(setup,scope.extend,c)
538
+ fun no-await(setup,scope,c,f) -> no-await(setup,scope.extend,c,f)
539
+ fun cancel(scope) -> cancel(scope.extend)
540
+ fun async-iox(f) -> async-iox(f)
532
541
533
542
// ----------------------------------------------------------------------------
534
543
// Async handle
535
544
// ----------------------------------------------------------------------------
536
545
537
- pub fun ".default-async"(action)
538
- async-handle(action);
546
+ pub fun "@default-async"(action)
547
+ with handle-uv
548
+ async/handle(action);
539
549
540
550
fun nodispose() : io-noexn ()
541
551
()
542
552
543
553
// The outer `:async` effect handler. This is automatically applied by the compiler
544
554
// around the `main` function if it has an `:async` effect.
545
- pub fun async- handle(action : () -> <async,io-noexn> () ) : io-noexn ()
546
- val callbacks : ref<global,list<(scope,() -> io-noexn ())>> = ref([])
555
+ pub fun async/ handle(action : () -> <async,io-noexn> () ) : io-noexn ()
556
+ val callbacks : ref<global,list<(scope,() -> io-noexn ())>> = unsafe-total{ ref([])}
547
557
fun handle-await( setup : await-setup<a>, scope : scope, f : error<a> -> io-noexn (), cancelable : bool) : io-noexn ()
548
558
val cscope = child-scope(unique(),scope)
549
559
val dispose = ref(nodispose)
@@ -573,12 +583,12 @@ pub fun async-handle(action : () -> <async,io-noexn> () ) : io-noexn ()
573
583
handle(action)
574
584
raw ctl do-await( setup, scope, c )
575
585
handle-await(setup,scope, fn(x) rcontext.resume(x), c) // returns to outer event loop
576
- ctl no-await( setup, scope, c, f )
577
- resume( handle-await(setup,scope,f,c))
578
- ctl cancel( scope )
579
- resume( handle-cancel(scope))
580
- ctl async-iox( f )
581
- resume(f() )
586
+ fun no-await( setup, scope, c, f )
587
+ handle-await(setup,scope,f,c)
588
+ fun cancel( scope )
589
+ handle-cancel(scope)
590
+ fun async-iox( f )
591
+ f( )
582
592
583
593
fun io-noexn( f : () -> io-noexn a ) : io a
584
594
f()
@@ -602,7 +612,7 @@ fun child-scope( id : int, scope : scope ) : scope
602
612
match scope
603
613
Scope(cids) -> Scope(cids ++ [id])
604
614
605
- fun in-scope-of( child : list<int>, parent : list<int> ) : bool
615
+ fun ids/ in-scope-of( child : list<int>, parent : list<int> ) : bool
606
616
match parent
607
617
Nil -> True
608
618
Cons(p,ps) -> match child
@@ -614,22 +624,11 @@ fun in-scope-of( child : scope, parent : scope ) : bool
614
624
Scope(pids) -> match child
615
625
Scope(cids) -> in-scope-of(cids,pids)
616
626
617
- fun (==)( ids1 : list<int>, ids2 : list<int> ) : bool
618
- match ids1
619
- Nil -> ids2.is-nil
620
- Cons(i1,is1) -> match ids2
621
- Cons(i2,is2) -> (i1 == i2 && is1 == is2)
622
- Nil -> False
623
-
624
- fun (==)(scope1 : scope, scope2 : scope ) : bool
627
+ fun scope/(==)(scope1 : scope, scope2 : scope ) : bool
625
628
match scope1
626
629
Scope(ids1) -> match scope2
627
630
Scope(ids2) -> ids1==ids2
628
631
629
- fun (!=)(scope1 : scope, scope2 : scope ) : bool
630
- !(scope1 == scope2)
631
-
632
-
633
632
// Convenience functions for scope maps
634
633
fun remove( xs : list<(scope,a)>, scope : scope ) : list<(scope,a)>
635
634
xs.remove( fn(x:(scope,_)) { x.fst == scope })
@@ -640,7 +639,6 @@ fun lookup( xs : list<(scope,a)>, scope : scope ) : maybe<a>
640
639
fun contains( xs : list<(scope,a)>, scope : scope ) : bool
641
640
xs.lookup(scope).bool
642
641
643
-
644
642
fun show( s : scope ) : string
645
643
match s
646
644
Scope(ids) -> ids.map(show).join("-")
@@ -652,13 +650,13 @@ abstract extend type exception-info
652
650
con Finalize(yld:yield-info)
653
651
654
652
// Was this a cancelation exception?
655
- fun is-cancel( exn : exception ) : bool
653
+ fun exn/ is-cancel( exn : exception ) : bool
656
654
match exn.info
657
655
Cancel -> True
658
656
_ -> False
659
657
660
658
// Was this a finalization exception?
661
- fun is-finalize(exn : exception) : bool
659
+ fun exn/ is-finalize(exn : exception) : bool
662
660
match exn.info
663
661
Finalize -> True
664
662
_ -> False
0 commit comments