mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2024-11-30 03:20:28 +00:00
994 lines
28 KiB
C++
994 lines
28 KiB
C++
/*
|
|
* Copyright (c) 2020, Sergey Bugaev <bugaevc@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <Kernel/FileSystem/Plan9FileSystem.h>
|
|
#include <Kernel/Process.h>
|
|
|
|
namespace Kernel {
|
|
|
|
NonnullRefPtr<Plan9FS> Plan9FS::create(FileDescription& file_description)
|
|
{
|
|
return adopt_ref(*new Plan9FS(file_description));
|
|
}
|
|
|
|
Plan9FS::Plan9FS(FileDescription& file_description)
|
|
: FileBackedFileSystem(file_description)
|
|
, m_completion_blocker(*this)
|
|
{
|
|
}
|
|
|
|
Plan9FS::~Plan9FS()
|
|
{
|
|
// Make sure to destroy the root inode before the FS gets destroyed.
|
|
if (m_root_inode) {
|
|
VERIFY(m_root_inode->ref_count() == 1);
|
|
m_root_inode = nullptr;
|
|
}
|
|
}
|
|
|
|
class Plan9FS::Message {
|
|
public:
|
|
enum class Type : u8 {
|
|
// 9P2000.L
|
|
Tlerror = 6,
|
|
Rlerror = 7,
|
|
Tstatfs = 8,
|
|
Rstatfs = 9,
|
|
|
|
Tlopen = 12,
|
|
Rlopen = 13,
|
|
Tlcreate = 14,
|
|
Rlcreate = 15,
|
|
Tsymlink = 16,
|
|
Rsymlink = 17,
|
|
Tmknod = 18,
|
|
Rmknod = 19,
|
|
Trename = 20,
|
|
Rrename = 21,
|
|
Treadlink = 22,
|
|
Rreadlink = 23,
|
|
Tgetattr = 24,
|
|
Rgetattr = 25,
|
|
Tsetattr = 26,
|
|
Rsetattr = 27,
|
|
|
|
Txattrwalk = 30,
|
|
Rxattrwalk = 31,
|
|
Txattrcreate = 32,
|
|
Rxattrcreate = 33,
|
|
|
|
Treaddir = 40,
|
|
Rreaddir = 41,
|
|
|
|
Tfsync = 50,
|
|
Rfsync = 51,
|
|
Tlock = 52,
|
|
Rlock = 53,
|
|
Tgetlock = 54,
|
|
Rgetlock = 55,
|
|
|
|
Tlink = 70,
|
|
Rlink = 71,
|
|
Tmkdir = 72,
|
|
Rmkdir = 73,
|
|
Trenameat = 74,
|
|
Rrenameat = 75,
|
|
Tunlinkat = 76,
|
|
Runlinkat = 77,
|
|
|
|
// 9P2000
|
|
Tversion = 100,
|
|
Rversion = 101,
|
|
Tauth = 102,
|
|
Rauth = 103,
|
|
Tattach = 104,
|
|
Rattach = 105,
|
|
Terror = 106,
|
|
Rerror = 107,
|
|
Tflush = 108,
|
|
Rflush = 109,
|
|
Twalk = 110,
|
|
Rwalk = 111,
|
|
Topen = 112,
|
|
Ropen = 113,
|
|
Tcreate = 114,
|
|
Rcreate = 115,
|
|
Tread = 116,
|
|
Rread = 117,
|
|
Twrite = 118,
|
|
Rwrite = 119,
|
|
Tclunk = 120,
|
|
Rclunk = 121,
|
|
Tremove = 122,
|
|
Rremove = 123,
|
|
Tstat = 124,
|
|
Rstat = 125,
|
|
Twstat = 126,
|
|
Rwstat = 127
|
|
};
|
|
|
|
class Decoder {
|
|
public:
|
|
explicit Decoder(const StringView& data)
|
|
: m_data(data)
|
|
{
|
|
}
|
|
|
|
Decoder& operator>>(u8&);
|
|
Decoder& operator>>(u16&);
|
|
Decoder& operator>>(u32&);
|
|
Decoder& operator>>(u64&);
|
|
Decoder& operator>>(StringView&);
|
|
Decoder& operator>>(qid&);
|
|
StringView read_data();
|
|
|
|
bool has_more_data() const { return !m_data.is_empty(); }
|
|
|
|
private:
|
|
StringView m_data;
|
|
|
|
template<typename N>
|
|
Decoder& read_number(N& number)
|
|
{
|
|
VERIFY(sizeof(number) <= m_data.length());
|
|
memcpy(&number, m_data.characters_without_null_termination(), sizeof(number));
|
|
m_data = m_data.substring_view(sizeof(number), m_data.length() - sizeof(number));
|
|
return *this;
|
|
}
|
|
};
|
|
|
|
Message& operator<<(u8);
|
|
Message& operator<<(u16);
|
|
Message& operator<<(u32);
|
|
Message& operator<<(u64);
|
|
Message& operator<<(const StringView&);
|
|
void append_data(const StringView&);
|
|
|
|
template<typename T>
|
|
Message& operator>>(T& t)
|
|
{
|
|
VERIFY(m_have_been_built);
|
|
m_built.decoder >> t;
|
|
return *this;
|
|
}
|
|
|
|
StringView read_data()
|
|
{
|
|
VERIFY(m_have_been_built);
|
|
return m_built.decoder.read_data();
|
|
}
|
|
|
|
Type type() const { return m_type; }
|
|
u16 tag() const { return m_tag; }
|
|
|
|
Message(Plan9FS&, Type);
|
|
Message(NonnullOwnPtr<KBuffer>&&);
|
|
~Message();
|
|
Message& operator=(Message&&);
|
|
|
|
const KBuffer& build();
|
|
|
|
static constexpr size_t max_header_size = 24;
|
|
|
|
private:
|
|
template<typename N>
|
|
Message& append_number(N number)
|
|
{
|
|
VERIFY(!m_have_been_built);
|
|
m_builder.append(reinterpret_cast<const char*>(&number), sizeof(number));
|
|
return *this;
|
|
}
|
|
|
|
union {
|
|
KBufferBuilder m_builder;
|
|
struct {
|
|
NonnullOwnPtr<KBuffer> buffer;
|
|
Decoder decoder;
|
|
} m_built;
|
|
};
|
|
|
|
u16 m_tag { 0 };
|
|
Type m_type { 0 };
|
|
bool m_have_been_built { false };
|
|
};
|
|
|
|
bool Plan9FS::initialize()
|
|
{
|
|
ensure_thread();
|
|
|
|
Message version_message { *this, Message::Type::Tversion };
|
|
version_message << (u32)m_max_message_size << "9P2000.L";
|
|
|
|
auto result = post_message_and_wait_for_a_reply(version_message);
|
|
if (result.is_error())
|
|
return false;
|
|
|
|
u32 msize;
|
|
StringView remote_protocol_version;
|
|
version_message >> msize >> remote_protocol_version;
|
|
dbgln("Remote supports msize={} and protocol version {}", msize, remote_protocol_version);
|
|
m_remote_protocol_version = parse_protocol_version(remote_protocol_version);
|
|
m_max_message_size = min(m_max_message_size, (size_t)msize);
|
|
|
|
// TODO: auth
|
|
|
|
u32 root_fid = allocate_fid();
|
|
Message attach_message { *this, Message::Type::Tattach };
|
|
// FIXME: This needs a user name and an "export" name; but how do we get them?
|
|
// Perhaps initialize() should accept a string of FS-specific options...
|
|
attach_message << root_fid << (u32)-1 << "sergey"
|
|
<< "/";
|
|
if (m_remote_protocol_version >= ProtocolVersion::v9P2000u)
|
|
attach_message << (u32)-1;
|
|
|
|
result = post_message_and_wait_for_a_reply(attach_message);
|
|
if (result.is_error()) {
|
|
dbgln("Attaching failed");
|
|
return false;
|
|
}
|
|
|
|
m_root_inode = Plan9FSInode::create(*this, root_fid);
|
|
return true;
|
|
}
|
|
|
|
Plan9FS::ProtocolVersion Plan9FS::parse_protocol_version(const StringView& s) const
|
|
{
|
|
if (s == "9P2000.L")
|
|
return ProtocolVersion::v9P2000L;
|
|
if (s == "9P2000.u")
|
|
return ProtocolVersion::v9P2000u;
|
|
return ProtocolVersion::v9P2000;
|
|
}
|
|
|
|
NonnullRefPtr<Inode> Plan9FS::root_inode() const
|
|
{
|
|
return *m_root_inode;
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator<<(u8 number)
|
|
{
|
|
return append_number(number);
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator<<(u16 number)
|
|
{
|
|
return append_number(number);
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator<<(u32 number)
|
|
{
|
|
return append_number(number);
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator<<(u64 number)
|
|
{
|
|
return append_number(number);
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator<<(const StringView& string)
|
|
{
|
|
*this << static_cast<u16>(string.length());
|
|
m_builder.append(string);
|
|
return *this;
|
|
}
|
|
|
|
void Plan9FS::Message::append_data(const StringView& data)
|
|
{
|
|
*this << static_cast<u32>(data.length());
|
|
m_builder.append(data);
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u8& number)
|
|
{
|
|
return read_number(number);
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u16& number)
|
|
{
|
|
return read_number(number);
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u32& number)
|
|
{
|
|
return read_number(number);
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u64& number)
|
|
{
|
|
return read_number(number);
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(qid& qid)
|
|
{
|
|
return *this >> qid.type >> qid.version >> qid.path;
|
|
}
|
|
|
|
Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(StringView& string)
|
|
{
|
|
u16 length;
|
|
*this >> length;
|
|
VERIFY(length <= m_data.length());
|
|
string = m_data.substring_view(0, length);
|
|
m_data = m_data.substring_view_starting_after_substring(string);
|
|
return *this;
|
|
}
|
|
|
|
StringView Plan9FS::Message::Decoder::read_data()
|
|
{
|
|
u32 length;
|
|
*this >> length;
|
|
VERIFY(length <= m_data.length());
|
|
auto data = m_data.substring_view(0, length);
|
|
m_data = m_data.substring_view_starting_after_substring(data);
|
|
return data;
|
|
}
|
|
|
|
Plan9FS::Message::Message(Plan9FS& fs, Type type)
|
|
: m_builder()
|
|
, m_tag(fs.allocate_tag())
|
|
, m_type(type)
|
|
, m_have_been_built(false)
|
|
{
|
|
u32 size_placeholder = 0;
|
|
*this << size_placeholder << (u8)type << m_tag;
|
|
}
|
|
|
|
Plan9FS::Message::Message(NonnullOwnPtr<KBuffer>&& buffer)
|
|
: m_built { move(buffer), Decoder({ buffer->data(), buffer->size() }) }
|
|
, m_have_been_built(true)
|
|
{
|
|
u32 size;
|
|
u8 raw_type;
|
|
*this >> size >> raw_type >> m_tag;
|
|
m_type = (Type)raw_type;
|
|
}
|
|
|
|
Plan9FS::Message::~Message()
|
|
{
|
|
if (m_have_been_built) {
|
|
m_built.buffer.~NonnullOwnPtr<KBuffer>();
|
|
m_built.decoder.~Decoder();
|
|
} else {
|
|
m_builder.~KBufferBuilder();
|
|
}
|
|
}
|
|
|
|
Plan9FS::Message& Plan9FS::Message::operator=(Message&& message)
|
|
{
|
|
m_tag = message.m_tag;
|
|
m_type = message.m_type;
|
|
|
|
if (m_have_been_built) {
|
|
m_built.buffer.~NonnullOwnPtr<KBuffer>();
|
|
m_built.decoder.~Decoder();
|
|
} else {
|
|
m_builder.~KBufferBuilder();
|
|
}
|
|
|
|
m_have_been_built = message.m_have_been_built;
|
|
if (m_have_been_built) {
|
|
new (&m_built.buffer) NonnullOwnPtr<KBuffer>(move(message.m_built.buffer));
|
|
new (&m_built.decoder) Decoder(move(message.m_built.decoder));
|
|
} else {
|
|
new (&m_builder) KBufferBuilder(move(message.m_builder));
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
const KBuffer& Plan9FS::Message::build()
|
|
{
|
|
VERIFY(!m_have_been_built);
|
|
|
|
auto tmp_buffer = m_builder.build();
|
|
|
|
// FIXME: We should not assume success here.
|
|
VERIFY(tmp_buffer);
|
|
|
|
m_have_been_built = true;
|
|
m_builder.~KBufferBuilder();
|
|
|
|
new (&m_built.buffer) NonnullOwnPtr<KBuffer>(tmp_buffer.release_nonnull());
|
|
new (&m_built.decoder) Decoder({ m_built.buffer->data(), m_built.buffer->size() });
|
|
u32* size = reinterpret_cast<u32*>(m_built.buffer->data());
|
|
*size = m_built.buffer->size();
|
|
return *m_built.buffer;
|
|
}
|
|
|
|
Plan9FS::ReceiveCompletion::ReceiveCompletion(u16 tag)
|
|
: tag(tag)
|
|
{
|
|
}
|
|
|
|
Plan9FS::ReceiveCompletion::~ReceiveCompletion()
|
|
{
|
|
}
|
|
|
|
bool Plan9FS::Blocker::unblock(u16 tag)
|
|
{
|
|
{
|
|
ScopedSpinLock lock(m_lock);
|
|
if (m_did_unblock)
|
|
return false;
|
|
m_did_unblock = true;
|
|
|
|
if (m_completion->tag != tag)
|
|
return false;
|
|
if (!m_completion->result.is_error())
|
|
m_message = move(*m_completion->message);
|
|
}
|
|
return unblock();
|
|
}
|
|
|
|
void Plan9FS::Blocker::not_blocking(bool)
|
|
{
|
|
{
|
|
ScopedSpinLock lock(m_lock);
|
|
if (m_did_unblock)
|
|
return;
|
|
}
|
|
|
|
m_fs.m_completion_blocker.try_unblock(*this);
|
|
}
|
|
|
|
bool Plan9FS::Blocker::is_completed() const
|
|
{
|
|
ScopedSpinLock lock(m_completion->lock);
|
|
return m_completion->completed;
|
|
}
|
|
|
|
bool Plan9FS::Plan9FSBlockCondition::should_add_blocker(Thread::Blocker& b, void*)
|
|
{
|
|
// NOTE: m_lock is held already!
|
|
auto& blocker = static_cast<Blocker&>(b);
|
|
return !blocker.is_completed();
|
|
}
|
|
|
|
void Plan9FS::Plan9FSBlockCondition::unblock_completed(u16 tag)
|
|
{
|
|
unblock([&](Thread::Blocker& b, void*, bool&) {
|
|
VERIFY(b.blocker_type() == Thread::Blocker::Type::Plan9FS);
|
|
auto& blocker = static_cast<Blocker&>(b);
|
|
return blocker.unblock(tag);
|
|
});
|
|
}
|
|
|
|
void Plan9FS::Plan9FSBlockCondition::unblock_all()
|
|
{
|
|
unblock([&](Thread::Blocker& b, void*, bool&) {
|
|
VERIFY(b.blocker_type() == Thread::Blocker::Type::Plan9FS);
|
|
auto& blocker = static_cast<Blocker&>(b);
|
|
return blocker.unblock();
|
|
});
|
|
}
|
|
|
|
void Plan9FS::Plan9FSBlockCondition::try_unblock(Plan9FS::Blocker& blocker)
|
|
{
|
|
if (m_fs.is_complete(*blocker.completion())) {
|
|
ScopedSpinLock lock(m_lock);
|
|
blocker.unblock(blocker.completion()->tag);
|
|
}
|
|
}
|
|
|
|
bool Plan9FS::is_complete(const ReceiveCompletion& completion)
|
|
{
|
|
Locker locker(m_lock);
|
|
if (m_completions.contains(completion.tag)) {
|
|
// If it's still in the map then it can't be complete
|
|
VERIFY(!completion.completed);
|
|
return false;
|
|
}
|
|
|
|
// if it's not in the map anymore, it must be complete. But we MUST
|
|
// hold m_lock to be able to check completion.completed!
|
|
VERIFY(completion.completed);
|
|
return true;
|
|
}
|
|
|
|
KResult Plan9FS::post_message(Message& message, RefPtr<ReceiveCompletion> completion)
|
|
{
|
|
auto& buffer = message.build();
|
|
const u8* data = buffer.data();
|
|
size_t size = buffer.size();
|
|
auto& description = file_description();
|
|
|
|
Locker locker(m_send_lock);
|
|
|
|
if (completion) {
|
|
// Save the completion record *before* we send the message. This
|
|
// ensures that it exists when the thread reads the response
|
|
Locker locker(m_lock);
|
|
auto tag = completion->tag;
|
|
m_completions.set(tag, completion.release_nonnull());
|
|
// TODO: What if there is a collision? Do we need to wait until
|
|
// the existing record with the tag completes before queueing
|
|
// this one?
|
|
}
|
|
|
|
while (size > 0) {
|
|
if (!description.can_write()) {
|
|
auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
|
|
if (Thread::current()->block<Thread::WriteBlocker>({}, description, unblock_flags).was_interrupted())
|
|
return EINTR;
|
|
}
|
|
auto data_buffer = UserOrKernelBuffer::for_kernel_buffer(const_cast<u8*>(data));
|
|
auto nwritten_or_error = description.write(data_buffer, size);
|
|
if (nwritten_or_error.is_error())
|
|
return nwritten_or_error.error();
|
|
auto nwritten = nwritten_or_error.value();
|
|
data += nwritten;
|
|
size -= nwritten;
|
|
}
|
|
|
|
return KSuccess;
|
|
}
|
|
|
|
KResult Plan9FS::do_read(u8* data, size_t size)
|
|
{
|
|
auto& description = file_description();
|
|
while (size > 0) {
|
|
if (!description.can_read()) {
|
|
auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
|
|
if (Thread::current()->block<Thread::ReadBlocker>({}, description, unblock_flags).was_interrupted())
|
|
return EINTR;
|
|
}
|
|
auto data_buffer = UserOrKernelBuffer::for_kernel_buffer(data);
|
|
auto nread_or_error = description.read(data_buffer, size);
|
|
if (nread_or_error.is_error())
|
|
return nread_or_error.error();
|
|
auto nread = nread_or_error.value();
|
|
if (nread == 0)
|
|
return EIO;
|
|
data += nread;
|
|
size -= nread;
|
|
}
|
|
return KSuccess;
|
|
}
|
|
|
|
KResult Plan9FS::read_and_dispatch_one_message()
|
|
{
|
|
struct [[gnu::packed]] Header {
|
|
u32 size;
|
|
u8 type;
|
|
u16 tag;
|
|
};
|
|
Header header;
|
|
KResult result = do_read(reinterpret_cast<u8*>(&header), sizeof(header));
|
|
if (result.is_error())
|
|
return result;
|
|
|
|
auto buffer = KBuffer::try_create_with_size(header.size, Region::Access::Read | Region::Access::Write);
|
|
if (!buffer)
|
|
return ENOMEM;
|
|
// Copy the already read header into the buffer.
|
|
memcpy(buffer->data(), &header, sizeof(header));
|
|
result = do_read(buffer->data() + sizeof(header), header.size - sizeof(header));
|
|
if (result.is_error())
|
|
return result;
|
|
|
|
Locker locker(m_lock);
|
|
|
|
auto optional_completion = m_completions.get(header.tag);
|
|
if (optional_completion.has_value()) {
|
|
auto completion = optional_completion.value();
|
|
ScopedSpinLock lock(completion->lock);
|
|
completion->result = KSuccess;
|
|
completion->message = adopt_own_if_nonnull(new (nothrow) Message { buffer.release_nonnull() });
|
|
completion->completed = true;
|
|
|
|
m_completions.remove(header.tag);
|
|
m_completion_blocker.unblock_completed(header.tag);
|
|
} else {
|
|
dbgln("Received a 9p message of type {} with an unexpected tag {}, dropping", header.type, header.tag);
|
|
}
|
|
|
|
return KSuccess;
|
|
}
|
|
|
|
KResult Plan9FS::post_message_and_explicitly_ignore_reply(Message& message)
|
|
{
|
|
return post_message(message, {});
|
|
}
|
|
|
|
KResult Plan9FS::post_message_and_wait_for_a_reply(Message& message)
|
|
{
|
|
auto request_type = message.type();
|
|
auto tag = message.tag();
|
|
auto completion = adopt_ref(*new ReceiveCompletion(tag));
|
|
auto result = post_message(message, completion);
|
|
if (result.is_error())
|
|
return result;
|
|
if (Thread::current()->block<Plan9FS::Blocker>({}, *this, message, completion).was_interrupted())
|
|
return EINTR;
|
|
|
|
if (completion->result.is_error()) {
|
|
dbgln("Plan9FS: Message was aborted with error {}", completion->result.error());
|
|
return EIO;
|
|
}
|
|
|
|
auto reply_type = message.type();
|
|
|
|
if (reply_type == Message::Type::Rlerror) {
|
|
// Contains a numerical Linux errno; hopefully our errno numbers match.
|
|
u32 error_code;
|
|
message >> error_code;
|
|
return KResult((ErrnoCode)error_code);
|
|
} else if (reply_type == Message::Type::Rerror) {
|
|
// Contains an error message. We could attempt to parse it, but for now
|
|
// we simply return -EIO instead. In 9P200.u, it can also contain a
|
|
// numerical errno in an unspecified encoding; we ignore those too.
|
|
StringView error_name;
|
|
message >> error_name;
|
|
dbgln("Plan9FS: Received error name {}", error_name);
|
|
return EIO;
|
|
} else if ((u8)reply_type != (u8)request_type + 1) {
|
|
// Other than those error messages. we only expect the matching reply
|
|
// message type.
|
|
dbgln("Plan9FS: Received unexpected message type {} in response to {}", (u8)reply_type, (u8)request_type);
|
|
return EIO;
|
|
} else {
|
|
return KSuccess;
|
|
}
|
|
}
|
|
|
|
size_t Plan9FS::adjust_buffer_size(size_t size) const
|
|
{
|
|
size_t max_size = m_max_message_size - Message::max_header_size;
|
|
return min(size, max_size);
|
|
}
|
|
|
|
void Plan9FS::thread_main()
|
|
{
|
|
dbgln("Plan9FS: Thread running");
|
|
do {
|
|
auto result = read_and_dispatch_one_message();
|
|
if (result.is_error()) {
|
|
// If we fail to read, wake up everyone with an error.
|
|
Locker locker(m_lock);
|
|
|
|
for (auto& it : m_completions) {
|
|
it.value->result = result;
|
|
it.value->completed = true;
|
|
}
|
|
m_completions.clear();
|
|
m_completion_blocker.unblock_all();
|
|
dbgln("Plan9FS: Thread terminating, error reading");
|
|
return;
|
|
}
|
|
} while (!m_thread_shutdown);
|
|
dbgln("Plan9FS: Thread terminating");
|
|
}
|
|
|
|
void Plan9FS::ensure_thread()
|
|
{
|
|
ScopedSpinLock lock(m_thread_lock);
|
|
if (!m_thread_running.exchange(true, AK::MemoryOrder::memory_order_acq_rel)) {
|
|
Process::create_kernel_process(m_thread, "Plan9FS", [&]() {
|
|
thread_main();
|
|
m_thread_running.store(false, AK::MemoryOrder::memory_order_release);
|
|
});
|
|
}
|
|
}
|
|
|
|
Plan9FSInode::Plan9FSInode(Plan9FS& fs, u32 fid)
|
|
: Inode(fs, fid)
|
|
{
|
|
}
|
|
|
|
NonnullRefPtr<Plan9FSInode> Plan9FSInode::create(Plan9FS& fs, u32 fid)
|
|
{
|
|
return adopt_ref(*new Plan9FSInode(fs, fid));
|
|
}
|
|
|
|
Plan9FSInode::~Plan9FSInode()
|
|
{
|
|
Plan9FS::Message clunk_request { fs(), Plan9FS::Message::Type::Tclunk };
|
|
clunk_request << fid();
|
|
// FIXME: Should we observe this error somehow?
|
|
[[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(clunk_request);
|
|
}
|
|
|
|
KResult Plan9FSInode::ensure_open_for_mode(int mode)
|
|
{
|
|
bool use_lopen = fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L;
|
|
u32 l_mode = 0;
|
|
u8 p9_mode = 0;
|
|
|
|
{
|
|
Locker locker(m_lock);
|
|
|
|
// If it's already open in this mode, we're done.
|
|
if ((m_open_mode & mode) == mode)
|
|
return KSuccess;
|
|
|
|
m_open_mode |= mode;
|
|
|
|
if ((m_open_mode & O_RDWR) == O_RDWR) {
|
|
l_mode |= 2;
|
|
p9_mode |= 2;
|
|
} else if (m_open_mode & O_WRONLY) {
|
|
l_mode |= 1;
|
|
p9_mode |= 1;
|
|
} else if (m_open_mode & O_RDONLY) {
|
|
// Leave the values at 0.
|
|
}
|
|
}
|
|
|
|
if (use_lopen) {
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tlopen };
|
|
message << fid() << l_mode;
|
|
return fs().post_message_and_wait_for_a_reply(message);
|
|
} else {
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Topen };
|
|
message << fid() << p9_mode;
|
|
return fs().post_message_and_wait_for_a_reply(message);
|
|
}
|
|
}
|
|
|
|
KResultOr<size_t> Plan9FSInode::read_bytes(off_t offset, size_t size, UserOrKernelBuffer& buffer, FileDescription*) const
|
|
{
|
|
auto result = const_cast<Plan9FSInode&>(*this).ensure_open_for_mode(O_RDONLY);
|
|
if (result.is_error())
|
|
return result;
|
|
|
|
size = fs().adjust_buffer_size(size);
|
|
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Treadlink };
|
|
StringView data;
|
|
|
|
// Try readlink first.
|
|
bool readlink_succeded = false;
|
|
if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L && offset == 0) {
|
|
message << fid();
|
|
result = fs().post_message_and_wait_for_a_reply(message);
|
|
if (result.is_success()) {
|
|
readlink_succeded = true;
|
|
message >> data;
|
|
}
|
|
}
|
|
|
|
if (!readlink_succeded) {
|
|
message = Plan9FS::Message { fs(), Plan9FS::Message::Type::Tread };
|
|
message << fid() << (u64)offset << (u32)size;
|
|
result = fs().post_message_and_wait_for_a_reply(message);
|
|
if (result.is_error())
|
|
return result.error();
|
|
data = message.read_data();
|
|
}
|
|
|
|
// Guard against the server returning more data than requested.
|
|
size_t nread = min(data.length(), size);
|
|
if (!buffer.write(data.characters_without_null_termination(), nread))
|
|
return EFAULT;
|
|
|
|
return nread;
|
|
}
|
|
|
|
KResultOr<size_t> Plan9FSInode::write_bytes(off_t offset, size_t size, const UserOrKernelBuffer& data, FileDescription*)
|
|
{
|
|
auto result = ensure_open_for_mode(O_WRONLY);
|
|
if (result.is_error())
|
|
return result.error();
|
|
|
|
size = fs().adjust_buffer_size(size);
|
|
|
|
auto data_copy = data.copy_into_string(size); // FIXME: this seems ugly
|
|
if (data_copy.is_null())
|
|
return EFAULT;
|
|
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Twrite };
|
|
message << fid() << (u64)offset;
|
|
message.append_data(data_copy);
|
|
result = fs().post_message_and_wait_for_a_reply(message);
|
|
if (result.is_error())
|
|
return result.error();
|
|
|
|
u32 nwritten;
|
|
message >> nwritten;
|
|
return nwritten;
|
|
}
|
|
|
|
InodeMetadata Plan9FSInode::metadata() const
|
|
{
|
|
InodeMetadata metadata;
|
|
metadata.inode = identifier();
|
|
|
|
// 9P2000.L; TODO: 9P2000 & 9P2000.u
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tgetattr };
|
|
message << fid() << (u64)GetAttrMask::Basic;
|
|
auto result = fs().post_message_and_wait_for_a_reply(message);
|
|
if (result.is_error()) {
|
|
// Just return blank metadata; hopefully that's enough to result in an
|
|
// error at some upper layer. Ideally, there would be a way for
|
|
// Inode::metadata() to return failure.
|
|
return metadata;
|
|
}
|
|
|
|
u64 valid;
|
|
Plan9FS::qid qid;
|
|
u32 mode;
|
|
u32 uid;
|
|
u32 gid;
|
|
u64 nlink;
|
|
u64 rdev;
|
|
u64 size;
|
|
u64 blksize;
|
|
u64 blocks;
|
|
message >> valid >> qid >> mode >> uid >> gid >> nlink >> rdev >> size >> blksize >> blocks;
|
|
// TODO: times...
|
|
|
|
if (valid & (u64)GetAttrMask::Mode)
|
|
metadata.mode = mode;
|
|
if (valid & (u64)GetAttrMask::NLink)
|
|
metadata.link_count = nlink;
|
|
|
|
#if 0
|
|
// FIXME: Map UID/GID somehow? Or what do we do?
|
|
if (valid & (u64)GetAttrMask::UID)
|
|
metadata.uid = uid;
|
|
if (valid & (u64)GetAttrMask::GID)
|
|
metadata.uid = gid;
|
|
// FIXME: What about device nodes?
|
|
if (valid & (u64)GetAttrMask::RDev)
|
|
metadata.encoded_device = 0; // TODO
|
|
#endif
|
|
|
|
if (valid & (u64)GetAttrMask::Size)
|
|
metadata.size = size;
|
|
if (valid & (u64)GetAttrMask::Blocks) {
|
|
metadata.block_size = blksize;
|
|
metadata.block_count = blocks;
|
|
}
|
|
|
|
return metadata;
|
|
}
|
|
|
|
void Plan9FSInode::flush_metadata()
|
|
{
|
|
// Do nothing.
|
|
}
|
|
|
|
KResultOr<size_t> Plan9FSInode::directory_entry_count() const
|
|
{
|
|
size_t count = 0;
|
|
KResult result = traverse_as_directory([&count](auto&) {
|
|
count++;
|
|
return true;
|
|
});
|
|
|
|
if (result.is_error())
|
|
return result;
|
|
|
|
return count;
|
|
}
|
|
|
|
KResult Plan9FSInode::traverse_as_directory(Function<bool(FileSystem::DirectoryEntryView const&)> callback) const
|
|
{
|
|
KResult result = KSuccess;
|
|
|
|
// TODO: Should we synthesize "." and ".." here?
|
|
|
|
if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L) {
|
|
// Start by cloning the fid and opening it.
|
|
auto clone_fid = fs().allocate_fid();
|
|
{
|
|
Plan9FS::Message clone_message { fs(), Plan9FS::Message::Type::Twalk };
|
|
clone_message << fid() << clone_fid << (u16)0;
|
|
result = fs().post_message_and_wait_for_a_reply(clone_message);
|
|
if (result.is_error())
|
|
return result;
|
|
Plan9FS::Message open_message { fs(), Plan9FS::Message::Type::Tlopen };
|
|
open_message << clone_fid << (u32)0;
|
|
result = fs().post_message_and_wait_for_a_reply(open_message);
|
|
if (result.is_error()) {
|
|
Plan9FS::Message close_message { fs(), Plan9FS::Message::Type::Tclunk };
|
|
close_message << clone_fid;
|
|
// FIXME: Should we observe this error?
|
|
[[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(close_message);
|
|
return result;
|
|
}
|
|
}
|
|
|
|
u64 offset = 0;
|
|
u32 count = fs().adjust_buffer_size(8 * MiB);
|
|
|
|
while (true) {
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Treaddir };
|
|
message << clone_fid << offset << count;
|
|
result = fs().post_message_and_wait_for_a_reply(message);
|
|
if (result.is_error())
|
|
break;
|
|
|
|
StringView data = message.read_data();
|
|
if (data.is_empty()) {
|
|
// We've reached the end.
|
|
break;
|
|
}
|
|
|
|
for (Plan9FS::Message::Decoder decoder { data }; decoder.has_more_data();) {
|
|
Plan9FS::qid qid;
|
|
u8 type;
|
|
StringView name;
|
|
decoder >> qid >> offset >> type >> name;
|
|
callback({ name, { fsid(), fs().allocate_fid() }, 0 });
|
|
}
|
|
}
|
|
|
|
Plan9FS::Message close_message { fs(), Plan9FS::Message::Type::Tclunk };
|
|
close_message << clone_fid;
|
|
// FIXME: Should we observe this error?
|
|
[[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(close_message);
|
|
return result;
|
|
} else {
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
}
|
|
|
|
RefPtr<Inode> Plan9FSInode::lookup(StringView name)
|
|
{
|
|
u32 newfid = fs().allocate_fid();
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Twalk };
|
|
message << fid() << newfid << (u16)1 << name;
|
|
auto result = fs().post_message_and_wait_for_a_reply(message);
|
|
|
|
if (result.is_error())
|
|
return nullptr;
|
|
|
|
return Plan9FSInode::create(fs(), newfid);
|
|
}
|
|
|
|
KResultOr<NonnullRefPtr<Inode>> Plan9FSInode::create_child(const String&, mode_t, dev_t, uid_t, gid_t)
|
|
{
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
|
|
KResult Plan9FSInode::add_child(Inode&, const StringView&, mode_t)
|
|
{
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
|
|
KResult Plan9FSInode::remove_child(const StringView&)
|
|
{
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
|
|
KResult Plan9FSInode::chmod(mode_t)
|
|
{
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
|
|
KResult Plan9FSInode::chown(uid_t, gid_t)
|
|
{
|
|
// TODO
|
|
return ENOTIMPL;
|
|
}
|
|
|
|
KResult Plan9FSInode::truncate(u64 new_size)
|
|
{
|
|
if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L) {
|
|
Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tsetattr };
|
|
SetAttrMask valid = SetAttrMask::Size;
|
|
u32 mode = 0;
|
|
u32 uid = 0;
|
|
u32 gid = 0;
|
|
u64 atime_sec = 0;
|
|
u64 atime_nsec = 0;
|
|
u64 mtime_sec = 0;
|
|
u64 mtime_nsec = 0;
|
|
message << fid() << (u64)valid << mode << uid << gid << new_size << atime_sec << atime_nsec << mtime_sec << mtime_nsec;
|
|
return fs().post_message_and_wait_for_a_reply(message);
|
|
} else {
|
|
// TODO: wstat version
|
|
return KSuccess;
|
|
}
|
|
}
|
|
|
|
}
|