/*---------------------------------------------------------------------------
  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/maybe2std/core/maybe2
pub import std/core/eitherstd/core/either
pub import std/core/tuplestd/core/tuple
pub import std/core/lazystd/core/lazy
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: (_177) -> _178 _179,xx: _177)result: -> 191 190
  ff: (_177) -> _178 _179(xx: _177)

// 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: (_215) -> _207 _208,gg: (_206) -> _207 _215)result: -> total (x : 231) -> 232 230
  fnfn: (x : _206) -> _207 _208(xx: _206) ff: (_215) -> _207 _208(gg: (_206) -> _207 _215(xx: _206))

// The `ignore` function ignores its argument.
pub fun ignorestd/core/ignore: forall<a> (x : a) -> ()( xx: $247 : 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: $260 : aa: V )result: -> total (x : 275) -> 274 : totalstd/core/types/total: E (( x : bb: V ) -> astd/core/types/total: E)
  fnfn: ($261) -> $260(_) defaultdefault: $260

// 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: $283 : aa: V, yy: $283 : aa: V )result: -> total 291 : astd/core/types/total: E
  xx: $283

// ----------------------------------------------------------------------------
// 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|$296> bool : () -> <divstd/core/types/div: X|std/core/types/effect-extend: (X, E) -> Eee: E> boolstd/core/types/bool: V, actionaction: () -> <div|$296> () : () -> <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|317> () : <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|$296> bool() then
    actionaction: () -> <div|$296> ()()
    whilestd/core/while: (predicate : () -> <div|$296> bool, action : () -> <div|$296> ()) -> <div|$296> ()(predicatepredicate: () -> <div|$296> bool, actionaction: () -> <div|$296> ())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: () -> $1286 () : () -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1302 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  forstd/core/for: (n : int, action : (int) -> $1286 ()) -> $1286 ()(nn: int) fnfn: (i : int) -> $1286 ()(ii: int)
    actionaction: () -> $1286 ()()


// 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) -> $1191 () : (intstd/core/types/int: V) -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1257 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  fun reprep: (i : int) -> $1191 ()( ^ii: int : intstd/core/types/int: V )result: -> $1191 ()
    if ii: int <=std/core/int/(<=): (x : int, y : int) -> $1191 bool endend: int then
      actionaction: (int) -> $1191 ()(ii: int)
      reprep: (i : int) -> $1191 ()(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $1191 int(ii: int.incstd/core/int/inc: (i : int) -> $1191 int))std/core/types/Unit: ()
  reprep: (i : int) -> $1191 ()(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) -> $1261 () : (intstd/core/types/int: V) -> ee: E (std/core/types/unit: V)std/core/types/unit: V )result: -> 1280 () : ee: E (std/core/types/unit: V)std/core/types/unit: V
  range/forstd/core/range/for: (start : int, end : int, action : (int) -> $1261 ()) -> $1261 ()(0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int -std/core/int/(-): (x : int, y : int) -> $1261 int 1literal: int
dec = 1
hex8 = 0x01
bit8 = 0b00000001
, actionaction: (int) -> $1261 ()
) // 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) -> $1307 maybe<$1306> : (intstd/core/types/int: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 1397 maybe<1396> : ee: E maybestd/core/types/maybe: V -> V<aa: V> fun reprep: (i : int) -> $1307 maybe<$1306>( ii: int : intstd/core/types/int: V )result: -> $1307 maybe<$1306> if ii: int <=std/core/int/(<=): (x : int, y : int) -> $1307 bool endend: int then match actionaction: (int) -> $1307 maybe<$1306>(ii: int) Nothingstd/core/types/Nothing: forall<a> maybe<a> -> reprep: (i : int) -> $1307 maybe<$1306>(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $1307 int(ii: int.incstd/core/int/inc: (i : int) -> $1307 int)) Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $1306) -> Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $1306) else Nothingstd/core/types/Nothing: forall<a> maybe<a> reprep: (i : int) -> $1307 maybe<$1306>(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) -> $1405 maybe<$1404> : (intstd/core/types/int: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 1432 maybe<1431> : ee: E maybestd/core/types/maybe: V -> V<aa: V> range/for-whilestd/core/range/for-while: (start : int, end : int, action : (int) -> $1405 maybe<$1404>) -> $1405 maybe<$1404>(0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int -std/core/int/(-): (x : int, y : int) -> $1405 int 1literal: int
dec = 1
hex8 = 0x01
bit8 = 0b00000001
, actionaction: (int) -> $1405 maybe<$1404>
) // 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: $698 : aa: V, ff: (int, $698) -> $699 $698 : (intstd/core/types/int: V,aa: V) -> ee: E aa: V )result: -> 778 777 : ee: E aa: V if startstart: int >std/core/int/(>): (x : int, y : int) -> $699 bool endend: int then initinit: $698 else val xx: $698 = ff: (int, $698) -> $699 $698(startstart: int,initinit: $698) foldstd/core/range/fold: (start : int, end : int, init : $698, f : (int, $698) -> $699 $698) -> $699 $698(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $699 int(startstart: int.incstd/core/int/inc: (i : int) -> $699 int), endend: int, xx: $698, ff: (int, $698) -> $699 $698) // 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: $669 : aa: V, ff: (int, $669) -> $670 $669 : (intstd/core/types/int: V,aa: V) -> ee: E aa: V )result: -> 697 696 : ee: E aa: V range/foldstd/core/range/fold: (start : int, end : int, init : $669, f : (int, $669) -> $670 $669) -> $670 $669( 0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, uptoupto: int.decstd/core/int/dec: (i : int) -> $670 int, initinit: $669, ff: (int, $669) -> $670 $669
) // 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: $824 : aa: V, ff: (int, $824) -> $825 maybe<$824> : (intstd/core/types/int: V,aa: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 912 911 : ee: E aa: V if startstart: int >std/core/int/(>): (x : int, y : int) -> $825 bool endend: int then initinit: $824 else match ff: (int, $824) -> $825 maybe<$824>(startstart: int,initinit: $824) Juststd/core/types/Just: forall<a> (value : a) -> maybe<a>(xx: $824) -> range/fold-whilestd/core/range/fold-while: (start : int, end : int, init : $824, f : (int, $824) -> $825 maybe<$824>) -> $825 $824(pretend-decreasingstd/core/undiv/pretend-decreasing: (x : int) -> $825 int(startstart: int.incstd/core/int/inc: (i : int) -> $825 int), endend: int, xx: $824, ff: (int, $824) -> $825 maybe<$824>) Nothingstd/core/types/Nothing: forall<a> maybe<a> -> initinit: $824 // 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: $795 : aa: V, ff: (int, $795) -> $796 maybe<$795> : (intstd/core/types/int: V,aa: V) -> ee: E maybestd/core/types/maybe: V -> V<aa: V> )result: -> 823 822 : ee: E aa: V range/fold-whilestd/core/range/fold-while: (start : int, end : int, init : $795, f : (int, $795) -> $796 maybe<$795>) -> $796 $795( 0literal: int
dec = 0
hex8 = 0x00
bit8 = 0b00000000
, nn: int.decstd/core/int/dec: (i : int) -> $796 int, initinit: $795, ff: (int, $795) -> $796 maybe<$795>
) // ---------------------------------------------------------------------------- // Generic equality and comparison // ---------------------------------------------------------------------------- // Generic inequality pub fun default/(!=)std/core/default/(!=): forall<a> (x : a, y : a, @implicit/(==) : (a, a) -> bool) -> bool(xx: $401 : aa: V, yy: $401 : aa: V, (@implicit/==)?(==): ($401, $401) -> 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: $401==?(==): ($401, $401) -> boolyy: $401) // Generic equality if `cmp` exists pub fun default/cmp/(==)std/core/default/cmp/(==): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $327 : aa: V, yy: $327 : aa: V, @implicit/cmp?cmp: ($327, $327) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V match cmp?cmp: ($327, $327) -> order(xx: $327,yy: $327) Eqstd/core/types/Eq: order -> Truestd/core/types/True: bool _ -> Falsestd/core/types/False: bool // Generic greater than pub fun default/cmp/(>)std/core/default/cmp/(>): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $509 : aa: V, yy: $509 : aa: V, @implicit/cmp?cmp: ($509, $509) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V cmp?cmp: ($509, $509) -> order(xx: $509,yy: $509) ==std/core/order/(==): (x : order, y : order) -> bool Gtstd/core/types/Gt: order // Generic lower than pub fun default/cmp/(<)std/core/default/cmp/(<): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $929 : aa: V, yy: $929 : aa: V, @implicit/cmp?cmp: ($929, $929) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V cmp?cmp: ($929, $929) -> order(xx: $929,yy: $929) ==std/core/order/(==): (x : order, y : order) -> bool Ltstd/core/types/Lt: order // Generic greater than or equal pub fun default/cmp/(>=)std/core/default/cmp/(>=): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $1089 : aa: V, yy: $1089 : aa: V, @implicit/cmp?cmp: ($1089, $1089) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V yy: $1089 <std/core/default/cmp/(<): (x : $1089, y : $1089, @implicit/cmp : ($1089, $1089) -> order) -> bool
?cmp=?cmp
xx: $1089 // Generic lower than or equal pub fun default/cmp/(<=)std/core/default/cmp/(<=): forall<a> (x : a, y : a, @implicit/cmp : (a, a) -> order) -> bool(xx: $1140 : aa: V, yy: $1140 : aa: V, @implicit/cmp?cmp: ($1140, $1140) -> order : (aa: V,aa: V) -> orderstd/core/types/order: V )result: -> total bool : boolstd/core/types/bool: V yy: $1140 >std/core/default/cmp/(>): (x : $1140, y : $1140, @implicit/cmp : ($1140, $1140) -> order) -> bool
?cmp=?cmp
xx: $1140 // ---------------------------------------------------------------------------- // 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|$1478> () : () -> <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|1666> () : <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|$1478> ()) -> <console|$1478> ()( actionaction: () -> <exn,console|$1478> () ) final ctl throw-exnthrow-exn: (exn : exception) -> <console|$1478> ()( exnexn: exception : exceptionstd/core/exn/exception: V ) "uncaught exception: "literal: string
count= 20
.printstd/core/console/string/print: (s : string) -> <console|$1478> () exnexn: exception.printlnstd/core/console/default/show/println: (x : exception, @implicit/show : (exception) -> <console|$1478> string) -> <console|$1478> ()
?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>|_1688> ref<global,int>) -> ref<global,int>{ refstd/core/types/ref: (value : int) -> <alloc<global>|_1688> 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>|_1740> int) -> ndet int val uu: int = !std/core/types/ref/(!): (ref : ref<global,int>, @implicit/hdiv : hdiv<global,int,<write<global>|_1740>>) -> <read<global>,write<global>|_1740> int
?hdiv=iev@1710
unique-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>|_1740> () uu: int+std/core/int/(+): (x : int, y : int) -> <write<global>,read<global>|_1740> 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> )