/*---------------------------------------------------------------------------
  Copyright 2012-2024, Microsoft Research, Daan Leijen.

  This is free software; you can redistribute it and/or modify it under the
  terms of the Apache License, Version 2.0. A copy of the License can be
  found in the LICENSE file at the root of this distribution.
---------------------------------------------------------------------------*/

/* Core functions.

   This module is implicitly imported and all functions and types
   are always available.
   Some types and operations are required to be defined for the compiler
   to work correctly (i.e. types like `:exn` or `:list`)
*/
module std/corestd/core

pub import std/core/typesstd/core/types
import std/core/undivstd/core/undiv
import std/core/unsafestd/core/unsafe
pub import std/core/hndstd/core/hnd
pub import std/core/exnstd/core/exn
pub import std/core/boolstd/core/bool
pub import std/core/orderstd/core/order
pub import std/core/charstd/core/char
pub import std/core/intstd/core/int
pub import std/core/vectorstd/core/vector
pub import std/core/stringstd/core/string
pub import std/core/sslicestd/core/sslice
pub import std/core/liststd/core/list
pub import std/core/maybestd/core/maybe
pub import std/core/eitherstd/core/either
pub import std/core/tuplestd/core/tuple
pub import std/core/showstd/core/show
pub import std/core/debugstd/core/debug
pub import std/core/delayedstd/core/delayed
pub import std/core/consolestd/core/console

extern import
  c  file "core/inline/core"
  cs file "core/inline/core.cs"

// ----------------------------------------------------------------------------
// Builtin effects
// ----------------------------------------------------------------------------

// An alias for the empty effect.
// pub alias total = <>

// An alias for pure effects: a pure function always returns the same result
// when called with the same arguments but may not terminate or raise an exception.
pub alias purestd/core/pure: E = <std/core/types/total: Eexnstd/core/exn/exn: (E, V) -> V,divstd/core/types/div: X>

// The `:global-scope` is a special type constant to denote the global scope
pub type global-scopestd/core/global-scope: S :: S

// The `:net` effect signifies a function may access the network
pub type netstd/core/net: X :: X

// The `:fsys` effect signifies a function may access the file system
pub type fsysstd/core/fsys: X :: X

// The `:ui` effect signifies a function may access the graphics system
pub type uistd/core/ui: X :: X

// The `:blocking` effect signifies that a function may block
pub type blockingstd/core/blocking: X :: X

// The `:io-total` effect is used for functions that perform arbitrary I/O operations, but are terminating without raising exceptions.
pub alias io-totalstd/core/io-total: E = <std/core/types/total: Endetstd/core/types/ndet: X,consolestd/core/console/console: X,netstd/core/net: X,fsysstd/core/fsys: X,uistd/core/ui: X,ststd/core/types/st: H -> E<globalstd/core/types/global: H>>

// The `:io-noexn` effect is used for functions that perform arbitrary I/O operations, but raise no exceptions
pub alias io-noexnstd/core/io-noexn: E = <std/core/types/total: Edivstd/core/types/div: X,io-totalstd/core/io-total: E>

// The `:io` effect is used for functions that perform arbitrary I/O operations.
pub alias iostd/core/io: E = <std/core/types/total: Eexnstd/core/exn/exn: (E, V) -> V,io-noexnstd/core/io-noexn: E>

// The `:named` effect is the default umbrella effect for named effects
pub type nmdstd/core/nmd: X :: X

// The `:scope` effect is used to ensure named effects cannot escape the scope of their handler
pub type scopestd/core/scope: S -> X :: S -> X


// ----------------------------------------------------------------------------
// Standard Functions
// ----------------------------------------------------------------------------

// Apply a function `f` to a specified argument `x`.
pub fun applystd/core/apply: forall<a,b,e> (f : (a) -> e b, x : a) -> e b(ff: (_438) -> _439 _440,xx: _438)result: -> 1888 1889
  ff: (_438) -> _439 _440(xx: _438)

