Luna/libos/src/FileSystem.cpp

107 lines
2.5 KiB
C++
Raw Normal View History

2023-04-13 16:33:04 +00:00
#include <luna/String.h>
#include <os/Directory.h>
2023-04-13 16:33:04 +00:00
#include <os/FileSystem.h>
#include <dirent.h>
2023-04-13 16:33:04 +00:00
#include <errno.h>
#include <fcntl.h>
2023-04-13 16:33:04 +00:00
#include <pwd.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/syscall.h>
#include <unistd.h>
namespace os::FileSystem
{
bool exists(StringView path)
{
struct stat st;
if (stat(path.chars(), &st) < 0) return false;
return true;
}
bool is_directory(StringView path)
{
struct stat st;
if (stat(path.chars(), &st) < 0) return false;
return S_ISDIR(st.st_mode);
}
Result<void> create_directory(StringView path, mode_t mode)
{
long rc = syscall(SYS_mkdir, path.chars(), mode);
return Result<void>::from_syscall(rc);
2023-04-13 16:33:04 +00:00
}
Result<void> remove(StringView path)
{
return removeat(AT_FDCWD, path);
}
Result<void> removeat(int dirfd, StringView path)
{
long rc = syscall(SYS_unlinkat, dirfd, path.chars(), 0);
2023-04-13 16:33:04 +00:00
return Result<void>::from_syscall(rc);
2023-04-13 16:33:04 +00:00
}
Result<void> remove_tree(StringView path)
{
return remove_tree_at(AT_FDCWD, path);
}
Result<void> remove_tree_at(int dirfd, StringView path)
{
auto rc = removeat(dirfd, path);
if (!rc.has_error()) return {};
if (rc.error() != ENOTEMPTY) return rc.release_error();
auto dir = TRY(os::Directory::openat(dirfd, path));
Vector<String> entries;
// FIXME: This is done because the kernel doesn't appreciate us deleting entries while iterating over
// directories. This means that we have to iterate first, then delete.
String ent;
while ((ent = TRY(dir->next(os::Directory::Filter::ParentAndBase))), !ent.is_empty())
{
TRY(entries.try_append(move(ent)));
}
for (const auto& entry : entries) { TRY(remove_tree_at(dir->fd(), entry.view())); }
return removeat(dirfd, path);
}
2023-04-13 16:33:04 +00:00
Result<String> working_directory()
{
char* ptr = getcwd(NULL, 0);
if (!ptr) return err(errno);
return String { ptr };
}
Result<String> home_directory()
{
char* home = getenv("HOME");
if (home) return String::from_cstring(home);
struct passwd* pw = getpwuid(getuid());
if (!pw) return err(ENOENT);
return String::from_cstring(pw->pw_dir);
}
2023-04-18 16:46:19 +00:00
Result<void> change_directory(StringView path)
{
long rc = syscall(SYS_chdir, path.chars());
return Result<void>::from_syscall(rc);
2023-04-18 16:46:19 +00:00
}
2023-04-13 16:33:04 +00:00
}