Files
jsix_import/src/user/srv.init/initfs.cpp
Justin C. Miller eb62588b79 [6s] Allow 6s to know about filesystems
Added a j6_proto_vfs_tag/_get_tag pair of messages to the VFS protocol
to allow filesystems to label themselves, and gave 6s the concept of
current fs and current working directory.
2024-04-30 22:23:04 -07:00

111 lines
2.8 KiB
C++

#include <stdint.h>
#include <j6/flags.h>
#include <j6/errors.h>
#include <j6/memutils.h>
#include <j6/protocols/vfs.h>
#include <j6/syscalls.h>
#include <j6/syslog.hh>
#include "j6romfs.h"
#include "initfs.h"
static uint64_t initfs_running = 1;
static constexpr size_t buffer_size = 2048;
static char fs_tag[] = "init";
static constexpr size_t fs_tag_len = sizeof(fs_tag) - 1;
j6_status_t
handle_load_request(j6romfs::fs &fs, const char *path, j6_handle_t &vma)
{
const j6romfs::inode *in = fs.lookup_inode(path);
if (!in) {
vma = j6_handle_invalid;
return j6_status_ok;
}
uintptr_t load_addr = 0;
j6_vma_create_map(&vma, in->size, &load_addr, j6_vm_flag_write);
util::buffer dest = util::buffer::from(load_addr, in->size);
fs.load_inode_data(in, dest);
j6_vma_unmap(vma, 0);
return j6_status_ok;
}
void
sanitize(char *s, size_t len)
{
if (!s) return;
if (len >= buffer_size)
len = buffer_size - 1;
for (size_t i = 0; i < len; ++i)
if (!s[i]) return;
s[len] = 0;
}
void
initfs_start(j6romfs::fs &fs, j6_handle_t mb)
{
uint64_t tag = 0;
char *buffer = new char [buffer_size];
size_t out_len = 0;
uint64_t reply_tag = 0;
static constexpr size_t max_handles = 1;
size_t handles_count = 0;
j6_handle_t give_handle = j6_handle_invalid;
uint64_t proto_id;
j6_status_t s = j6_mailbox_respond(mb, &tag,
buffer, &out_len, buffer_size,
&give_handle, &handles_count, max_handles,
&reply_tag, j6_flag_block);
while (initfs_running) {
if (s != j6_status_ok) {
j6::syslog(j6::logs::srv, j6::log_level::error, "initfs: error in j6_mailbox_respond: %x", s);
return;
}
switch (tag) {
case j6_proto_vfs_load:
sanitize(buffer, out_len);
s = handle_load_request(fs, buffer, give_handle);
if (s != j6_status_ok) {
tag = j6_proto_base_status;
*reinterpret_cast<j6_status_t*>(buffer) = s;
out_len = sizeof(j6_status_t);
break;
}
handles_count = 1;
tag = j6_proto_vfs_file;
break;
case j6_proto_vfs_get_tag:
out_len = fs_tag_len;
handles_count = 0;
tag = j6_proto_vfs_tag;
memcpy(buffer, fs_tag, fs_tag_len);
break;
default:
tag = j6_proto_base_status;
*reinterpret_cast<j6_status_t*>(buffer) = j6_err_invalid_arg;
out_len = sizeof(j6_status_t);
give_handle = j6_handle_invalid;
handles_count = 0;
}
s = j6_mailbox_respond(mb, &tag,
buffer, &out_len, buffer_size,
&give_handle, &handles_count, max_handles,
&reply_tag, j6_flag_block);
}
}