summaryrefslogtreecommitdiff
path: root/src/tangara/lua/lua_filesystem.cpp
diff options
context:
space:
mode:
authorjacqueline <me@jacqueline.id.au>2024-05-10 14:28:03 +1000
committerjacqueline <me@jacqueline.id.au>2024-05-10 14:28:03 +1000
commit41426151a1cb9e7bf27fae4d1ee2dbba7553e3fa (patch)
treeda28cf369c04a0d39977de6b23fb100d43b6ba92 /src/tangara/lua/lua_filesystem.cpp
parent1d0ad4cbf9bd7e9a9c3deabed31342a4b10cac4b (diff)
parent35c6125b25f3f0c1852c74f9d165a46282494a7c (diff)
downloadtangara-fw-41426151a1cb9e7bf27fae4d1ee2dbba7553e3fa.tar.gz
Merge branch 'main' of codeberg.org:cool-tech-zone/tangara-fw
Diffstat (limited to 'src/tangara/lua/lua_filesystem.cpp')
-rw-r--r--src/tangara/lua/lua_filesystem.cpp179
1 files changed, 179 insertions, 0 deletions
diff --git a/src/tangara/lua/lua_filesystem.cpp b/src/tangara/lua/lua_filesystem.cpp
new file mode 100644
index 00000000..de51f555
--- /dev/null
+++ b/src/tangara/lua/lua_filesystem.cpp
@@ -0,0 +1,179 @@
+/*
+ * Copyright 2023 jacqueline <me@jacqueline.id.au>
+ *
+ * SPDX-License-Identifier: GPL-3.0-only
+ */
+
+#include "lua/lua_filesystem.hpp"
+#include <string>
+#include <cstring>
+#include "lauxlib.h"
+
+namespace lua {
+
+[[maybe_unused]] static constexpr char kTag[] = "lua_fs";
+
+static constexpr char kFileEntryMetatable[] = "fs_file_entry";
+static constexpr char kFileIteratorMetatable[] = "fs_iterator";
+
+// Todo: Use std::pmr::string for paths/dirs
+struct LuaFileEntry {
+ bool isHidden;
+ bool isDirectory;
+ bool isTrack;
+ size_t path_size;
+ char path[];
+};
+
+static_assert(std::is_trivially_destructible<LuaFileEntry>());
+static_assert(std::is_trivially_copy_assignable<LuaFileEntry>());
+
+static auto push_lua_file_entry(lua_State* L, const lua::FileEntry& r) -> void {
+ // Create and init the userdata.
+ LuaFileEntry* file_entry = reinterpret_cast<LuaFileEntry*>(
+ lua_newuserdata(L, sizeof(LuaFileEntry) + r.filepath.size()));
+ luaL_setmetatable(L, kFileEntryMetatable);
+
+ // Init all the fields
+ *file_entry = {
+ .isHidden = r.isHidden,
+ .isDirectory = r.isDirectory,
+ .isTrack = r.isTrack,
+ .path_size = r.filepath.size(),
+ };
+
+ // Copy the string data across.
+ std::memcpy(file_entry->path, r.filepath.data(), r.filepath.size());
+}
+
+auto check_file_iterator(lua_State* L, int stack_pos) -> lua::FileIterator* {
+ lua::FileIterator* it = *reinterpret_cast<lua::FileIterator**>(
+ luaL_checkudata(L, stack_pos, kFileIteratorMetatable));
+ return it;
+}
+
+static auto push_iterator(lua_State* state, const lua::FileIterator& it)
+ -> void {
+ lua::FileIterator** data = reinterpret_cast<lua::FileIterator**>(
+ lua_newuserdata(state, sizeof(uintptr_t)));
+ *data = new lua::FileIterator(it); // TODO...
+ luaL_setmetatable(state, kFileIteratorMetatable);
+}
+
+static auto fs_iterate_prev(lua_State* state) -> int {
+ lua::FileIterator* it = check_file_iterator(state, 1);
+ it->prev();
+ std::optional<lua::FileEntry> res = it->value();
+
+ if (res) {
+ push_lua_file_entry(state, *res);
+ } else {
+ lua_pushnil(state);
+ }
+
+ return 1;
+}
+
+static auto fs_iterate(lua_State* state) -> int {
+ lua::FileIterator* it = check_file_iterator(state, 1);
+ it->next();
+ std::optional<lua::FileEntry> res = it->value();
+
+ if (res) {
+ push_lua_file_entry(state, *res);
+ } else {
+ lua_pushnil(state);
+ }
+
+ return 1;
+}
+
+static auto fs_iterator_clone(lua_State* state) -> int {
+ lua::FileIterator* it = check_file_iterator(state, 1);
+ push_iterator(state, *it);
+ return 1;
+}
+
+static auto fs_iterator_gc(lua_State* state) -> int {
+ lua::FileIterator* it = check_file_iterator(state, 1);
+ delete it;
+ return 0;
+}
+
+static const struct luaL_Reg kFileIteratorFuncs[] = {{"next", fs_iterate},
+ {"prev", fs_iterate_prev},
+ {"clone", fs_iterator_clone},
+ {"__call", fs_iterate},
+ {"__gc", fs_iterator_gc},
+ {NULL, NULL}};
+
+static auto file_entry_path(lua_State* state) -> int {
+ LuaFileEntry* data = reinterpret_cast<LuaFileEntry*>(
+ luaL_checkudata(state, 1, kFileEntryMetatable));
+ lua_pushlstring(state, data->path, data->path_size);
+ return 1;
+}
+
+static auto file_entry_is_dir(lua_State* state) -> int {
+ LuaFileEntry* data = reinterpret_cast<LuaFileEntry*>(
+ luaL_checkudata(state, 1, kFileEntryMetatable));
+ lua_pushboolean(state, data->isDirectory);
+ return 1;
+}
+
+static auto file_entry_is_hidden(lua_State* state) -> int {
+ LuaFileEntry* data = reinterpret_cast<LuaFileEntry*>(
+ luaL_checkudata(state, 1, kFileEntryMetatable));
+ lua_pushboolean(state, data->isHidden);
+ return 1;
+}
+
+static auto file_entry_is_track(lua_State* state) -> int {
+ LuaFileEntry* data = reinterpret_cast<LuaFileEntry*>(
+ luaL_checkudata(state, 1, kFileEntryMetatable));
+ lua_pushboolean(state, data->isTrack);
+ return 1;
+}
+
+static const struct luaL_Reg kFileEntryFuncs[] = {{"filepath", file_entry_path},
+ {"is_directory", file_entry_is_dir},
+ {"is_hidden", file_entry_is_hidden},
+ {"is_track", file_entry_is_track},
+ {"__tostring", file_entry_path},
+ {NULL, NULL}};
+
+static auto fs_new_iterator(lua_State* state) -> int {
+ // Takes a filepath as a string and returns a new FileIterator
+ // on that directory
+ std::string filepath = luaL_checkstring(state, -1);
+ lua::FileIterator iter(filepath);
+ push_iterator(state, iter);
+ return 1;
+}
+
+static const struct luaL_Reg kFilesystemFuncs[] = {{"iterator", fs_new_iterator},
+ {NULL, NULL}};
+
+static auto lua_filesystem(lua_State* state) -> int {
+ luaL_newmetatable(state, kFileIteratorMetatable);
+ lua_pushliteral(state, "__index");
+ lua_pushvalue(state, -2);
+ lua_settable(state, -3); // metatable.__index = metatable
+ luaL_setfuncs(state, kFileIteratorFuncs, 0);
+
+ luaL_newmetatable(state, kFileEntryMetatable);
+ lua_pushliteral(state, "__index");
+ lua_pushvalue(state, -2);
+ lua_settable(state, -3); // metatable.__index = metatable
+ luaL_setfuncs(state, kFileEntryFuncs, 0);
+
+ luaL_newlib(state, kFilesystemFuncs);
+ return 1;
+}
+
+auto RegisterFileSystemModule(lua_State* s) -> void {
+ luaL_requiref(s, "filesystem", lua_filesystem, true);
+ lua_pop(s, 1);
+}
+
+} // namespace lua