2023-02-03 21:18:52 +00:00
|
|
|
#include "fs/VFS.h"
|
2023-03-10 21:18:48 +00:00
|
|
|
#include "Log.h"
|
2023-04-08 11:12:49 +00:00
|
|
|
#include "thread/Thread.h"
|
|
|
|
#include <bits/modes.h>
|
2023-03-10 21:18:48 +00:00
|
|
|
#include <luna/PathParser.h>
|
2023-02-03 21:18:52 +00:00
|
|
|
|
|
|
|
namespace VFS
|
|
|
|
{
|
|
|
|
SharedPtr<FileSystem> root_fs;
|
2023-02-25 17:05:25 +00:00
|
|
|
|
|
|
|
Inode& root_inode()
|
|
|
|
{
|
2023-03-10 21:18:48 +00:00
|
|
|
return *root_fs->root_inode();
|
|
|
|
}
|
|
|
|
|
2023-04-08 11:12:49 +00:00
|
|
|
Result<SharedPtr<Inode>> resolve_path(const char* path, Credentials auth)
|
2023-03-10 21:18:48 +00:00
|
|
|
{
|
|
|
|
auto parser = TRY(PathParser::create(path));
|
|
|
|
|
2023-03-11 00:13:44 +00:00
|
|
|
SharedPtr<Inode> current_inode = root_fs->root_inode();
|
2023-03-10 21:18:48 +00:00
|
|
|
|
2023-03-11 00:13:44 +00:00
|
|
|
// FIXME: Properly handle relative paths.
|
2023-03-10 21:18:48 +00:00
|
|
|
|
|
|
|
const char* section;
|
2023-04-08 11:12:49 +00:00
|
|
|
while (parser.next().try_set_value(section))
|
|
|
|
{
|
|
|
|
if (!can_execute(current_inode, auth)) return err(EACCES);
|
|
|
|
current_inode = TRY(current_inode->find(section));
|
|
|
|
}
|
2023-03-10 21:18:48 +00:00
|
|
|
|
|
|
|
return current_inode;
|
2023-02-25 17:05:25 +00:00
|
|
|
}
|
2023-03-11 00:13:44 +00:00
|
|
|
|
2023-04-08 11:12:49 +00:00
|
|
|
Result<SharedPtr<Inode>> create_directory(const char* path, Credentials auth)
|
2023-03-11 00:13:44 +00:00
|
|
|
{
|
|
|
|
auto parser = TRY(PathParser::create(path));
|
|
|
|
auto parent_path = TRY(parser.dirname());
|
|
|
|
|
2023-04-08 11:12:49 +00:00
|
|
|
auto parent_inode = TRY(resolve_path(parent_path.chars(), auth));
|
|
|
|
|
|
|
|
if (!can_write(parent_inode, auth)) return err(EACCES);
|
2023-03-11 00:13:44 +00:00
|
|
|
|
|
|
|
auto child_name = TRY(parser.basename());
|
|
|
|
|
2023-03-11 09:23:46 +00:00
|
|
|
// kdbgln("vfs: creating directory '%s' in parent '%s'", child_name.chars(), parent_path.chars());
|
2023-03-11 00:13:44 +00:00
|
|
|
|
|
|
|
return parent_inode->create_subdirectory(child_name.chars());
|
|
|
|
}
|
2023-03-11 09:23:46 +00:00
|
|
|
|
2023-04-08 11:12:49 +00:00
|
|
|
Result<SharedPtr<Inode>> create_file(const char* path, Credentials auth)
|
2023-03-11 09:23:46 +00:00
|
|
|
{
|
|
|
|
auto parser = TRY(PathParser::create(path));
|
|
|
|
auto parent_path = TRY(parser.dirname());
|
|
|
|
|
2023-04-08 11:12:49 +00:00
|
|
|
auto parent_inode = TRY(resolve_path(parent_path.chars(), auth));
|
|
|
|
|
|
|
|
if (!can_write(parent_inode, auth)) return err(EACCES);
|
2023-03-11 09:23:46 +00:00
|
|
|
|
|
|
|
auto child_name = TRY(parser.basename());
|
|
|
|
|
|
|
|
// kdbgln("vfs: creating file '%s' in parent '%s'", child_name.chars(), parent_path.chars());
|
|
|
|
|
|
|
|
return parent_inode->create_file(child_name.chars());
|
|
|
|
}
|
2023-04-08 11:12:49 +00:00
|
|
|
|
|
|
|
bool can_execute(SharedPtr<Inode> inode, Credentials auth)
|
|
|
|
{
|
|
|
|
if (inode->uid() == auth.euid) { return inode->mode() & S_IXUSR; }
|
|
|
|
if (inode->gid() == auth.egid) { return inode->mode() & S_IXGRP; }
|
|
|
|
|
|
|
|
return inode->mode() & S_IXOTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool can_write(SharedPtr<Inode> inode, Credentials auth)
|
|
|
|
{
|
|
|
|
if (inode->uid() == auth.euid) { return inode->mode() & S_IWUSR; }
|
|
|
|
if (inode->gid() == auth.egid) { return inode->mode() & S_IWGRP; }
|
|
|
|
|
|
|
|
return inode->mode() & S_IWOTH;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool can_read(SharedPtr<Inode> inode, Credentials auth)
|
|
|
|
{
|
|
|
|
if (inode->uid() == auth.euid) { return inode->mode() & S_IRUSR; }
|
|
|
|
if (inode->gid() == auth.egid) { return inode->mode() & S_IRGRP; }
|
|
|
|
|
|
|
|
return inode->mode() & S_IROTH;
|
|
|
|
}
|
2023-02-03 21:18:52 +00:00
|
|
|
}
|