Changelog
View SourceAll notable changes to this project will be documented in this file.
The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.
Unreleased
[v1.0.0-rc.2] - 2026-06-10
The third release candidate for 1.0.0. It builds on rc.1 with a major
table-storage performance win, two non-standard os epoch helpers, and
a batch of protected-call and error-value correctness fixes that bring
pcall / xpcall and Lua.call_function/3 in line with Lua 5.3 §6.1.
The public API is unchanged from rc.1.
Performance
- Split-storage tables (Erlang
:array+ map) — dense positive-integer keys (1..n) now route to an Erlang:arrayfor O(1) functional read/write and dense iteration ordering, while strings, sparse/non-positive integers, and other key types stay in the hash map with the existing iteration bookkeeping (#328). String-keyed reads (globals, fields, metatable lookups) are unchanged. Table-heavy workloads improve 28–37% at n=1000 (Apple M4,luachunk path): Build −36%, Iterate/Sum −36%, Map+Reduce −37%, Sort −28%. Build, Iterate, and Map+Reduce now beat Luerl; Sort closes most of the gap.
Added
os.time_ms()andos.time_us()— non-standard extensions returning the current epoch in milliseconds / microseconds, for programs that need sub-second precision (os.time()is unchanged and still returns whole seconds). Both are current-time-only and are documented as extensions not present in PUC-Lua (#340).- The
os.clock()monotonic origin is now seeded ininstall/1rather than lazily on the first call, so elapsed time is measured from a stable startup point instead of drifting to whenever a program first happened to callos.clock()(#340).
Fixed
deflua/2guarded heads register under their real name (#344). A guarded head with no state argument (deflua clamp(a) when is_integer(a)) was registered under the name:wheninstead ofclamp, making it uncallable from Lua (calling it raised an undefined-function error). The macro now unwraps the:whenAST node to reach the real name, matching thedeflua/3(state-arg) variant, which was never affected.Lua.call_function/3returns the terse Lua error value, not the terminal render (#336). Its{:error, reason, _}previously surfaced the terminal-formatted error string — ANSI escape codes, theat <source>:<line>:header, theSuggestion:block, stack-trace frames, and a doubledLua runtime error: … runtime error:prefix — where a programmatic value was expected.reasonis now exactly whatpcallhands back (§6.1): thesource:line:-prefixed message for string errors, and the raw value (table/number/nil/false) passed through verbatim for non-string error objects. Notereasonmay therefore now be a non-string Lua value. The raising variantLua.call_function!/3is unchanged — it still raises aLua.RuntimeExceptioncarrying the rich formatted render.pcallpasses the raised error value through as-is (#334). Per Lua 5.3 §6.1,error(value)raises an arbitrary Lua value andpcallreturns it verbatim as its second result — previously non-string values were stringified (error({code = 1})came back as"table: 0x..."). Structured error objects, numbers, booleans, andnilnow survivepcall/xpcall, and thexpcallmessage handler receives the untouched value. String messages gain the referencesource:line:position prefix (suppressed byerror(msg, 0)); notepcall's second result may therefore now be a non-string Lua value. Host-facingLua.VM.RuntimeErrorrendering is unchanged.- Protected calls no longer roll back heap effects (#331). When a function
called via
pcall/xpcall(orLua.call_function/3from Elixir) raised an error, mutations made before the error — global writes, table field updates, upvalue assignments, metatable changes — were silently discarded, diverging from reference Lua. VM exceptions now carry the raise-time state, and protected-call boundaries recover it: heap state is kept, control state (call stack, open upvalues) unwinds. Thexpcallmessage handler now also observes those mutations, matching PUC-Lua's handler semantics.
Known issues
- Deep recursion is ~25% slower than rc.0. Carried forward from
rc.1: the configurable call-depth limit (#283) adds per-call
bookkeeping that recursion-dense workloads (e.g. naive
fib(30)) pay in full. Workloads that do real work per call are unaffected or faster. This remains a deliberate safety/speed tradeoff for the RC and will be addressed before1.0.0final.
[v1.0.0-rc.1] - 2026-06-02
The second release candidate for 1.0.0. It builds on rc.0 with a new
os and utf8 standard library, richer debug and error introspection,
a sizeable string.format and table performance pass, and a batch of
correctness fixes around upvalue lifetimes and Lua 5.3 semantics. The
public API is unchanged from rc.0.
Performance
A focused pass on the two hottest stdlib areas and the VM dispatcher. Numbers are pre-compiled-chunk throughput vs. rc.0, full Benchee runs on the same machine (Luerl and PUC-Lua used as drift controls, ±3%):
string.formatrewritten around iolist accumulation — literal runs and padding are appended to an iolist instead of repeatedly concatenating binaries, format flags are parsed once and dispatched on the integer specifier, and float conversion goes throughio_lib.format(#299, #317, #319, #316). The literal-heavy path is +143% (now ~2.7× faster than Luerl); width-flagged specifiers +83%; many-specifier strings +26%.- Plain-table
table.sort/table.concatfast paths for the common array-like case, with batched write-back in the sort path (#299, #318).table.sortis +35% at n=1000;table.concat-based string building is +66%. - Expanded VM dispatcher coverage to closures, varargs, multiple
returns, numeric/generic loops,
selfmethod calls, concat, and table opcodes (#275, #277). Object-oriented method dispatch is +41% (now faster than Luerl). - Batched table-literal construction through
Table.put_many/2in a single pass instead of element-by-element (#321).
Added
osstandard library, sandboxed:time,clock,date,difftime,getenv(no-op), and friends, with host-affecting calls neutered (#289).utf8standard library, plus aligned integer-arithmetic error wording (#258).debugintrospection: upvalue name tracking withdebug.getupvalue/debug.setupvalue(#285), anddebug.getinfonow populatesname/namewhatfrom the call site (#290).- Position captures
()in patterns acrossfind/match/gmatch/gsub(#288). - Configurable maximum call depth to bound recursion (#283).
- Allocation-bomb DoS hardening with documented sandboxing limits (#305).
- Structured error data on runtime errors (#246); arithmetic and bitwise type errors now thread operand hints into the message (#270).
Changed
- Rendered errors now lead with the source location, and ANSI colour is gated on whether output is a TTY (#304).
- Broader Lua 5.3 official test suite coverage: triaged and promoted
literals/goto/events/nextvar, and narrowed the remaining skips (pm,gc,constructs) to precise sub-ranges (#251, #282, #287, #294, #295). - README rewritten for 1.0 positioning with a quickstart and tour (#298),
runnable embedding examples under
examples/(#300), and@spec/ type definitions across the public API (#301).
Fixed
- Open upvalues are now closed at the exit of
do,if,while,for, andrepeatblocks (#286, #303), and the caller'sopen_upvaluesare restored after a nested execution returns (#245). requireno longer leaks the loaded module'sopen_upvaluesmap back to the calling chunk. Loading a module whose body created closures over its own top-level locals could alias the caller's locals to stale inner upvalue cells, breaking real-world libraries (e.g.luassert.assertions,luassert.array,luassert.spy) that follow the patternlocal x = require(...)→ manylocal functiondefs →x:method(...). As a side effect,Lua.call_function/3(public API) now preserves the caller'sopen_upvaluesacross calls (#244).- Integer divide and modulo by zero now match PUC-Lua semantics (#292).
table.unpackrejects oversized ranges instead of attempting a huge allocation (#293).gsubvalidates its replacement string and value (#291).- A parenthesised call or vararg now adjusts to a single value (#278).
- Function-declaration head names resolve during scope analysis (#274).
obj:method(...)calls expand multiple values in the argument list (#248).require()converts dotted module names to path separators (#242).
Known issues
- Deep recursion is ~25% slower than rc.0. The configurable call-depth
limit (#283) adds per-call bookkeeping that recursion-dense workloads
(e.g. naive
fib(30)) pay in full. Workloads that do real work per call are unaffected or faster. This is a deliberate safety/speed tradeoff for the RC and will be addressed before1.0.0final.
[v1.0.0-rc.0] - 2026-05-26
This is the first release candidate for 1.0.0. The library has been
rewritten on a new Elixir-native Lua 5.3 virtual machine, and the public
API is intended to be stable. Please report any regressions before final.
Added
- New Elixir-native Lua 5.3 virtual machine: lexer, parser, compiler, and register-based executor, with no Erlang or C dependencies.
- Standard library:
string(includingstring.formatwidth/precision,string.pack/unpack/packsize, and the full pattern engine forfind/match/gmatch/gsub),table,math(includingmath.fmod),debug,iostubs (sandboxed),os(sandboxed),package/require. _Gglobal table and Lua 5.3_ENVsemantics for global access.- Full metamethod dispatch:
__index,__newindex,__call, plus the arithmetic, comparison (including~=via__eqand<=/>=falling back through__lt), length, concat, andtostringmetamethods. - Varargs (
...), multiple returns, genericfor,goto/label,break, protected calls (pcall,xpcall). userdatasupport for passing arbitrary Elixir terms across the boundary.- Beautiful Lua-style stack traces and error messages with source line
tracking. Every runtime error carries line and source info (#214, #215),
and
attempt to call/attempt to indexerrors name the offending callee/target (#228). Inspectprotocol support for VM values returned across theLua.eval!/2boundary via display structs for tables, closures, userdata, and native functions (#218).- Mix tasks:
mix lua.eval,mix lua.suite,mix lua.bench(#220). - Lua 5.3 official test suite integration with per-file rationale for
suite files that are deferred as intentional non-goals (
main.lua,files.lua,attrib.lua,verybig.lua— shell-out, file I/O, and filesystemrequiresemantics that conflict with a sandboxed embedded VM) (#216). - Benchmark harness comparing against Luerl and PUC-Lua, with quick mode
and multi-
ninputs (#230) and asetup_luaport.shhelper (#225).
Changed
- VM backend: Luerl is no longer a runtime dependency. The library now
runs on its own Elixir-native VM. Luerl is kept only as a
:benchmark-env dependency for performance comparison. - Encoded value tags now use the new VM's internal representation:
{:tref, integer()}for tables (replacing:luerl.tref()),{:udref, integer()}for userdata (replacing:luerl.usdref()),{:native_func, fun}for Elixir-defined Lua callables (replacing:luerl.erl_func()), and{:lua_closure, _, _}for compiled Lua functions. - Parser error messages have a new format. The old Luerl-style
"Line 1: syntax error before: ';'"is now produced by the new parser (e.g."Expected expression"); user-visible string contents differ. - Chunks no longer require a separate "load" step —
Lua.Chunknow holds a compiled prototype and is reusable acrossLua.eval!/2calls. - 64-bit integer arithmetic and bitwise ops wrap on overflow per Lua 5.3 §3.4.1, instead of widening to bignums (Luerl's behaviour).
Lua.RuntimeExceptionandLua.CompilerExceptionare now publicly documented; user code can pattern-match and rescue them.
Removed
- The
{module(), atom(), list()}MFA encoding form is no longer accepted byLua.encode!/2. Use a function literal or adefluacallback instead.
Performance
- Right-size register tuple allocations (#153).
- O(N²) → O(N) upvalue collection in the closure handler (#154).
- O(1) upvalue access by storing upvalues as a tuple (#155).
- Fully tail-recursive CPS executor with line tracking moved off the heap (#156).
- Fast-path the executor dispatch loop (#223).
- Fast-path
Numeric.to_signed_int64for in-range integers (#227).
Fixed
- 64-bit integer overflow wrapping for arithmetic and bitwise ops (#177).
- Empty/missing-key table reads now return
nilper Lua 5.3 §3.4.11 (#179, #200). - Long-string
[[ ... ]]lexer handles embedded]and bracket levels like[==[ ... ]==], includingmain.lua-style headers (#180). - Comment tokens no longer leak past the lexer in expression lists (#182).
- Stdlib modules are pre-populated in
package.loadedsorequire"io"resolves (#184); module sentinel is set before executing required modules (#191). - For-loop variable now binds per statement, fixing register reuse (#195).
- Closure-handler crash on missing upvalue cells in
get_open_upvalueandset_open_upvalue(#196). _ENVsemantics for global variable access (#197).- Hex literal and string coercion in bitwise ops (#198);
math.fmodimplemented forbitwise.luaverification (#199). - Function declaration assigned to in-scope local rather than shadowing it (#185).
- Multi-return expansion no longer overflows the register tuple (#189).
- Pattern engine threads VM state through
gsubcallbacks and preserves capture order (#188, #190). - Atom values encode to strings (#158).
- Files containing only comments load successfully.
- Unicode characters supported in Lua scripts.
pairssurvives mid-iteration deletion by tracking dead keys (#202).- Metamethod closures receive operands through varargs (#203).
- Float division by zero yields ±
math.hugeinstead of raising (#204);//and%with a float-zero divisor returninf/nan(#211). - Lexer treats vertical tab and form feed as whitespace (#206).
- Table-library functions (
insert,remove,concat, etc.) honor__index,__newindex, and__len(#208). - Numeric
forcoerces string control values per Lua 5.3 §3.3.5 (#209). iois now exposed as a table of sandboxed stubs (#210).~=routes through the__eqmetamethod (#212);<=/>=fall back through__ltper Lua 5.3 §3.4.4 (#213).- Parser threads position info through bare-expression and unexpected-end errors (#222).
- Internal Lua VM frames pruned from
Lua.RuntimeExceptionstacks by default; opt back in withLua.new(debug: true)(#221). - Line number attribution for the first line of a chunk (#240).
string.packno longer emits compile warnings (#224).
[v0.4.0] - 2025-12-06
Changed
- Upgrade to Luerl 1.5.1
Fixed
- Warnings on Elixir 1.19
[v0.3.0] - 2025-06-09
Added
- Guards for encoded Lua values in
defluafunctionsis_table/1is_userdata/1is_lua_func/1is_erl_func/1is_mfa/1
Fixed
defluafunction can now specify guards when using or not using state
[v0.2.1] - 2025-05-14
Added
Lua.encode_list!/2andLua.decode_list!/2for encoding and decoding function arguments and return values
Fixed
- Ensure that list return values are properly encoded
[v0.2.0] - 2025-05-14
Changed
- Any data returned from a
defluafunction, or a function set byLua.set!/3is now validated. If the data is not an identity value, or an encoded value, it will raise an exception. In the past,Luaand Luerl would happily accept bad values, causing downstream problems in the program. This led to unexpected behavior, where depending on if the data passed was decoded or not, the program would succeed or fail.
[v0.1.1] - 2025-05-13
Added
Lua.put_private/3,Lua.get_private/2,Lua.get_private!/2, andLua.delete_private/2for working with private state
[v0.1.0] - 2025-05-12
Fixed
- Errors now correctly propagate state updates
- Fixed version requirements issues, causing references to undefined
luerl_new - Allow Unicode characters to be used in Lua scripts
- Files with only comments can be loaded
Changed
- Upgrade to Luerl 1.4.1
- Tables must now be explicitly decoded when receiving as arguments
defluaand other Elixir callbacks