2023-04-12 19:46:10 +00:00
|
|
|
#include <fcntl.h>
|
2023-04-13 15:31:21 +00:00
|
|
|
#include <luna/StringBuilder.h>
|
2023-04-12 19:46:10 +00:00
|
|
|
#include <os/File.h>
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2023-04-13 15:31:21 +00:00
|
|
|
static SharedPtr<os::File> g_stdin = {};
|
|
|
|
static SharedPtr<os::File> g_stdout = {};
|
|
|
|
static SharedPtr<os::File> g_stderr = {};
|
|
|
|
|
2023-04-12 19:46:10 +00:00
|
|
|
namespace os
|
|
|
|
{
|
|
|
|
File::File(Badge<File>)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
File::~File()
|
|
|
|
{
|
|
|
|
if (m_fd >= 0) close(m_fd);
|
|
|
|
}
|
|
|
|
|
2023-04-13 15:31:21 +00:00
|
|
|
void File::initialize_standard_streams()
|
|
|
|
{
|
|
|
|
g_stdin = adopt_shared_if_nonnull(new (std::nothrow) File({}))
|
|
|
|
.expect_release_value("Cannot open standard input stream");
|
|
|
|
g_stdin->m_fd = STDIN_FILENO;
|
|
|
|
|
|
|
|
g_stdout = adopt_shared_if_nonnull(new (std::nothrow) File({}))
|
|
|
|
.expect_release_value("Cannot open standard output stream");
|
|
|
|
g_stdout->m_fd = STDOUT_FILENO;
|
|
|
|
|
|
|
|
g_stderr = adopt_shared_if_nonnull(new (std::nothrow) File({}))
|
|
|
|
.expect_release_value("Cannot open standard error stream");
|
|
|
|
g_stderr->m_fd = STDERR_FILENO;
|
|
|
|
}
|
|
|
|
|
|
|
|
SharedPtr<File> File::standard_input()
|
|
|
|
{
|
|
|
|
if (!g_stdin) initialize_standard_streams();
|
|
|
|
|
|
|
|
return g_stdin;
|
|
|
|
}
|
|
|
|
|
|
|
|
SharedPtr<File> File::standard_output()
|
|
|
|
{
|
|
|
|
if (!g_stdout) initialize_standard_streams();
|
|
|
|
|
|
|
|
return g_stdout;
|
|
|
|
}
|
|
|
|
|
|
|
|
SharedPtr<File> File::standard_error()
|
|
|
|
{
|
|
|
|
if (!g_stderr) initialize_standard_streams();
|
|
|
|
|
|
|
|
return g_stderr;
|
|
|
|
}
|
|
|
|
|
2023-04-12 19:46:10 +00:00
|
|
|
Result<SharedPtr<File>> File::construct(StringView path, int flags, mode_t mode)
|
|
|
|
{
|
2023-04-13 15:31:21 +00:00
|
|
|
auto file = TRY(adopt_shared_if_nonnull(new (std::nothrow) File({})));
|
2023-04-12 19:46:10 +00:00
|
|
|
|
2023-04-15 18:26:15 +00:00
|
|
|
long rc = syscall(SYS_openat, AT_FDCWD, path.chars(), flags, mode);
|
2023-04-12 19:46:10 +00:00
|
|
|
int fd = TRY(Result<int>::from_syscall(rc));
|
|
|
|
|
|
|
|
file->m_fd = fd;
|
|
|
|
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<SharedPtr<File>> File::open(StringView path, OpenMode flags)
|
|
|
|
{
|
|
|
|
return construct(path, (int)flags, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<SharedPtr<File>> File::open_or_create(StringView path, OpenMode flags, mode_t mode)
|
|
|
|
{
|
|
|
|
return construct(path, (int)flags | O_CREAT, mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<SharedPtr<File>> File::create(StringView path, OpenMode flags, mode_t mode)
|
|
|
|
{
|
|
|
|
return construct(path, (int)flags | (O_CREAT | O_EXCL), mode);
|
|
|
|
}
|
|
|
|
|
2023-04-26 18:41:03 +00:00
|
|
|
Result<SharedPtr<File>> File::open_input_file(StringView path)
|
|
|
|
{
|
|
|
|
if (path == "-"_sv) return standard_input();
|
|
|
|
|
|
|
|
return construct(path, O_RDONLY, 0);
|
|
|
|
}
|
|
|
|
|
2023-04-12 19:46:10 +00:00
|
|
|
Result<usize> File::raw_read(u8* buf, usize length)
|
|
|
|
{
|
|
|
|
long rc = syscall(SYS_read, m_fd, buf, length);
|
|
|
|
return Result<usize>::from_syscall(rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<usize> File::raw_write(const u8* buf, usize length)
|
|
|
|
{
|
|
|
|
long rc = syscall(SYS_write, m_fd, buf, length);
|
|
|
|
return Result<usize>::from_syscall(rc);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> File::write(StringView str)
|
|
|
|
{
|
|
|
|
TRY(raw_write((const u8*)str.chars(), str.length()));
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-04-18 14:41:17 +00:00
|
|
|
Result<void> File::write(const Buffer& buf)
|
|
|
|
{
|
|
|
|
TRY(raw_write(buf.data(), buf.size()));
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-04-22 13:21:04 +00:00
|
|
|
Result<String> File::read_line()
|
2023-04-13 15:31:21 +00:00
|
|
|
{
|
|
|
|
Vector<char> data;
|
|
|
|
|
|
|
|
int current;
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
current = TRY(getchar());
|
|
|
|
|
|
|
|
if (current == -1) break;
|
|
|
|
|
|
|
|
TRY(data.try_append((char)current));
|
|
|
|
|
2023-04-22 13:21:04 +00:00
|
|
|
if (current == '\n') break;
|
2023-04-13 15:31:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!data.size()) return String {};
|
|
|
|
|
|
|
|
TRY(data.try_append('\0'));
|
|
|
|
|
|
|
|
return String::from_cstring(data.data());
|
|
|
|
}
|
|
|
|
|
2023-04-26 18:34:09 +00:00
|
|
|
Result<String> File::read_all_as_string()
|
|
|
|
{
|
|
|
|
StringBuilder sb;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
auto line = TRY(read_line());
|
|
|
|
if (line.is_empty()) break;
|
|
|
|
TRY(sb.add(line));
|
|
|
|
}
|
|
|
|
|
|
|
|
return sb.string();
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<Buffer> File::read_all()
|
|
|
|
{
|
|
|
|
Vector<u8> data;
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
int c = TRY(getchar());
|
|
|
|
if (c == -1) break;
|
|
|
|
TRY(data.try_append((u8)c));
|
|
|
|
}
|
|
|
|
|
|
|
|
Buffer buf;
|
|
|
|
TRY(buf.append_data(data.data(), data.size()));
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
2023-04-18 14:41:17 +00:00
|
|
|
Result<void> File::read(Buffer& buf, usize size)
|
|
|
|
{
|
|
|
|
u8* slice = TRY(buf.slice(0, size));
|
|
|
|
|
|
|
|
usize nread = TRY(raw_read(slice, size));
|
|
|
|
|
|
|
|
TRY(buf.try_resize(nread));
|
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2023-04-13 15:31:21 +00:00
|
|
|
Result<int> File::getchar()
|
|
|
|
{
|
2023-04-26 18:34:09 +00:00
|
|
|
u8 value;
|
2023-04-13 15:31:21 +00:00
|
|
|
|
2023-04-26 18:34:09 +00:00
|
|
|
usize nread = TRY(raw_read(&value, 1));
|
2023-04-13 15:31:21 +00:00
|
|
|
if (!nread) return -1;
|
|
|
|
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
2023-04-12 19:46:10 +00:00
|
|
|
void File::set_close_on_exec()
|
|
|
|
{
|
|
|
|
fcntl(m_fd, F_SETFD, FD_CLOEXEC);
|
|
|
|
}
|
2023-05-01 17:31:15 +00:00
|
|
|
|
|
|
|
// FIXME: Do not allocate memory for printing.
|
|
|
|
Result<void> print_impl(SharedPtr<File> f, StringView fmt, va_list ap)
|
|
|
|
{
|
|
|
|
auto str = TRY(String::vformat(fmt, ap));
|
|
|
|
return f->write(str.view());
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> print(StringView fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
|
|
|
auto rc = print_impl(File::standard_output(), fmt, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> println(StringView fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
|
|
|
auto rc = print_impl(File::standard_output(), fmt, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
TRY(rc);
|
|
|
|
|
|
|
|
return File::standard_output()->write("\n"_sv);
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> eprint(StringView fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
|
|
|
auto rc = print_impl(File::standard_error(), fmt, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
Result<void> eprintln(StringView fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
|
|
|
|
|
|
auto rc = print_impl(File::standard_error(), fmt, ap);
|
|
|
|
|
|
|
|
va_end(ap);
|
|
|
|
|
|
|
|
TRY(rc);
|
|
|
|
|
|
|
|
return File::standard_error()->write("\n"_sv);
|
|
|
|
}
|
2023-04-12 19:46:10 +00:00
|
|
|
}
|