summary refs log tree commit diff stats
path: root/src/cratera/bootstrap.lua
diff options
context:
space:
mode:
authorSoniEx2 <endermoneymod@gmail.com>2024-05-29 23:00:53 -0300
committerSoniEx2 <endermoneymod@gmail.com>2024-05-29 23:00:53 -0300
commit5eecbd6ef771a54b455d96f4033212062d7c3f8f (patch)
tree9fef3d210df849e82baa763b8b2b885c4806ca96 /src/cratera/bootstrap.lua
parentbbb9f04eecc66a9e8e208cc4abc0466697807765 (diff)
Implement cratera bootstrap
Diffstat (limited to 'src/cratera/bootstrap.lua')
-rw-r--r--src/cratera/bootstrap.lua258
1 files changed, 258 insertions, 0 deletions
diff --git a/src/cratera/bootstrap.lua b/src/cratera/bootstrap.lua
new file mode 100644
index 0000000..88cef17
--- /dev/null
+++ b/src/cratera/bootstrap.lua
@@ -0,0 +1,258 @@
+--[[
+    Cratera Bootstrap
+    Copyright (C) 2024  Soni L.
+
+    This program is free software: you can redistribute it and/or modify
+    it under the terms of the GNU Affero General Public License as published by
+    the Free Software Foundation, either version 3 of the License, or
+    (at your option) any later version.
+
+    This program is distributed in the hope that it will be useful,
+    but WITHOUT ANY WARRANTY; without even the implied warranty of
+    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+    GNU Affero General Public License for more details.
+
+    You should have received a copy of the GNU Affero General Public License
+    along with this program.  If not, see <https://www.gnu.org/licenses/>.
+--]]
+
+--[[
+    Parts of this file are based on PUC-Rio Lua, available under the following
+    license:
+
+    * Copyright (C) 1994-2018 Lua.org, PUC-Rio.
+    *
+    * Permission is hereby granted, free of charge, to any person obtaining
+    * a copy of this software and associated documentation files (the
+    * "Software"), to deal in the Software without restriction, including
+    * without limitation the rights to use, copy, modify, merge, publish,
+    * distribute, sublicense, and/or sell copies of the Software, and to
+    * permit persons to whom the Software is furnished to do so, subject to
+    * the following conditions:
+    *
+    * The above copyright notice and this permission notice shall be
+    * included in all copies or substantial portions of the Software.
+    *
+    * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+    * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+    * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
+    * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
+    * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
+    * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
+    * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+--]]
+
+-- The Cratera Bootstrap library sets up Cratera "proper", not just the
+-- compiler. However, it also expects a vanilla(-compatible) Lua environment.
+-- In particular, it uses the io library to access the filesystem.
+-- It also expects _G to be the default environment.
+
+-- NOTE: file:read() has the a-bit-annoying property of resetting EOF and other
+-- errors every time it's called, so we must pass (n, 0) if we want a reliable
+-- EOF indicator (which we do).
+
+local cratera = require "cratera"
+
+local cratera_env = setmetatable({lua=_G}, {__index=_G})
+cratera_env._G = cratera_env
+-- cratera.lib actually injects itself into *us*!
+package.loaded[...] = cratera_env
+
+local io_open = io.open
+local stdin = io.stdin
+-- only use setfenv if load doesn't support env argument
+local setfenv = (loadstring or load)("return setfenv", nil, nil, {})()
+-- we don't know the real BUFSIZ, use a common memory page size (16K, as on M1)
+local BUFSIZ = 16384
+
+local cratera_loader_base = cratera.load
+-- it is convenient to set cratera_env.load
+-- it will be replaced with the proper function in cratera.lib
+cratera_env.load = cratera_loader_base
+
+-- duplicated with init.lua
+local LUA_SIGNATURE
+if string.dump then
+    LUA_SIGNATURE = string.dump(function() end):sub(1,1)
+else
+    LUA_SIGNATURE = string.char(27)
+end
+
+-- based on lauxlib.c, PUC Rio
+local function skip_interpreter_line(file)
+    -- lua skips UTF-8 BOM, we do not.
+    -- join us, reject the BOM.
+    local initial_buf = ""
+    -- see NOTE at the start of this file
+    local cdata, err_or_more_data = file:read(1, 0)
+    if cdata == "#" and err_or_more_data then
+        -- skip interpreter line
+        repeat
+            cdata, err_or_more_data = file:read(1, 0)
+        until cdata == "\n" or not cdata or not err_or_more_data
+        -- align line numbers so error messages are correct!
+        initial_buf = initial_buf .. "\n"
+        if cdata and err_or_more_data then
+            cdata, err_or_more_data = file:read(1, 0)
+        end
+    end
+    return initial_buf, cdata, err_or_more_data
+end
+
+local function cratera_loadfile(filename, mode, env)
+    if env == nil then
+        -- load default env - if it doesn't exist, load lua env
+        env = cratera_env
+    end
+    local file, err
+    if filename ~= nil then
+        file, err = io_open(filename, "r")
+        if not file then
+            return nil, ("cannot open %s: %s"):format(filename, err)
+        end
+    else
+        file = stdin
+    end
+    local initial_buf, cdata, err_or_more_data = skip_interpreter_line(file)
+    if cdata == LUA_SIGNATURE and err_or_more_data and filename then
+        -- binary file
+        file:close()
+        file, err = io_open(filename, "rb")
+        if not file then
+            return nil, ("cannot reopen %s: %s"):format(filename, err)
+        end
+        initial_buf, cdata, err_or_more_data = skip_interpreter_line(file)
+    end
+    if cdata then
+        initial_buf = initial_buf .. cdata
+    end
+    -- must declare and set separately
+    -- this is a bit involved. importantly: the chunk (passed to load()) always
+    -- tail calls into the "reader", however we redefine "reader" based on
+    -- various condtions (on first call, on eof, on error)
+    local reader
+    reader = function()
+        reader = function()
+            local buf
+            buf, err_or_more_data = file:read(BUFSIZ, 0)
+            if not buf or not err_or_more_data then
+                reader = function() end
+            end
+            if buf == "" then return reader() end
+            return buf
+        end
+        if not err_or_more_data then
+            reader = function() end
+        end
+        if initial_buf == "" then return reader() end
+        return initial_buf
+    end
+    local chunkname
+    if not filename then
+        chunkname = "=stdin"
+    else
+        chunkname = "@" .. filename
+    end
+    local func, loader_err = cratera_loader_base(function()
+        return reader()
+    end, filename, mode, env)
+    local closeok, closeerr = file:close()
+    -- at this point, err_or_more_data *should* be nil.
+    if err_or_more_data then
+        -- unless it's an error
+        return nil, ("cannot read %s: %s"):format(filename, err_or_more_data)
+    end
+    -- lua doesn't quite check this properly but it's possible for an error to
+    -- be delayed until close
+    if not closeok and closeerr then
+        return nil, ("cannot read %s: %s"):format(filename, closeerr)
+    end
+    if setfenv and func and env then
+        setfenv(func, env)
+    end
+    -- arity adjustments
+    if func then
+        return func
+    else
+        return func, loader_err
+    end
+end
+
+cratera_env.loadfile = cratera_loadfile
+
+local searchers = package.searchers or package.loaders
+local dirsep, pathsep, pathmark = package.config:match("^([^\n]+)\n([^\n]+)\n([^\n]+)")
+local CRATERA_SEP = ".cratera.d" .. dirsep
+local tconcat = table.concat
+do
+    -- discover the cratera path
+    local default_path = require "cratera.prebuilt.path"
+    local cratera_path_env = os.getenv("CRATERA_PATH")
+
+    if cratera_path_env then
+        local default_mark = cratera_path_env:find(pathsep..pathsep, 1, true)
+        if not default_mark then
+            cratera_env.craterapath = cratera_path_env
+        else
+            local left = cratera_path_env:sub(1, default_mark - 1)
+            local right = cratera_path_env:sub(default_mark + 2)
+            if #left > 0 and #right > 0 then
+                cratera_env.craterapath = left .. ";" .. default_path .. ";" .. right
+            elseif #left > 0 then
+                cratera_env.craterapath = left .. ";" .. default_path
+            elseif #right > 0 then
+                cratera_env.craterapath = default_path .. ";" .. right
+            else
+                cratera_env.craterapath = default_path
+            end
+        end
+    else
+        cratera_env.craterapath = default_path
+    end
+end
+
+local error = error
+local type = type
+local assert = assert
+
+local psearchpath = package.searchpath or (function()
+    assert(#pathsep == 1, "unsupported lua configuration")
+    assert(#pathmark == 1, "unsupported lua configuration")
+    local pathfinder = "[^"..pathsep.."]*"
+    return function(name, path, sep, rep)
+        assert(sep == ".")
+        -- table trick is safe to use with "untrusted" input (e.g. dirsep)
+        local filename = name:gsub("%.", {["."]=rep})
+        -- this is a bit inefficient but it's the best we can do (safely)...
+        local filepaths = path:gsub(".", {[pathmark]=filename})
+        -- this is safe given the previous assertions
+        for filepath in filepaths:gmatch(pathfinder) do
+            local file = io_open(filepath)
+            if file then
+                file:close()
+                return filepath
+            end
+        end
+        return nil, "no file '" .. filepaths:gsub(".", {[pathsep]="\n\tno file'"}) .. "'"
+    end
+end)()
+
+-- put cratera searcher first, before lua searcher
+table.insert(searchers, 2, function(modname)
+    if type(cratera_env.craterapath) ~= "string" then
+        error("craterapath must be a string")
+    end
+    local filename, err = psearchpath(modname, cratera_env.craterapath, ".", CRATERA_SEP)
+    if not filename then
+        return err
+    end
+    local mod, err = cratera_loadfile(filename)
+    if mod then
+        return mod, filename
+    else
+        error(("error loading module '%s' from file '%s':\n\t%s"):format(modname, filename, err))
+    end
+end)
+
+-- load the cratera lib.
+require "cratera.lib"