diff options
| author | jacqueline <me@jacqueline.id.au> | 2024-05-10 14:28:03 +1000 |
|---|---|---|
| committer | jacqueline <me@jacqueline.id.au> | 2024-05-10 14:28:03 +1000 |
| commit | 41426151a1cb9e7bf27fae4d1ee2dbba7553e3fa (patch) | |
| tree | da28cf369c04a0d39977de6b23fb100d43b6ba92 /src/tangara/lua/lua_filesystem.cpp | |
| parent | 1d0ad4cbf9bd7e9a9c3deabed31342a4b10cac4b (diff) | |
| parent | 35c6125b25f3f0c1852c74f9d165a46282494a7c (diff) | |
| download | tangara-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.cpp | 179 |
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 |
