12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016 |
- /*
- * Copyright (c) 2020, Sergey Bugaev <bugaevc@serenityos.org>
- * All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions are met:
- *
- * 1. Redistributions of source code must retain the above copyright notice, this
- * list of conditions and the following disclaimer.
- *
- * 2. Redistributions in binary form must reproduce the above copyright notice,
- * this list of conditions and the following disclaimer in the documentation
- * and/or other materials provided with the distribution.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
- * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
- * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
- * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
- * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
- #include <Kernel/FileSystem/Plan9FileSystem.h>
- #include <Kernel/Process.h>
- namespace Kernel {
- NonnullRefPtr<Plan9FS> Plan9FS::create(FileDescription& file_description)
- {
- return adopt(*new Plan9FS(file_description));
- }
- Plan9FS::Plan9FS(FileDescription& file_description)
- : FileBackedFS(file_description)
- , m_completion_blocker(*this)
- {
- }
- Plan9FS::~Plan9FS()
- {
- // Make sure to destroy the root inode before the FS gets destroyed.
- if (m_root_inode) {
- ASSERT(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)
- {
- ASSERT(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)
- {
- ASSERT(m_have_been_built);
- m_built.decoder >> t;
- return *this;
- }
- StringView read_data()
- {
- ASSERT(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 ssize_t max_header_size = 24;
- private:
- template<typename N>
- Message& append_number(N number)
- {
- ASSERT(!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;
- dbg() << "Remote supports msize=" << msize << " and protocol version " << 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()) {
- dbg() << "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;
- ASSERT(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;
- ASSERT(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()
- {
- ASSERT(!m_have_been_built);
- auto tmp_buffer = m_builder.build();
- // FIXME: We should not assume success here.
- ASSERT(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*) {
- ASSERT(b.blocker_type() == Thread::Blocker::Type::Plan9FS);
- auto& blocker = static_cast<Blocker&>(b);
- return blocker.unblock(tag);
- });
- }
- void Plan9FS::Plan9FSBlockCondition::unblock_all()
- {
- BlockCondition::unblock_all([&](Thread::Blocker& b, void*) {
- ASSERT(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(m_lock);
- if (m_completions.contains(completion.tag)) {
- // If it's still in the map then it can't be complete
- ASSERT(!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!
- ASSERT(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(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(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>(nullptr, description, unblock_flags).was_interrupted())
- return KResult(-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>(nullptr, description, unblock_flags).was_interrupted())
- return KResult(-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 KResult(-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 KResult(-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(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 = new Message { buffer.release_nonnull() };
- completion->completed = true;
- m_completions.remove(header.tag);
- m_completion_blocker.unblock_completed(header.tag);
- } else {
- dbg() << "Received a 9p message of type " << header.type << " with an unexpected tag " << header.tag << ", dropping";
- }
- 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(*new ReceiveCompletion(tag));
- auto result = post_message(message, completion);
- if (result.is_error())
- return result;
- if (Thread::current()->block<Plan9FS::Blocker>(nullptr, *this, message, completion).was_interrupted())
- return KResult(-EINTR);
- if (completion->result.is_error()) {
- dbg() << "Plan9FS: Message was aborted with error " << completion->result;
- return KResult(-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(-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;
- dbg() << "Plan9FS: Received error name " << error_name;
- return KResult(-EIO);
- } else if ((u8)reply_type != (u8)request_type + 1) {
- // Other than those error messages. we only expect the matching reply
- // message type.
- dbg() << "Plan9FS: Received unexpected message type " << (u8)reply_type
- << " in response to " << (u8)request_type;
- return KResult(-EIO);
- } else {
- return KSuccess;
- }
- }
- ssize_t Plan9FS::adjust_buffer_size(ssize_t size) const
- {
- ssize_t max_size = m_max_message_size - Message::max_header_size;
- return min(size, max_size);
- }
- void Plan9FS::thread_main()
- {
- dbg() << "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(m_lock);
- for (auto& it : m_completions) {
- it.value->result = result;
- it.value->completed = true;
- }
- m_completions.clear();
- m_completion_blocker.unblock_all();
- dbg() << "Plan9FS: Thread terminating, error reading";
- return;
- }
- } while (!m_thread_shutdown.load(AK::MemoryOrder::memory_order_relaxed));
- dbg() << "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(*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?
- (void)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(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);
- }
- }
- ssize_t Plan9FSInode::read_bytes(off_t offset, ssize_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_t)size);
- if (!buffer.write(data.characters_without_null_termination(), nread))
- return -EFAULT;
- return nread;
- }
- ssize_t Plan9FSInode::write_bytes(off_t offset, ssize_t size, const UserOrKernelBuffer& data, FileDescription*)
- {
- auto result = ensure_open_for_mode(O_WRONLY);
- if (result.is_error())
- return result;
- 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(const FS::DirectoryEntryView&)> 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?
- (void)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?
- (void)fs().post_message_and_explicitly_ignore_reply(close_message);
- return result;
- } else {
- // TODO
- return KResult(-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 KResult(-ENOTIMPL);
- }
- KResult Plan9FSInode::add_child(Inode&, const StringView&, mode_t)
- {
- // TODO
- return KResult(-ENOTIMPL);
- }
- KResult Plan9FSInode::remove_child(const StringView&)
- {
- // TODO
- return KResult(-ENOTIMPL);
- }
- KResult Plan9FSInode::chmod(mode_t)
- {
- // TODO
- return KResult(-ENOTIMPL);
- }
- KResult Plan9FSInode::chown(uid_t, gid_t)
- {
- // TODO
- return KResult(-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;
- }
- }
- }
|