Skip to content

Commit c2905c2

Browse files
committed
libuv
1 parent 4481bcc commit c2905c2

27 files changed

+3358
-67
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -33,3 +33,4 @@ bench
3333
test/bench
3434
.koka
3535
scratch
36+
.cache

kklib/include/kklib.h

+4
Original file line numberDiff line numberDiff line change
@@ -422,6 +422,7 @@ typedef struct kk_context_s {
422422
kk_yield_t yield; // inlined yield structure (for efficiency)
423423
int32_t marker_unique; // unique marker generation
424424
kk_block_t* delayed_free; // list of blocks that still need to be freed
425+
void* loop; // a reference to an event loop (e.g. uv_loop_t* or NULL)
425426
kk_integer_t unique; // thread local unique number generation
426427
size_t thread_id; // unique thread id
427428
kk_box_any_t kk_box_any; // used when yielding as a value of any type
@@ -444,6 +445,9 @@ typedef struct kk_context_s {
444445
kk_decl_export kk_context_t* kk_get_context(void);
445446
kk_decl_export void kk_free_context(void);
446447

448+
449+
kk_decl_export void kk_debugger_break(kk_context_t* ctx);
450+
447451
// The current context is passed as a _ctx parameter in the generated code
448452
#define kk_context() _ctx
449453

kklib/src/string.c

+3
Original file line numberDiff line numberDiff line change
@@ -865,20 +865,23 @@ kk_string_t kk_string_trim_right(kk_string_t str, kk_context_t* ctx) {
865865
kk_unit_t kk_println(kk_string_t s, kk_context_t* ctx) {
866866
// TODO: set locale to utf-8?
867867
puts(kk_string_cbuf_borrow(s, NULL, ctx)); // todo: allow printing embedded 0 characters?
868+
fflush(stdout);
868869
kk_string_drop(s, ctx);
869870
return kk_Unit;
870871
}
871872

872873
kk_unit_t kk_print(kk_string_t s, kk_context_t* ctx) {
873874
// TODO: set locale to utf-8?
874875
fputs(kk_string_cbuf_borrow(s, NULL, ctx), stdout); // todo: allow printing embedded 0 characters?
876+
fflush(stdout);
875877
kk_string_drop(s, ctx);
876878
return kk_Unit;
877879
}
878880

879881
kk_unit_t kk_trace(kk_string_t s, kk_context_t* ctx) {
880882
fputs(kk_string_cbuf_borrow(s, NULL, ctx), stderr); // todo: allow printing embedded 0 characters?
881883
fputs("\n", stderr);
884+
fflush(stdout);
882885
kk_string_drop(s, ctx);
883886
return kk_Unit;
884887
}

lib/std/async.kk

+46-48
Original file line numberDiff line numberDiff line change
@@ -30,13 +30,20 @@ import std/num/int32
3030
import std/num/ddouble // for C# backend
3131
import std/time/duration
3232
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
3336

37+
extern import
38+
c file "async/async-inline.h"
39+
js file "async/async-inline.js"
40+
cs file "async/async-inline.cs"
3441

3542
// A type alias for asynchronous operations that can raise exceptions non-deterministically.
3643
// This is common for almost all `:async` operations since `cancel` and `timeout` can
3744
// cancel operations non-deterministically which raises the `Cancel` exception and cancels
3845
// outstanding asynchronous requests.
39-
pub alias asyncx = <async,exn,ndet>
46+
pub alias asyncx = <async,io>
4047

4148

4249
// ----------------------------------------------------------------------------
@@ -51,7 +58,7 @@ abstract struct promise<a>
5158
state : ref<global,promise-state<a>>
5259

5360

54-
abstract type promise-state<a>
61+
abstract value type promise-state<a>
5562
Resolved( value : a )
5663
Awaiting( listeners : list<a -> io ()> )
5764

@@ -61,7 +68,7 @@ pub fun promise() : async promise<a>
6168

6269
// Await a promise; returns immediately if the promise was already resolved and otherwise
6370
// waits asynchronously.
64-
pub fun await( p : promise<a> ) : asyncx a
71+
pub fun promise/await( p : promise<a> ) : asyncx a
6572
fun setup(cb : _ -> io-noexn ())
6673
val r = p.state
6774
match (!r)
@@ -96,7 +103,7 @@ pub fun resolve( p : promise<a>, value : a ) : asyncx ()
96103

97104
// A _channel_ of values of type `:a`. Values can be asynchronously `emit`ed into
98105
// a channel, and asynchronously `receive`d.
99-
abstract struct channel<a>(
106+
abstract value struct channel<a>(
100107
chid : int,
101108
state : ref<global,channel-state<a>>
102109
)
@@ -191,7 +198,7 @@ fun trace-anyx( s : string, x : a ) : async ()
191198
// abstraction can reliably time out over any composition of asynchronous operations
192199
// and is therefore quite expressive.
193200

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>
195202
firstof { wait(secs); Nothing} { Just(action()) }
196203

197204
// 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
208215

209216
// Wait (asynchronously) for `secs` seconds as a `:double`.
210217
// 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> ()
212219
wait(secs.duration)
213220

214221
// Wait (asynchronously) for optional `secs` seconds `:duration` (`= 0.seconds`).
215222
// 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> ()
217224
if secs <= zero then return yield()
218225
val msecs = max(zero:int32,secs.milli-seconds.int32)
219226
await fn(cb)
220227
val tid = set-timeout( fn(){ cb(Ok(())) }, msecs )
221228
Just( { clear-timeout(tid) } )
222229

223230
// Yield to other asynchronous operations. Same as `wait(0)`.
224-
pub fun yield() : <async,exn> ()
231+
pub fun yield() : <async,io> ()
225232
await0 fn(cb)
226233
set-timeout( cb, zero )
227234
()
@@ -237,13 +244,15 @@ fun set-timeout( cb : () -> io-noexn (), ms : int32 ) : io-noexn timeout-id
237244
extern set-timeoutx( cb : () -> io-noexn (), ms : int32 ) : io-noexn any
238245
cs "_Async.SetTimeout"
239246
js "setTimeout"
247+
c "kk_set_timeout"
240248

241249
fun clear-timeout( tid : timeout-id ) : io-noexn ()
242250
clear-timeoutx(tid.timer)
243251

244252
extern clear-timeoutx( tid : any) : io-noexn ()
245253
cs "_Async.ClearTimeout"
246254
js "clearTimeout"
255+
c "kk_clear_timeout"
247256

248257

249258
// ----------------------------------------------------------------------------
@@ -257,7 +266,7 @@ pub fun interleaved( action1 : () -> <async,exn|e> a, action2 : () -> <async,exn
257266
(ra.untry,rb.untry)
258267

259268
// 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>
261270
val ress = xs.map( fn(f) { return { mask behind<exn>(f) } } ).interleavedx
262271
//ress.map(maybe).ordered_throw
263272
ress.map(untry)
@@ -331,7 +340,7 @@ fun insert( xs : list<(int,a)>, idx : int, value : a, n : int = 0 ) : list<(int
331340

332341
// Interleave a list actions around their asynchronous operations and explicitly returning either
333342
// 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>>
335344
val n = xs.length
336345
if n==0 then []
337346
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
359368
inline extern inject-effects : forall<a,h,e> (() -> e a) -> total (() -> <strands<a>,ndet,div|e> a)
360369
inline "#1"
361370

362-
fun is-finalize( t : error<a> ) : bool
371+
fun error/is-finalize( t : error<a> ) : bool
363372
match t
364373
Error(exn) -> exn.is-finalize
365374
_ -> False
366375

367-
fun is-cancel( t : error<a> ) : bool
376+
fun error/is-cancel( t : error<a> ) : bool
368377
match t
369378
Error(exn) -> exn.is-cancel
370379
_ -> False
@@ -408,13 +417,13 @@ fun interleaved-div( xs : list<() -> <async,exn|e> a> ) : <async,ndet,div,strand
408417
// ----------------------------------------------------------------------------
409418

410419
// 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> ()
412421
await fn(cb)
413422
setup( fn(nexn) cb(nexn.unnull(())) )
414423

415424
// Convenience function for awaiting a NodeJS style callback where the first argument is a possible exception
416425
// 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
418427
await fn(cb)
419428
setup( fn(nexn,x) cb(nexn.unnull(x)) )
420429

@@ -424,13 +433,13 @@ fun unnull( nexn : null<exception>, x : a ) : error<a>
424433
Just(exn) -> Error(exn)
425434

426435
// 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> ()
428437
await fn(cb)
429438
setup( fn() cb(Ok(())) )
430439
Nothing
431440

432441
// 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
434443
await fn(cb)
435444
setup( fn(x) cb(Ok(x)) )
436445
Nothing
@@ -440,7 +449,7 @@ pub fun await1( setup : (cb : (a) -> io-noexn () ) -> io () ) : <async,exn> a
440449
// value where the `cleanup` functions is invoked on cancellation to dispose of any resources (see the implementation of `wait`).
441450
// The callback should be invoked exactly once -- when that happens `await` is resumed with the result using `untry`
442451
// 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
444453
await-exn(setup).untry
445454

446455

@@ -463,7 +472,7 @@ pub effect async
463472
// The `cancel` operations cancels any outstanding asynchronous operation under the innermost
464473
// `cancelable` handler by returning the `Cancel` exception. The `cancel` operation itself returns normally
465474
// without raising a `Cancel` exception.
466-
pub fun cancel() : async ()
475+
pub fun noscope/cancel() : async ()
467476
cancel(empty-scope)
468477

469478
// 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
525534
// this might be needed for `no-await` operations.
526535
cancel(empty-scope.extend)
527536
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)
532541

533542
// ----------------------------------------------------------------------------
534543
// Async handle
535544
// ----------------------------------------------------------------------------
536545

537-
pub fun ".default-async"(action)
538-
async-handle(action);
546+
pub fun "@default-async"(action)
547+
with handle-uv
548+
async/handle(action);
539549

540550
fun nodispose() : io-noexn ()
541551
()
542552

543553
// The outer `:async` effect handler. This is automatically applied by the compiler
544554
// 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([])}
547557
fun handle-await( setup : await-setup<a>, scope : scope, f : error<a> -> io-noexn (), cancelable : bool) : io-noexn ()
548558
val cscope = child-scope(unique(),scope)
549559
val dispose = ref(nodispose)
@@ -573,12 +583,12 @@ pub fun async-handle(action : () -> <async,io-noexn> () ) : io-noexn ()
573583
handle(action)
574584
raw ctl do-await( setup, scope, c )
575585
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()
582592

583593
fun io-noexn( f : () -> io-noexn a ) : io a
584594
f()
@@ -602,7 +612,7 @@ fun child-scope( id : int, scope : scope ) : scope
602612
match scope
603613
Scope(cids) -> Scope(cids ++ [id])
604614

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
606616
match parent
607617
Nil -> True
608618
Cons(p,ps) -> match child
@@ -614,22 +624,11 @@ fun in-scope-of( child : scope, parent : scope ) : bool
614624
Scope(pids) -> match child
615625
Scope(cids) -> in-scope-of(cids,pids)
616626

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
625628
match scope1
626629
Scope(ids1) -> match scope2
627630
Scope(ids2) -> ids1==ids2
628631

629-
fun (!=)(scope1 : scope, scope2 : scope ) : bool
630-
!(scope1 == scope2)
631-
632-
633632
// Convenience functions for scope maps
634633
fun remove( xs : list<(scope,a)>, scope : scope ) : list<(scope,a)>
635634
xs.remove( fn(x:(scope,_)) { x.fst == scope })
@@ -640,7 +639,6 @@ fun lookup( xs : list<(scope,a)>, scope : scope ) : maybe<a>
640639
fun contains( xs : list<(scope,a)>, scope : scope ) : bool
641640
xs.lookup(scope).bool
642641

643-
644642
fun show( s : scope ) : string
645643
match s
646644
Scope(ids) -> ids.map(show).join("-")
@@ -652,13 +650,13 @@ abstract extend type exception-info
652650
con Finalize(yld:yield-info)
653651

654652
// Was this a cancelation exception?
655-
fun is-cancel( exn : exception ) : bool
653+
fun exn/is-cancel( exn : exception ) : bool
656654
match exn.info
657655
Cancel -> True
658656
_ -> False
659657

660658
// Was this a finalization exception?
661-
fun is-finalize(exn : exception) : bool
659+
fun exn/is-finalize(exn : exception) : bool
662660
match exn.info
663661
Finalize -> True
664662
_ -> False

lib/std/async/async-inline.h

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
#include "std_time_timer.h"
2+
3+
4+
kk_box_t kk_set_timeout(kk_function_t cb, int64_t time, kk_context_t* _ctx) {
5+
kk_std_time_timer__timer t = kk_std_time_timer_timer_init(_ctx);
6+
kk_std_time_timer_timer_start(t, time, 0, cb, _ctx);
7+
return kk_std_time_timer__timer_box(t, _ctx);
8+
}
9+
10+
11+
static kk_box_t kk_unit_closure(kk_function_t _fself, kk_context_t* _ctx);
12+
static kk_function_t kk_new_unit_closure(kk_context_t* _ctx) {
13+
kk_define_static_function(_fself, kk_unit_closure, _ctx)
14+
return kk_function_dup(_fself,kk_context());
15+
}
16+
17+
static kk_box_t kk_unit_closure(kk_function_t _fself, kk_context_t* _ctx) {
18+
kk_unused(_fself);
19+
return kk_unit_box(kk_Unit);
20+
}
21+
22+
23+
kk_unit_t kk_clear_timeout(kk_box_t t, kk_context_t* _ctx) {
24+
kk_std_time_timer__timer timer = kk_std_time_timer__timer_unbox(t, KK_OWNED, _ctx);
25+
kk_std_os_uv_close(kk_std_os_uv__new_UvHandle(timer.internal, _ctx), kk_new_unit_closure(_ctx), _ctx);
26+
return kk_Unit;
27+
}
28+
29+
30+

0 commit comments

Comments
 (0)