aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--lux-lua/source/program.lux287
-rw-r--r--stdlib/source/lux/target/lua.lux13
-rw-r--r--stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/lua/common.lux15
-rw-r--r--stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/python/common.lux2
-rw-r--r--stdlib/source/lux/tool/compiler/language/lux/phase/generation/lua/runtime.lux54
-rw-r--r--stdlib/source/lux/tool/compiler/language/lux/phase/generation/python/runtime.lux17
-rw-r--r--stdlib/source/test/lux/extension.lux6
7 files changed, 301 insertions, 93 deletions
diff --git a/lux-lua/source/program.lux b/lux-lua/source/program.lux
index 180757de6..0dc8c5564 100644
--- a/lux-lua/source/program.lux
+++ b/lux-lua/source/program.lux
@@ -20,7 +20,7 @@
[macro
["." template]]
[math
- [number
+ [number (#+ hex)
["n" nat]
["." i64]]]
["." world #_
@@ -329,55 +329,244 @@
Expander
(#try.Success ((:coerce Macro' macro) inputs lux)))})
-(for {@.old (def: host
- (IO [Baggage (Host _.Expression _.Statement)])
- (io (let [runtime_env (net/sandius/rembulan/env/RuntimeEnvironments::system)
- std_lib (net/sandius/rembulan/lib/StandardLibrary::in runtime_env)
- state_context (net/sandius/rembulan/impl/StateContexts::newDefaultInstance)
- table (net/sandius/rembulan/lib/StandardLibrary::installInto state_context std_lib)
- variable (net/sandius/rembulan/Variable::new table)
- loader (net/sandius/rembulan/compiler/CompilerChunkLoader::of "_lux_definition")
- executor (net/sandius/rembulan/exec/DirectCallExecutor::newExecutor)
- scheduling_context (net/sandius/rembulan/exec/DirectCallExecutor::schedulingContextFactory executor)
- run! (: (-> _.Statement (Try Any))
- (function (_ code)
+(for {@.old (as_is (with_expansions [$var_args (_.var "...")
+ $str_rel_to_abs (_.var "_utf8_str_rel_to_abs")
+ $decode (_.var "_utf8_decode")]
+ (template.with [(!int <hex>)
+ (_.int (.int (hex <hex>)))
+
+ (!&| <or> <and> <raw>)
+ (|> <raw>
+ (_.bit_and (!int <and>))
+ (_.bit_or (!int <or>)))
+
+ (!&|< <or> <and> <shift> <raw>)
+ (|> <raw>
+ (_.bit_shr (_.int <shift>))
+ (_.bit_and (!int <and>))
+ (_.bit_or (!int <or>)))]
+ (as_is (def: rembulan//char
+ (let [$buffer (_.var "buffer")
+ $k (_.var "k")
+ $v (_.var "v")
+ $b1 (_.var "b1")
+ $b2 (_.var "b2")
+ $b3 (_.var "b3")
+ $b4 (_.var "b4")
+ table/insert (_.apply/2 (_.var "table.insert"))]
+ (_.function (_.var "utf8.char") (list $var_args)
+ ($_ _.then
+ (_.local/1 $buffer (_.array (list)))
+ (<| (_.for_in (list $k $v) (_.ipairs/1 (_.array (list $var_args))))
+ ($_ _.then
+ (_.when (_.or (_.< (_.int +0) $v)
+ (_.> (!int "10FFFF") $v))
+ (_.statement (_.error/2 (|> (_.string "bad argument #")
+ (_.concat $k)
+ (_.concat (_.string " to char (out of range)")))
+ (_.int +2))))
+ (<| (_.if (_.< (!int "80") $v)
+ ## Single-byte sequence
+ (_.statement (|> (_.var "string.char")
+ (_.apply/* (list $v))
+ (table/insert $buffer))))
+ (_.if (_.< (!int "800") $v)
+ ## Two-byte sequence
+ (_.statement (|> (_.var "string.char")
+ (_.apply/* (list (!&|< "C0" "1F" +6 $v)
+ (!&| "80" "3F" $v)))
+ (table/insert $buffer))))
+ (_.if (_.< (!int "10000") $v)
+ ## Three-byte sequence
+ (_.statement (|> (_.var "string.char")
+ (_.apply/* (list (!&|< "E0" "0F" +12 $v)
+ (!&|< "80" "3F" +6 $v)
+ (!&| "80" "3F" $v)))
+ (table/insert $buffer))))
+ ## Four-byte sequence
+ (_.statement (|> (_.var "string.char")
+ (_.apply/* (list (!&|< "F0" "07" +18 $v)
+ (!&|< "80" "3F" +12 $v)
+ (!&|< "80" "3F" +6 $v)
+ (!&| "80" "3F" $v)))
+ (table/insert $buffer))))
+ ))
+ (_.return (_.apply/2 (_.var "table.concat") $buffer (_.string "")))
+ ))))
+
+ ## (def: rembulan//str_rel_to_abs
+ ## (let [$string (_.var "string")
+ ## $args (_.var "args")
+ ## $k (_.var "k")
+ ## $v (_.var "v")]
+ ## (<| (_.local_function $str_rel_to_abs (list $string $var_args))
+ ## ($_ _.then
+ ## (_.local/1 $args (_.array (list $var_args)))
+ ## (<| (_.for_in (list $k $v) (_.ipairs/1 $args))
+ ## ($_ _.then
+ ## (_.if (_.> (_.int +0) $v)
+ ## (_.set (list $v) $v)
+ ## (_.set (list $v) (|> $v (_.+ (_.length $string)) (_.+ (_.int +1)))))
+ ## (_.when (_.or (_.< (_.int +1) $v)
+ ## (_.> (_.length $string) $v))
+ ## (_.statement (_.error/2 (_.string "bad index to string (out of range)") (_.int +3))))
+ ## (_.set (list (_.nth $k $args)) $v)))
+ ## (_.return (_.apply/1 (_.var "table.unpack") $args))
+ ## ))))
+
+ ## (def: rembulan//decode
+ ## (let [$string (_.var "string")
+ ## $start (_.var "start")
+ ## $b1 (_.var "b1")
+ ## $idx (_.var "idx")
+ ## $bx (_.var "bx")
+ ## $end (_.var "_end")]
+ ## (<| (_.local_function $decode (list $string $start))
+ ## ($_ _.then
+ ## (_.set (list $start) (_.apply/2 $str_rel_to_abs $string (_.or (_.int +1) $start)))
+ ## (_.local/1 $b1 (_.do "byte" (list $start $start) $string))
+ ## (<| (_.if (_.< (!int "80") $b1)
+ ## ## Single-byte sequence
+ ## (_.return (_.multi (list $start $start))))
+ ## ## Validate first byte of multi-byte sequence
+ ## (_.if (_.or (_.> (!int "F4") $b1)
+ ## (_.< (!int "C2") $b1))
+ ## (_.return _.nil))
+ ## ## Get 'supposed' amount of continuation bytes from primary byte
+ ## ($_ _.then
+ ## (_.local/1 $end (|> (|> $b1 (_.>= (!int "F0")) (_.and (_.int +3)))
+ ## (_.or (|> $b1 (_.>= (!int "E0")) (_.and (_.int +2))))
+ ## (_.or (|> $b1 (_.>= (!int "C0")) (_.and (_.int +1))))
+ ## (_.+ $start)))
+ ## ## Validate our continuation bytes
+ ## (<| (_.for_in (list $idx $bx) (_.ipairs/1 (_.array (list (_.do "byte"
+ ## (list (_.+ (_.int +1) $start) $end)
+ ## $string)))))
+ ## (_.when (|> $bx
+ ## (_.bit_and (!int "C0"))
+ ## (_.= (!int "80"))
+ ## _.not)
+ ## (_.return _.nil)))
+ ## (_.return (_.multi (list $start $end)))
+ ## ))
+ ## ))))
+
+ ## (def: rembulan//codes
+ ## (let [$string (_.var "string")
+ ## $i (_.var "i")
+ ## $start (_.var "start")
+ ## $end (_.var "_end")]
+ ## (_.function (_.var "utf8.codes") (list $string)
+ ## ($_ _.then
+ ## (_.local/1 $i (_.int +1))
+ ## (_.return (<| (_.closure (list))
+ ## (_.if (_.> (_.length $string) $i)
+ ## (_.return _.nil)
+ ## ($_ _.then
+ ## (_.let (list $start $end) (_.apply/2 $decode $string $i))
+ ## (_.if (_.not $start)
+ ## (_.statement (_.error/2 (_.string "invalid UTF-8 code") (_.int +2)))
+ ## ($_ _.then
+ ## (_.set (list $i) (_.+ (_.int +1) $end))
+ ## (_.return (_.multi (list $start (_.do "sub" (list $start $end) $string))))
+ ## ))
+ ## ))))
+ ## ))))
+
+ ## (def: rembulan//len
+ ## (let [$string (_.var "string")
+ ## $start (_.var "start")
+ ## $end (_.var "_end")
+ ## $seq_start (_.var "seq_start")
+ ## $seq_end (_.var "seq_end")
+ ## $size (_.var "size")]
+ ## (_.function (_.var "utf8.len") (list $string $start $end)
+ ## ($_ _.then
+ ## (_.set (list $start $end) (_.apply/3 $str_rel_to_abs $string (_.or (_.int +1) $start) (_.or (_.int -1) $end)))
+ ## (_.local/1 $size (_.int +0))
+ ## (_.repeat (_.>= $end $seq_end)
+ ## ($_ _.then
+ ## (_.let (list $seq_start $seq_end) (_.apply/2 $decode $string $start))
+ ## (_.if (_.not $seq_start)
+ ## ## Hit an invalid sequence!
+ ## (_.return (_.multi (list (_.bool false) $start)))
+ ## ($_ _.then
+ ## (_.set (list $start) (_.+ (_.int +1) $seq_end))
+ ## (_.set (list $size) (_.+ (_.int +1) $size))
+ ## ))
+ ## ))
+ ## (_.return $size)
+ ## ))))
+
+ ## (def: rembulan//charpattern
+ ## (_.set (list (_.var "utf8.charpattern"))
+ ## (_.string "[%z\x01-\x7F\xC2-\xF4][\x80-\xBF]*")))
+
+ (def: rembulan_prelude
+ _.Statement
+ ($_ _.then
+ (_.function (_.var "os.time") (list)
+ (_.return (_.int +0)))
+
+ ## Ported from https://github.com/meepen/Lua-5.1-UTF-8
+ ..rembulan//char
+ ## ..rembulan//str_rel_to_abs
+ ## ..rembulan//decode
+ ## ..rembulan//codes
+ ## ..rembulan//len
+ ## ..rembulan//charpattern
+ )))))
+
+ (def: host
+ (IO [Baggage (Host _.Expression _.Statement)])
+ (io (let [runtime_env (net/sandius/rembulan/env/RuntimeEnvironments::system)
+ std_lib (net/sandius/rembulan/lib/StandardLibrary::in runtime_env)
+ state_context (net/sandius/rembulan/impl/StateContexts::newDefaultInstance)
+ table (net/sandius/rembulan/lib/StandardLibrary::installInto state_context std_lib)
+ variable (net/sandius/rembulan/Variable::new table)
+ loader (net/sandius/rembulan/compiler/CompilerChunkLoader::of "_lux_definition")
+ executor (net/sandius/rembulan/exec/DirectCallExecutor::newExecutor)
+ scheduling_context (net/sandius/rembulan/exec/DirectCallExecutor::schedulingContextFactory executor)
+ run! (: (-> _.Statement (Try Any))
+ (function (_ code)
+ (do try.monad
+ [lua_function (net/sandius/rembulan/load/ChunkLoader::loadTextChunk variable "lux compilation" (_.code code) loader)
+ output (net/sandius/rembulan/exec/DirectCallExecutor::call state_context (:coerce java/lang/Object lua_function) (array.new 0)
+ executor)]
+ (case (array.read 0 output)
+ #.None
+ (wrap [])
+
+ (#.Some value)
+ (read value)))))
+ _ (try.assume (run! ..rembulan_prelude))]
+ [[state_context executor]
+ (: (Host _.Expression _.Statement)
+ (structure
+ (def: (evaluate! context code)
+ (run! (_.return code)))
+
+ (def: execute! run!)
+
+ (def: (define! context input)
+ (let [global (reference.artifact context)
+ @global (_.var global)]
+ (do try.monad
+ [#let [definition (_.set (list @global) input)]
+ _ (run! definition)
+ value (run! (_.return @global))]
+ (wrap [global value definition]))))
+
+ (def: (ingest context content)
+ (|> content (\ encoding.utf8 decode) try.assume (:coerce _.Statement)))
+
+ (def: (re_learn context content)
+ (run! content))
+
+ (def: (re_load context content)
(do try.monad
- [lua_function (net/sandius/rembulan/load/ChunkLoader::loadTextChunk variable "lux compilation" (_.code code) loader)
- output (net/sandius/rembulan/exec/DirectCallExecutor::call state_context (:coerce java/lang/Object lua_function) (array.new 0)
- executor)]
- (case (array.read 0 output)
- #.None
- (wrap [])
-
- (#.Some value)
- (read value)))))]
- [[state_context executor]
- (: (Host _.Expression _.Statement)
- (structure
- (def: (evaluate! context code)
- (run! (_.return code)))
-
- (def: execute! run!)
-
- (def: (define! context input)
- (let [global (reference.artifact context)
- @global (_.var global)]
- (do try.monad
- [#let [definition (_.set (list @global) input)]
- _ (run! definition)
- value (run! (_.return @global))]
- (wrap [global value definition]))))
-
- (def: (ingest context content)
- (|> content (\ encoding.utf8 decode) try.assume (:coerce _.Statement)))
-
- (def: (re_learn context content)
- (run! content))
-
- (def: (re_load context content)
- (do try.monad
- [_ (run! content)]
- (run! (_.return (_.var (reference.artifact context))))))))])))
+ [_ (run! content)]
+ (run! (_.return (_.var (reference.artifact context))))))))]))))
@.lua (as_is (host.import: (load [host.String] #try host.Function))
(def: host
(IO (Host _.Expression _.Statement))
diff --git a/stdlib/source/lux/target/lua.lux b/stdlib/source/lux/target/lua.lux
index c557c7feb..29d4b82b3 100644
--- a/stdlib/source/lux/target/lua.lux
+++ b/stdlib/source/lux/target/lua.lux
@@ -297,6 +297,13 @@
(..nest (:representation body!))
text.new_line "end")))
+ (def: #export (repeat until body!)
+ (-> Expression Statement Statement)
+ (:abstraction
+ (format "repeat"
+ (..nest (:representation body!))
+ text.new_line "until " (:representation until))))
+
(def: #export (for_in vars source body!)
(-> (List Var) Expression Statement Statement)
(:abstraction
@@ -396,10 +403,12 @@
[["error"]
["print"]
["require"]
- ["type"]]]
+ ["type"]
+ ["ipairs"]]]
[2
- [["print"]]]
+ [["print"]
+ ["error"]]]
[3
[["print"]]]
diff --git a/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/lua/common.lux b/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/lua/common.lux
index ba12035c7..29d3704fe 100644
--- a/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/lua/common.lux
+++ b/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/lua/common.lux
@@ -104,13 +104,10 @@
(/.install "-" (binary (product.uncurry _.-)))
(/.install "<" (binary (product.uncurry _.<)))
(/.install "*" (binary (product.uncurry _.*)))
- (/.install "/" (binary (product.uncurry _.//)))
- (/.install "%" (binary (product.uncurry _.%)))
+ (/.install "/" (binary (product.uncurry //runtime.i64//division)))
+ (/.install "%" (binary (product.uncurry //runtime.i64//remainder)))
(/.install "f64" (unary (_./ (_.float +1.0))))
- (/.install "char" (unary //runtime.i64//char))
- ## TODO: Use version below once the Lua compiler becomes self-hosted.
- ## (/.install "char" (unary (for {@.lua (!unary "utf8.char")}
- ## (!unary "string.char"))))
+ (/.install "char" (unary (_.apply/1 (_.var "utf8.char"))))
)))
(def: f64//decode
@@ -129,7 +126,7 @@
(/.install "=" (binary (product.uncurry _.=)))
(/.install "<" (binary (product.uncurry _.<)))
(/.install "i64" (unary (!unary "math.floor")))
- (/.install "encode" (unary (_.apply/2 (_.var "string.format") (_.string "%.17f"))))
+ (/.install "encode" (unary (_.apply/2 (_.var "string.format") (_.string "%.17g"))))
(/.install "decode" (unary ..f64//decode)))))
(def: (text//char [paramO subjectO])
@@ -171,7 +168,9 @@
(|> /.empty
(/.install "log" (unary ..io//log!))
(/.install "error" (unary (!unary "error")))
- (/.install "current-time" (nullary (function.constant (//runtime.io//current_time //runtime.unit)))))))
+ (/.install "current-time" (nullary (function.constant (|> (_.var "os.time")
+ (_.apply/* (list))
+ (_.* (_.int +1,000)))))))))
(def: #export bundle
Bundle
diff --git a/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/python/common.lux b/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/python/common.lux
index 90cafc75b..b87390e9a 100644
--- a/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/python/common.lux
+++ b/stdlib/source/lux/tool/compiler/language/lux/phase/extension/generation/python/common.lux
@@ -108,7 +108,7 @@
(/.install "+" (binary (product.uncurry (..capped _.+))))
(/.install "-" (binary (product.uncurry (..capped _.-))))
(/.install "*" (binary (product.uncurry (..capped _.*))))
- (/.install "/" (binary (product.uncurry _.//)))
+ (/.install "/" (binary (product.uncurry //runtime.i64//division)))
(/.install "%" (binary (product.uncurry //runtime.i64//remainder)))
(/.install "f64" (unary _.float/1))
(/.install "char" (unary //runtime.i64//char))
diff --git a/stdlib/source/lux/tool/compiler/language/lux/phase/generation/lua/runtime.lux b/stdlib/source/lux/tool/compiler/language/lux/phase/generation/lua/runtime.lux
index 503969782..14d206e23 100644
--- a/stdlib/source/lux/tool/compiler/language/lux/phase/generation/lua/runtime.lux
+++ b/stdlib/source/lux/tool/compiler/language/lux/phase/generation/lua/runtime.lux
@@ -274,25 +274,31 @@
(_.bit_shr param)
(_.bit_and mask))))))
-## TODO: Remove this once the Lua compiler becomes self-hosted.
-(def: on_rembulan?
- (_.= (_.string "Lua 5.3")
- (_.var "_VERSION")))
-
-(runtime: (i64//char subject)
- (with_expansions [<rembulan> (_.return (_.apply/1 (_.var "string.char") subject))
- <normal> (_.return (_.apply/1 (_.var "utf8.char") subject))]
- (for {@.lua (_.return <normal>)}
- (_.if ..on_rembulan?
- <rembulan>
- <normal>))))
+(runtime: (i64//division param subject)
+ (with_vars [floored]
+ ($_ _.then
+ (_.local/1 floored (_.// param subject))
+ (let [potentially_floored? (_.< (_.int +0) floored)
+ inexact? (|> floored
+ (_.* param)
+ (_.= subject)
+ _.not)]
+ (_.if (_.and potentially_floored?
+ inexact?)
+ (_.return (_.+ (_.int +1) floored))
+ (_.return floored))))))
+
+(runtime: (i64//remainder param subject)
+ (_.return (_.- (|> subject (..i64//division param) (_.* param))
+ subject)))
(def: runtime//i64
Statement
($_ _.then
@i64//left_shift
@i64//right_shift
- @i64//char
+ @i64//division
+ @i64//remainder
))
(def: (find_byte_index subject param start)
@@ -314,6 +320,11 @@
(-> Expression Expression)
(_.- (_.int +1)))
+## TODO: Remove this once the Lua compiler becomes self-hosted.
+(def: on_rembulan?
+ (_.= (_.string "Lua 5.3")
+ (_.var "_VERSION")))
+
(runtime: (text//index subject param start)
(with_expansions [<rembulan> ($_ _.then
(_.local/1 byte_index (|> start
@@ -398,22 +409,6 @@
@array//write
))
-(runtime: (io//current_time _)
- (with_expansions [<rembulan> (_.return (_.int +0))
- <normal> (_.return (|> (_.var "os.time")
- (_.apply/* (list))
- (_.* (_.int +1,000))))]
- (for {@.lua <normal>}
- (_.if ..on_rembulan?
- <rembulan>
- <normal>))))
-
-(def: runtime//io
- Statement
- ($_ _.then
- @io//current_time
- ))
-
(def: runtime
Statement
($_ _.then
@@ -422,7 +417,6 @@
..runtime//i64
..runtime//text
..runtime//array
- ..runtime//io
))
(def: #export artifact ..prefix)
diff --git a/stdlib/source/lux/tool/compiler/language/lux/phase/generation/python/runtime.lux b/stdlib/source/lux/tool/compiler/language/lux/phase/generation/python/runtime.lux
index a2e18808a..1af62cf7e 100644
--- a/stdlib/source/lux/tool/compiler/language/lux/phase/generation/python/runtime.lux
+++ b/stdlib/source/lux/tool/compiler/language/lux/phase/generation/python/runtime.lux
@@ -309,8 +309,22 @@
..as_nat
(_.bit_shr param))))
+(runtime: (i64//division param subject)
+ (with_vars [floored]
+ ($_ _.then
+ (_.set (list floored) (_.// param subject))
+ (_.return (let [potentially_floored? (_.< (_.int +0) floored)
+ inexact? (|> floored
+ (_.* param)
+ (_.= subject)
+ _.not)]
+ (_.? (_.and potentially_floored?
+ inexact?)
+ (_.+ (_.int +1) floored)
+ floored))))))
+
(runtime: (i64//remainder param subject)
- (_.return (_.- (|> subject (_.// param) (_.* param))
+ (_.return (_.- (|> subject (..i64//division param) (_.* param))
subject)))
(template [<runtime> <python>]
@@ -342,6 +356,7 @@
@i64//nat_top
@i64//left_shift
@i64//right_shift
+ @i64//division
@i64//remainder
@i64//and
@i64//or
diff --git a/stdlib/source/test/lux/extension.lux b/stdlib/source/test/lux/extension.lux
index d032a47b5..67abd0eca 100644
--- a/stdlib/source/test/lux/extension.lux
+++ b/stdlib/source/test/lux/extension.lux
@@ -4,7 +4,8 @@
["@" target
["." jvm]
["." js]
- ["." python]]
+ ["." python]
+ ["." lua]]
[abstract
[monad (#+ do)]]
[control
@@ -63,7 +64,8 @@
(row.row (#jvm.Constant (#jvm.LDC (#jvm.String self))))
@.js (js.string self)
- @.python (python.unicode self)})))))
+ @.python (python.unicode self)
+ @.lua (lua.string self)})))))
(for {@.old
(as_is)}