// Compose two functions `f` and `g`.
pub fun ostd/core/o: forall<a,b,c,e> (f : (a) -> e b, g : (c) -> e a) -> ((x : c) -> e b)(ff: (_531) -> _523 _524,gg: (_522) -> _523 _531)result: -> total (x : 2064) -> 2065 2066
  fnfn: (x : _522) -> _523 _524(xx: _522) ff: (_531) -> _523 _524(gg: (_522) -> _523 _531(xx: _522))

// The `ignore` function ignores its argument.
pub fun ignorestd/core/ignore: forall<a> (x : a) -> ()( xx: $484 : aa: V )result: -> total () : (std/core/types/unit: V)std/core/types/unit: V
  (std/core/types/Unit: ())std/core/types/Unit: ()

// Return a 'constant' function that ignores its argument and always returns the same result
pub fun conststd/core/const: forall<a,b> (default : a) -> ((x : b) -> a)( defaultdefault: $459 : aa: V )result: -> total (x : 470) -> 469 : totalstd/core/types/total: E (( x : bb: V ) -> astd/core/types/total: E)
  fnfn: ($460) -> $459(_) defaultdefault: $459

// Concise way to ensure two expressions have the same type.
pub fun same-typestd/core/same-type: forall<a> (x : a, y : a) -> a( xx: $556 : aa: V, yy: $556 : aa: V )result: -> total 562 : astd/core/types/total: E
  xx: $556

// ----------------------------------------------------------------------------
// Control statements
// ----------------------------------------------------------------------------

// The `while` fun executes `action`  as long as `pred`  is `true`.
pub fun whilestd/core/while: forall<e> (predicate : () -> <div|e> bool, action : () -> <div|e> ()) -> <div|e> ()( predicatepredicate: () -> <div|$1230> bool : () -> <divstd/core/types/div: X|std/core/types/effect-extend: (X, E) -> Eee: E> boolstd/core/types/bool: V, actionaction: () -> <div|$1230> () : () -> <divstd/core/types/div: X|std/core/types/effect-extend: (X, E) -> Eee: E> (std/core/types/unit: V)std/core/types/unit: V )result: -> <div|1246> () : <divstd/core/types/div: X|std/core/types/effect-extend: (X, E) -> Eee: E> (std/core/types/unit: V)std/core/types/unit: V
  if predicatepredicate: () -> <div|$1230> bool() then
    actionaction: () -> <div|$1230> ()()
    whilestd/core/while: (predicate : () -> <div|$1230> bool, action : () -> <div|$1230> ()) -> <div|$1230> ()(predicatepredicate: () -> <div|$1230> bool, actionaction: () -> <div|$1230> ())std/core/types/Unit: ()

// The `repeat` fun executes `action`  `n`  times.
pub fun repeatstd/core/repeat: forall<e> (n : int, action : () -> e ()) -> e ()( ^nn: int : intstd/core/types/int: V, actionaction: () -> $1162 () : () -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1175 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  forstd/core/for: (n : int, action : (int) -> $1162 ()) -> $1162 ()(nn: int) fnfn: (i : int) -> $1162 ()(ii: int)
    actionaction: () -> $1162 ()()


// Executes `action`  for each integer from `start` to `end` (including `end` ).
// If `start > end`  the function returns without any call to `action` .
pub fun range/forstd/core/range/for: forall<e> (start : int, end : int, action : (int) -> e ()) -> e ()( ^startstart: int: intstd/core/types/int: V, endend: int : intstd/core/types/int: V, actionaction: (int) -> $1000 () : (intstd/core/types/int: V) -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1039 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  fun reprep: (i : int) -> $1000 ()( ^ii: int : intstd/core/types/int: V )result: -> $1000 ()
    if ii: int <=std/core/int/(<=): (x : int, y : int) -> $1000 bool endend: int then
      actionaction: (int) -> $1000 ()(ii: int)
      reprep: (i : int) -> $1000 ()(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $1000 int(ii: int.incstd/core/int/inc: (i : int) -> $1000 int))std/core/types/Unit: ()
  reprep: (i : int) -> $1000 ()(startstart: int)

// Executes `action` `n` times for each integer from `0` to `n-1`.
// If `n <= 0`  the function returns without any call to `action` .
pub fun forstd/core/for: forall<e> (n : int, action : (int) -> e ()) -> e ()( ^nn: int : intstd/core/types/int: V, actionaction: (int) -> $1043 () : (intstd/core/types/int: V) -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1058 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  range/forstd/core/range/for: (start : int, end : int, action : (int) -> $1043 ()) -> $1043 ()(0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int -std/core/int/(-): (x : int, y : int) -> $1043 int 1literal: int
dec = 1
hex8 = 0x01
bit8 = 0b00000001
, actionaction: (int) -> $1043 ()
) // Executes `action` for each integer between `start` to `end` (including `end` ). // If `start > end` the function returns without any call to `action` . // If `action` returns `Just`, the iteration is stopped and the result returned pub fun range/for-whilestd/core/range/for-while: forall<a,e> (start : int, end : int, action : (int) -> e maybe<a>) -> e maybe<a>( startstart: int: intstd/core/types/int: V, endend: int : intstd/core/types/int: V, actionaction: (int) -> $1065 maybe<$1064> : (intstd/core/types/int: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 1122 maybe<1121> : ee: E maybestd/core/types/maybe: V -> V<aa: V> fun reprep: (i : int) -> $1065 maybe<$1064>( ii: int : intstd/core/types/int: V )result: -> $1065 maybe<$1064> if ii: int <=std/core/int/(<=): (x : int, y : int) -> $1065 bool endend: int then match actionaction: (int) -> $1065 maybe<$1064>(ii: int) Nothingstd/core/types/Nothing: forall<a> maybe<a> -> reprep: (i : int) -> $1065 maybe<$1064>(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $1065 int(ii: int.incstd/core/int/inc: (i : int) -> $1065 int)) Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $1064) -> Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $1064) else Nothingstd/core/types/Nothing: forall<a> maybe<a> reprep: (i : int) -> $1065 maybe<$1064>(startstart: int) // Executes `action` for each integer between [0,`n`) (excluding `n` ). // If `n <= 0` the function returns without any call to `action` . // If `action` returns `Just`, the iteration is stopped and the result returned pub fun for-whilestd/core/for-while: forall<a,e> (n : int, action : (int) -> e maybe<a>) -> e maybe<a>( nn: int : intstd/core/types/int: V, actionaction: (int) -> $1130 maybe<$1129> : (intstd/core/types/int: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 1151 maybe<1150> : ee: E maybestd/core/types/maybe: V -> V<aa: V> range/for-whilestd/core/range/for-while: (start : int, end : int, action : (int) -> $1130 maybe<$1129>) -> $1130 maybe<$1129>(0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int -std/core/int/(-): (x : int, y : int) -> $1130 int 1literal: int
dec = 1
hex8 = 0x01
bit8 = 0b00000001
, actionaction: (int) -> $1130 maybe<$1129>
) // Fold over the integers between [`start`,`end`] (including `end`). pub fun range/foldstd/core/range/fold: forall<a,e> (start : int, end : int, init : a, f : (int, a) -> e a) -> e a( startstart: int : intstd/core/types/int: V, endend: int : intstd/core/types/int: V, initinit: $834 : aa: V, ff: (int, $834) -> $835 $834 : (intstd/core/types/int: V,aa: V) -> ee: E aa: V )result: -> 886 885 : ee: E aa: V if startstart: int >std/core/int/(>): (x : int, y : int) -> $835 bool endend: int then initinit: $834 else val xx: $834 = ff: (int, $834) -> $835 $834(startstart: int,initinit: $834) foldstd/core/range/fold: (start : int, end : int, init : $834, f : (int, $834) -> $835 $834) -> $835 $834(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $835 int(startstart: int.incstd/core/int/inc: (i : int) -> $835 int), endend: int, xx: $834, ff: (int, $834) -> $835 $834) // Fold over the integers between [0,`upto`) (excluding `upto`). pub fun foldstd/core/fold: forall<a,e> (upto : int, init : a, f : (int, a) -> e a) -> e a( uptoupto: int : intstd/core/types/int: V, initinit: $811 : aa: V, ff: (int, $811) -> $812 $811 : (intstd/core/types/int: V,aa: V) -> ee: E aa: V )result: -> 833 832 : ee: E aa: V range/foldstd/core/range/fold: (start : int, end : int, init : $811, f : (int, $811) -> $812 $811) -> $812 $811( 0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, uptoupto: int.decstd/core/int/dec: (i : int) -> $812 int, initinit: $811, ff: (int, $811) -> $812 $811
) // Fold over the integers between [`start`,`end`] (including `end`) or until `f` returns `Nothing` pub fun range/fold-whilestd/core/range/fold-while: forall<a,e> (start : int, end : int, init : a, f : (int, a) -> e maybe<a>) -> e a( startstart: int : intstd/core/types/int: V, endend: int : intstd/core/types/int: V, initinit: $926 : aa: V, ff: (int, $926) -> $927 maybe<$926> : (intstd/core/types/int: V,aa: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 983 982 : ee: E aa: V if startstart: int >std/core/int/(>): (x : int, y : int) -> $927 bool endend: int then initinit: $926 else match ff: (int, $926) -> $927 maybe<$926>(startstart: int,initinit: $926) Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $926) -> range/fold-whilestd/core/range/fold-while: (start : int, end : int, init : $926, f : (int, $926) -> $927 maybe<$926>) -> $927 $926(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $927 int(startstart: int.incstd/core/int/inc: (i : int) -> $927 int), endend: int, xx: $926, ff: (int, $926) -> $927 maybe<$926>) Nothingstd/core/types/Nothing: forall<a> maybe<a> -> initinit: $926 // Fold over the integers between [0,`n`) (excluding `n`) or until `f` returns `Nothing` pub fun fold-whilestd/core/fold-while: forall<a,e> (n : int, init : a, f : (int, a) -> e maybe<a>) -> e a( nn: int : intstd/core/types/int: V, initinit: $903 : aa: V, ff: (int, $903) -> $904 maybe<$903> : (intstd/core/types/int: V,aa: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 925 924 : ee: E aa: V range/fold-whilestd/core/range/fold-while: (start : int, end : int, init : $903, f : (int, $903) -> $904 maybe<$903>) -> $904 $903( 0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int.decstd/core/int/dec: (i : int) -> $904 int, initinit: $903, ff: (int, $903) -> $904 maybe<$903>
) // ---------------------------------------------------------------------------- // Generic equality and comparison // ---------------------------------------------------------------------------- // Generic inequality pub fun (!=)std/core/(!=): forall<a> (x : a, y : a, @implicit/(==) : (a, a) -> bool) -> bool(xx: $587 : aa: V, yy: $587 : aa: V, (@implicit/==)?(==): ($587, $587) -> bool : (aa: V,aa: V) -> boolstd/core/types/bool: V )result: -> total bool : boolstd/core/types/bool: V notstd/core/types/not: (b : bool) -> bool(xx: $587==?(==): ($587, $587) -> boolyy: $587) // Generic equality if `cmp` exists pub fun cmp/(==)std/core/cmp/(==): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $402 : aa: V, yy: $402 : aa: V, @implicit/cmp?cmp: ($402, $402) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V match cmp?cmp: ($402, $402) -> order(xx: $402,yy: $402) Eqstd/core/types/Eq: order -> Truestd/core/types/True: bool _ -> Falsestd/core/types/False: bool // Generic greater than pub fun (>)std/core/(>): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $691 : aa: V, yy: $691 : aa: V, @implicit/cmp?cmp: ($691, $691) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V cmp?cmp: ($691, $691) -> order(xx: $691,yy: $691) ==std/core/order/(==): (x : order, y : order) -> bool Gtstd/core/types/Gt: order // Generic lower than pub fun (<)std/core/(<): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $631 : aa: V, yy: $631 : aa: V, @implicit/cmp?cmp: ($631, $631) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V cmp?cmp: ($631, $631) -> order(xx: $631,yy: $631) ==std/core/order/(==): (x : order, y : order) -> bool Ltstd/core/types/Lt: order // Generic greater than or equal pub fun (>=)std/core/(>=): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $781 : aa: V, yy: $781 : aa: V, @implicit/cmp?cmp: ($781, $781) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V yy: $781 <std/core/(<): (x : $781, y : $781, @implicit/cmp : ($781, $781) -> order) -> bool
?cmp=?cmp
xx: $781 // Generic lower than or equal pub fun (<=)std/core/(<=): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $751 : aa: V, yy: $751 : aa: V, @implicit/cmp?cmp: ($751, $751) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V yy: $751 >std/core/(>): (x : $751, y : $751, @implicit/cmp : ($751, $751) -> order) -> bool
?cmp=?cmp
xx: $751 // ---------------------------------------------------------------------------- // Main // ---------------------------------------------------------------------------- // Used by the compiler to wrap main console applications pub extern main-consolestd/core/main-console: forall<a,e> (main : () -> e a) -> e a : forall<aa: V,ee: E> ( main : () -> ee: E aa: V ) -> ee: E aa: V c "kk_main_console" cs inline "Primitive.MainConsole<##1>(#1)" js inline "(#1)()" // Return the host environment: `dotnet`, `browser`, `webworker`, `node`, or `libc`. pub extern hoststd/core/host: () -> ndet string() : ndetstd/core/types/ndet: X stringstd/core/types/string: V c "kk_get_host" cs inline "\"dotnet\"" js inline "$std_core_console._host" // The default exception handler pub fun @default-exn(actionaction: () -> <exn,console|$249> () : () -> <exnstd/core/exn/exn: (E, V) -> V,consolestd/core/console/console: X|std/core/types/effect-extend: (X, E) -> Eee: E> (std/core/types/unit: V)std/core/types/unit: V )result: -> <console|398> () : <consolestd/core/console/console: X|std/core/types/effect-extend: (X, E) -> Eee: E> (std/core/types/unit: V)std/core/types/unit: V handlehandler: (() -> <exn,console|$249> ()) -> <console|$249> ()( actionaction: () -> <exn,console|$249> () ) final ctl throw-exnthrow-exn: (exn : exception) -> <console|$249> ()( exnexn: exception : exceptionstd/core/exn/exception: V ) "uncaught exception: "literal: string
count= 20
.printstd/core/console/string/print: (s : string) -> <console|$249> () exnexn: exception.printlnstd/core/console/show/println: (x : exception, @implicit/show : (exception) -> string) -> <console|$249> ()
?show=show
// ---------------------------------------------------------------------------- // Non determinism // ---------------------------------------------------------------------------- noinline val unique-countstd/core/unique-count: ref<global,int> : refstd/core/types/ref: (H, V) -> V<globalstd/core/types/global: H,intstd/core/types/int: V> = unsafe-totalstd/core/unsafe/unsafe-total: (action : () -> <alloc<global>|_584> ref<global,int>) -> ref<global,int>{ refstd/core/types/ref: (value : int) -> <alloc<global>|_584> ref<global,int>(0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
)
} // Returns a unique integer (modulo 32-bits). pub fun uniquestd/core/unique: () -> ndet int()result: -> ndet int : ndetstd/core/types/ndet: X intstd/core/types/int: V unsafe-totalstd/core/unsafe/unsafe-total: (action : () -> <read<global>,write<global>|_1224> int) -> ndet int val uu: int = !std/core/types/ref/(!): (ref : ref<global,int>) -> <read<global>,write<global>|_1224> intunique-countstd/core/unique-count: ref<global,int> unique-countstd/core/unique-count: ref<global,int> :=std/core/types/set: (ref : ref<global,int>, assigned : int) -> <write<global>,read<global>|_1224> () uu: int+std/core/int/(+): (x : int, y : int) -> <write<global>,read<global>|_1224> int1literal: int
dec = 1
hex8 = 0x01
bit8 = 0b00000001
uu: int // ---------------------------------------------------------------------------- // Stream // ---------------------------------------------------------------------------- // A `:stream` is a co-inductive type representing an infinite list of elements. pub co type streamstd/core/stream: V -> V<aa: V> con Nextstd/core/Next: forall<a> (head : a, tail : stream<a>) -> stream<a>(headstd/core/stream/head: forall<a> (stream : stream<a>) -> a:aa: V, tailstd/core/stream/tail: forall<a> (stream : stream<a>) -> stream<a>: streamstd/core/stream: V -> V<aa: V> )