Plan9FileSystem.cpp 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987
  1. /*
  2. * Copyright (c) 2020, Sergey Bugaev <bugaevc@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <Kernel/FileSystem/Plan9FileSystem.h>
  7. #include <Kernel/Process.h>
  8. namespace Kernel {
  9. NonnullRefPtr<Plan9FS> Plan9FS::create(FileDescription& file_description)
  10. {
  11. return adopt_ref(*new Plan9FS(file_description));
  12. }
  13. Plan9FS::Plan9FS(FileDescription& file_description)
  14. : FileBackedFileSystem(file_description)
  15. , m_completion_blocker(*this)
  16. {
  17. }
  18. Plan9FS::~Plan9FS()
  19. {
  20. // Make sure to destroy the root inode before the FS gets destroyed.
  21. if (m_root_inode) {
  22. VERIFY(m_root_inode->ref_count() == 1);
  23. m_root_inode = nullptr;
  24. }
  25. }
  26. class Plan9FS::Message {
  27. public:
  28. enum class Type : u8 {
  29. // 9P2000.L
  30. Tlerror = 6,
  31. Rlerror = 7,
  32. Tstatfs = 8,
  33. Rstatfs = 9,
  34. Tlopen = 12,
  35. Rlopen = 13,
  36. Tlcreate = 14,
  37. Rlcreate = 15,
  38. Tsymlink = 16,
  39. Rsymlink = 17,
  40. Tmknod = 18,
  41. Rmknod = 19,
  42. Trename = 20,
  43. Rrename = 21,
  44. Treadlink = 22,
  45. Rreadlink = 23,
  46. Tgetattr = 24,
  47. Rgetattr = 25,
  48. Tsetattr = 26,
  49. Rsetattr = 27,
  50. Txattrwalk = 30,
  51. Rxattrwalk = 31,
  52. Txattrcreate = 32,
  53. Rxattrcreate = 33,
  54. Treaddir = 40,
  55. Rreaddir = 41,
  56. Tfsync = 50,
  57. Rfsync = 51,
  58. Tlock = 52,
  59. Rlock = 53,
  60. Tgetlock = 54,
  61. Rgetlock = 55,
  62. Tlink = 70,
  63. Rlink = 71,
  64. Tmkdir = 72,
  65. Rmkdir = 73,
  66. Trenameat = 74,
  67. Rrenameat = 75,
  68. Tunlinkat = 76,
  69. Runlinkat = 77,
  70. // 9P2000
  71. Tversion = 100,
  72. Rversion = 101,
  73. Tauth = 102,
  74. Rauth = 103,
  75. Tattach = 104,
  76. Rattach = 105,
  77. Terror = 106,
  78. Rerror = 107,
  79. Tflush = 108,
  80. Rflush = 109,
  81. Twalk = 110,
  82. Rwalk = 111,
  83. Topen = 112,
  84. Ropen = 113,
  85. Tcreate = 114,
  86. Rcreate = 115,
  87. Tread = 116,
  88. Rread = 117,
  89. Twrite = 118,
  90. Rwrite = 119,
  91. Tclunk = 120,
  92. Rclunk = 121,
  93. Tremove = 122,
  94. Rremove = 123,
  95. Tstat = 124,
  96. Rstat = 125,
  97. Twstat = 126,
  98. Rwstat = 127
  99. };
  100. class Decoder {
  101. public:
  102. explicit Decoder(const StringView& data)
  103. : m_data(data)
  104. {
  105. }
  106. Decoder& operator>>(u8&);
  107. Decoder& operator>>(u16&);
  108. Decoder& operator>>(u32&);
  109. Decoder& operator>>(u64&);
  110. Decoder& operator>>(StringView&);
  111. Decoder& operator>>(qid&);
  112. StringView read_data();
  113. bool has_more_data() const { return !m_data.is_empty(); }
  114. private:
  115. StringView m_data;
  116. template<typename N>
  117. Decoder& read_number(N& number)
  118. {
  119. VERIFY(sizeof(number) <= m_data.length());
  120. memcpy(&number, m_data.characters_without_null_termination(), sizeof(number));
  121. m_data = m_data.substring_view(sizeof(number), m_data.length() - sizeof(number));
  122. return *this;
  123. }
  124. };
  125. Message& operator<<(u8);
  126. Message& operator<<(u16);
  127. Message& operator<<(u32);
  128. Message& operator<<(u64);
  129. Message& operator<<(const StringView&);
  130. void append_data(const StringView&);
  131. template<typename T>
  132. Message& operator>>(T& t)
  133. {
  134. VERIFY(m_have_been_built);
  135. m_built.decoder >> t;
  136. return *this;
  137. }
  138. StringView read_data()
  139. {
  140. VERIFY(m_have_been_built);
  141. return m_built.decoder.read_data();
  142. }
  143. Type type() const { return m_type; }
  144. u16 tag() const { return m_tag; }
  145. Message(Plan9FS&, Type);
  146. Message(NonnullOwnPtr<KBuffer>&&);
  147. ~Message();
  148. Message& operator=(Message&&);
  149. const KBuffer& build();
  150. static constexpr size_t max_header_size = 24;
  151. private:
  152. template<typename N>
  153. Message& append_number(N number)
  154. {
  155. VERIFY(!m_have_been_built);
  156. m_builder.append(reinterpret_cast<const char*>(&number), sizeof(number));
  157. return *this;
  158. }
  159. union {
  160. KBufferBuilder m_builder;
  161. struct {
  162. NonnullOwnPtr<KBuffer> buffer;
  163. Decoder decoder;
  164. } m_built;
  165. };
  166. u16 m_tag { 0 };
  167. Type m_type { 0 };
  168. bool m_have_been_built { false };
  169. };
  170. KResult Plan9FS::initialize()
  171. {
  172. ensure_thread();
  173. Message version_message { *this, Message::Type::Tversion };
  174. version_message << (u32)m_max_message_size << "9P2000.L";
  175. auto result = post_message_and_wait_for_a_reply(version_message);
  176. if (result.is_error())
  177. return result;
  178. u32 msize;
  179. StringView remote_protocol_version;
  180. version_message >> msize >> remote_protocol_version;
  181. dbgln("Remote supports msize={} and protocol version {}", msize, remote_protocol_version);
  182. m_remote_protocol_version = parse_protocol_version(remote_protocol_version);
  183. m_max_message_size = min(m_max_message_size, (size_t)msize);
  184. // TODO: auth
  185. u32 root_fid = allocate_fid();
  186. Message attach_message { *this, Message::Type::Tattach };
  187. // FIXME: This needs a user name and an "export" name; but how do we get them?
  188. // Perhaps initialize() should accept a string of FS-specific options...
  189. attach_message << root_fid << (u32)-1 << "sergey"
  190. << "/";
  191. if (m_remote_protocol_version >= ProtocolVersion::v9P2000u)
  192. attach_message << (u32)-1;
  193. result = post_message_and_wait_for_a_reply(attach_message);
  194. if (result.is_error()) {
  195. dbgln("Attaching failed");
  196. return result;
  197. }
  198. m_root_inode = Plan9FSInode::create(*this, root_fid);
  199. if (!m_root_inode)
  200. return ENOMEM;
  201. return KSuccess;
  202. }
  203. Plan9FS::ProtocolVersion Plan9FS::parse_protocol_version(const StringView& s) const
  204. {
  205. if (s == "9P2000.L")
  206. return ProtocolVersion::v9P2000L;
  207. if (s == "9P2000.u")
  208. return ProtocolVersion::v9P2000u;
  209. return ProtocolVersion::v9P2000;
  210. }
  211. Inode& Plan9FS::root_inode()
  212. {
  213. return *m_root_inode;
  214. }
  215. Plan9FS::Message& Plan9FS::Message::operator<<(u8 number)
  216. {
  217. return append_number(number);
  218. }
  219. Plan9FS::Message& Plan9FS::Message::operator<<(u16 number)
  220. {
  221. return append_number(number);
  222. }
  223. Plan9FS::Message& Plan9FS::Message::operator<<(u32 number)
  224. {
  225. return append_number(number);
  226. }
  227. Plan9FS::Message& Plan9FS::Message::operator<<(u64 number)
  228. {
  229. return append_number(number);
  230. }
  231. Plan9FS::Message& Plan9FS::Message::operator<<(const StringView& string)
  232. {
  233. *this << static_cast<u16>(string.length());
  234. m_builder.append(string);
  235. return *this;
  236. }
  237. void Plan9FS::Message::append_data(const StringView& data)
  238. {
  239. *this << static_cast<u32>(data.length());
  240. m_builder.append(data);
  241. }
  242. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u8& number)
  243. {
  244. return read_number(number);
  245. }
  246. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u16& number)
  247. {
  248. return read_number(number);
  249. }
  250. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u32& number)
  251. {
  252. return read_number(number);
  253. }
  254. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(u64& number)
  255. {
  256. return read_number(number);
  257. }
  258. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(qid& qid)
  259. {
  260. return *this >> qid.type >> qid.version >> qid.path;
  261. }
  262. Plan9FS::Message::Decoder& Plan9FS::Message::Decoder::operator>>(StringView& string)
  263. {
  264. u16 length;
  265. *this >> length;
  266. VERIFY(length <= m_data.length());
  267. string = m_data.substring_view(0, length);
  268. m_data = m_data.substring_view_starting_after_substring(string);
  269. return *this;
  270. }
  271. StringView Plan9FS::Message::Decoder::read_data()
  272. {
  273. u32 length;
  274. *this >> length;
  275. VERIFY(length <= m_data.length());
  276. auto data = m_data.substring_view(0, length);
  277. m_data = m_data.substring_view_starting_after_substring(data);
  278. return data;
  279. }
  280. Plan9FS::Message::Message(Plan9FS& fs, Type type)
  281. : m_builder()
  282. , m_tag(fs.allocate_tag())
  283. , m_type(type)
  284. , m_have_been_built(false)
  285. {
  286. u32 size_placeholder = 0;
  287. *this << size_placeholder << (u8)type << m_tag;
  288. }
  289. Plan9FS::Message::Message(NonnullOwnPtr<KBuffer>&& buffer)
  290. : m_built { move(buffer), Decoder({ buffer->data(), buffer->size() }) }
  291. , m_have_been_built(true)
  292. {
  293. u32 size;
  294. u8 raw_type;
  295. *this >> size >> raw_type >> m_tag;
  296. m_type = (Type)raw_type;
  297. }
  298. Plan9FS::Message::~Message()
  299. {
  300. if (m_have_been_built) {
  301. m_built.buffer.~NonnullOwnPtr<KBuffer>();
  302. m_built.decoder.~Decoder();
  303. } else {
  304. m_builder.~KBufferBuilder();
  305. }
  306. }
  307. Plan9FS::Message& Plan9FS::Message::operator=(Message&& message)
  308. {
  309. m_tag = message.m_tag;
  310. m_type = message.m_type;
  311. if (m_have_been_built) {
  312. m_built.buffer.~NonnullOwnPtr<KBuffer>();
  313. m_built.decoder.~Decoder();
  314. } else {
  315. m_builder.~KBufferBuilder();
  316. }
  317. m_have_been_built = message.m_have_been_built;
  318. if (m_have_been_built) {
  319. new (&m_built.buffer) NonnullOwnPtr<KBuffer>(move(message.m_built.buffer));
  320. new (&m_built.decoder) Decoder(move(message.m_built.decoder));
  321. } else {
  322. new (&m_builder) KBufferBuilder(move(message.m_builder));
  323. }
  324. return *this;
  325. }
  326. const KBuffer& Plan9FS::Message::build()
  327. {
  328. VERIFY(!m_have_been_built);
  329. auto tmp_buffer = m_builder.build();
  330. // FIXME: We should not assume success here.
  331. VERIFY(tmp_buffer);
  332. m_have_been_built = true;
  333. m_builder.~KBufferBuilder();
  334. new (&m_built.buffer) NonnullOwnPtr<KBuffer>(tmp_buffer.release_nonnull());
  335. new (&m_built.decoder) Decoder({ m_built.buffer->data(), m_built.buffer->size() });
  336. u32* size = reinterpret_cast<u32*>(m_built.buffer->data());
  337. *size = m_built.buffer->size();
  338. return *m_built.buffer;
  339. }
  340. Plan9FS::ReceiveCompletion::ReceiveCompletion(u16 tag)
  341. : tag(tag)
  342. {
  343. }
  344. Plan9FS::ReceiveCompletion::~ReceiveCompletion()
  345. {
  346. }
  347. bool Plan9FS::Blocker::unblock(u16 tag)
  348. {
  349. {
  350. SpinlockLocker lock(m_lock);
  351. if (m_did_unblock)
  352. return false;
  353. m_did_unblock = true;
  354. if (m_completion->tag != tag)
  355. return false;
  356. if (!m_completion->result.is_error())
  357. m_message = move(*m_completion->message);
  358. }
  359. return unblock();
  360. }
  361. bool Plan9FS::Blocker::setup_blocker()
  362. {
  363. return add_to_blocker_set(m_fs.m_completion_blocker);
  364. }
  365. void Plan9FS::Blocker::will_unblock_immediately_without_blocking(UnblockImmediatelyReason)
  366. {
  367. {
  368. SpinlockLocker lock(m_lock);
  369. if (m_did_unblock)
  370. return;
  371. }
  372. m_fs.m_completion_blocker.try_unblock(*this);
  373. }
  374. bool Plan9FS::Blocker::is_completed() const
  375. {
  376. SpinlockLocker lock(m_completion->lock);
  377. return m_completion->completed;
  378. }
  379. bool Plan9FS::Plan9FSBlockerSet::should_add_blocker(Thread::Blocker& b, void*)
  380. {
  381. // NOTE: m_lock is held already!
  382. auto& blocker = static_cast<Blocker&>(b);
  383. return !blocker.is_completed();
  384. }
  385. void Plan9FS::Plan9FSBlockerSet::unblock_completed(u16 tag)
  386. {
  387. unblock_all_blockers_whose_conditions_are_met([&](Thread::Blocker& b, void*, bool&) {
  388. VERIFY(b.blocker_type() == Thread::Blocker::Type::Plan9FS);
  389. auto& blocker = static_cast<Blocker&>(b);
  390. return blocker.unblock(tag);
  391. });
  392. }
  393. void Plan9FS::Plan9FSBlockerSet::unblock_all()
  394. {
  395. unblock_all_blockers_whose_conditions_are_met([&](Thread::Blocker& b, void*, bool&) {
  396. VERIFY(b.blocker_type() == Thread::Blocker::Type::Plan9FS);
  397. auto& blocker = static_cast<Blocker&>(b);
  398. return blocker.unblock();
  399. });
  400. }
  401. void Plan9FS::Plan9FSBlockerSet::try_unblock(Plan9FS::Blocker& blocker)
  402. {
  403. if (m_fs.is_complete(*blocker.completion())) {
  404. SpinlockLocker lock(m_lock);
  405. blocker.unblock(blocker.completion()->tag);
  406. }
  407. }
  408. bool Plan9FS::is_complete(const ReceiveCompletion& completion)
  409. {
  410. MutexLocker locker(m_lock);
  411. if (m_completions.contains(completion.tag)) {
  412. // If it's still in the map then it can't be complete
  413. VERIFY(!completion.completed);
  414. return false;
  415. }
  416. // if it's not in the map anymore, it must be complete. But we MUST
  417. // hold m_lock to be able to check completion.completed!
  418. VERIFY(completion.completed);
  419. return true;
  420. }
  421. KResult Plan9FS::post_message(Message& message, RefPtr<ReceiveCompletion> completion)
  422. {
  423. auto& buffer = message.build();
  424. const u8* data = buffer.data();
  425. size_t size = buffer.size();
  426. auto& description = file_description();
  427. MutexLocker locker(m_send_lock);
  428. if (completion) {
  429. // Save the completion record *before* we send the message. This
  430. // ensures that it exists when the thread reads the response
  431. MutexLocker locker(m_lock);
  432. auto tag = completion->tag;
  433. m_completions.set(tag, completion.release_nonnull());
  434. // TODO: What if there is a collision? Do we need to wait until
  435. // the existing record with the tag completes before queueing
  436. // this one?
  437. }
  438. while (size > 0) {
  439. if (!description.can_write()) {
  440. auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
  441. if (Thread::current()->block<Thread::WriteBlocker>({}, description, unblock_flags).was_interrupted())
  442. return EINTR;
  443. }
  444. auto data_buffer = UserOrKernelBuffer::for_kernel_buffer(const_cast<u8*>(data));
  445. auto nwritten_or_error = description.write(data_buffer, size);
  446. if (nwritten_or_error.is_error())
  447. return nwritten_or_error.error();
  448. auto nwritten = nwritten_or_error.value();
  449. data += nwritten;
  450. size -= nwritten;
  451. }
  452. return KSuccess;
  453. }
  454. KResult Plan9FS::do_read(u8* data, size_t size)
  455. {
  456. auto& description = file_description();
  457. while (size > 0) {
  458. if (!description.can_read()) {
  459. auto unblock_flags = Thread::FileBlocker::BlockFlags::None;
  460. if (Thread::current()->block<Thread::ReadBlocker>({}, description, unblock_flags).was_interrupted())
  461. return EINTR;
  462. }
  463. auto data_buffer = UserOrKernelBuffer::for_kernel_buffer(data);
  464. auto nread_or_error = description.read(data_buffer, size);
  465. if (nread_or_error.is_error())
  466. return nread_or_error.error();
  467. auto nread = nread_or_error.value();
  468. if (nread == 0)
  469. return EIO;
  470. data += nread;
  471. size -= nread;
  472. }
  473. return KSuccess;
  474. }
  475. KResult Plan9FS::read_and_dispatch_one_message()
  476. {
  477. struct [[gnu::packed]] Header {
  478. u32 size;
  479. u8 type;
  480. u16 tag;
  481. };
  482. Header header;
  483. KResult result = do_read(reinterpret_cast<u8*>(&header), sizeof(header));
  484. if (result.is_error())
  485. return result;
  486. auto buffer = KBuffer::try_create_with_size(header.size, Memory::Region::Access::ReadWrite);
  487. if (!buffer)
  488. return ENOMEM;
  489. // Copy the already read header into the buffer.
  490. memcpy(buffer->data(), &header, sizeof(header));
  491. result = do_read(buffer->data() + sizeof(header), header.size - sizeof(header));
  492. if (result.is_error())
  493. return result;
  494. MutexLocker locker(m_lock);
  495. auto optional_completion = m_completions.get(header.tag);
  496. if (optional_completion.has_value()) {
  497. auto completion = optional_completion.value();
  498. SpinlockLocker lock(completion->lock);
  499. completion->result = KSuccess;
  500. completion->message = adopt_own_if_nonnull(new (nothrow) Message { buffer.release_nonnull() });
  501. completion->completed = true;
  502. m_completions.remove(header.tag);
  503. m_completion_blocker.unblock_completed(header.tag);
  504. } else {
  505. dbgln("Received a 9p message of type {} with an unexpected tag {}, dropping", header.type, header.tag);
  506. }
  507. return KSuccess;
  508. }
  509. KResult Plan9FS::post_message_and_explicitly_ignore_reply(Message& message)
  510. {
  511. return post_message(message, {});
  512. }
  513. KResult Plan9FS::post_message_and_wait_for_a_reply(Message& message)
  514. {
  515. auto request_type = message.type();
  516. auto tag = message.tag();
  517. auto completion = adopt_ref(*new ReceiveCompletion(tag));
  518. auto result = post_message(message, completion);
  519. if (result.is_error())
  520. return result;
  521. if (Thread::current()->block<Plan9FS::Blocker>({}, *this, message, completion).was_interrupted())
  522. return EINTR;
  523. if (completion->result.is_error()) {
  524. dbgln("Plan9FS: Message was aborted with error {}", completion->result.error());
  525. return EIO;
  526. }
  527. auto reply_type = message.type();
  528. if (reply_type == Message::Type::Rlerror) {
  529. // Contains a numerical Linux errno; hopefully our errno numbers match.
  530. u32 error_code;
  531. message >> error_code;
  532. return KResult((ErrnoCode)error_code);
  533. } else if (reply_type == Message::Type::Rerror) {
  534. // Contains an error message. We could attempt to parse it, but for now
  535. // we simply return EIO instead. In 9P200.u, it can also contain a
  536. // numerical errno in an unspecified encoding; we ignore those too.
  537. StringView error_name;
  538. message >> error_name;
  539. dbgln("Plan9FS: Received error name {}", error_name);
  540. return EIO;
  541. } else if ((u8)reply_type != (u8)request_type + 1) {
  542. // Other than those error messages. we only expect the matching reply
  543. // message type.
  544. dbgln("Plan9FS: Received unexpected message type {} in response to {}", (u8)reply_type, (u8)request_type);
  545. return EIO;
  546. } else {
  547. return KSuccess;
  548. }
  549. }
  550. size_t Plan9FS::adjust_buffer_size(size_t size) const
  551. {
  552. size_t max_size = m_max_message_size - Message::max_header_size;
  553. return min(size, max_size);
  554. }
  555. void Plan9FS::thread_main()
  556. {
  557. dbgln("Plan9FS: Thread running");
  558. do {
  559. auto result = read_and_dispatch_one_message();
  560. if (result.is_error()) {
  561. // If we fail to read, wake up everyone with an error.
  562. MutexLocker locker(m_lock);
  563. for (auto& it : m_completions) {
  564. it.value->result = result;
  565. it.value->completed = true;
  566. }
  567. m_completions.clear();
  568. m_completion_blocker.unblock_all();
  569. dbgln("Plan9FS: Thread terminating, error reading");
  570. return;
  571. }
  572. } while (!m_thread_shutdown);
  573. dbgln("Plan9FS: Thread terminating");
  574. }
  575. void Plan9FS::ensure_thread()
  576. {
  577. SpinlockLocker lock(m_thread_lock);
  578. if (!m_thread_running.exchange(true, AK::MemoryOrder::memory_order_acq_rel)) {
  579. Process::create_kernel_process(m_thread, "Plan9FS", [&]() {
  580. thread_main();
  581. m_thread_running.store(false, AK::MemoryOrder::memory_order_release);
  582. });
  583. }
  584. }
  585. Plan9FSInode::Plan9FSInode(Plan9FS& fs, u32 fid)
  586. : Inode(fs, fid)
  587. {
  588. }
  589. NonnullRefPtr<Plan9FSInode> Plan9FSInode::create(Plan9FS& fs, u32 fid)
  590. {
  591. return adopt_ref(*new Plan9FSInode(fs, fid));
  592. }
  593. Plan9FSInode::~Plan9FSInode()
  594. {
  595. Plan9FS::Message clunk_request { fs(), Plan9FS::Message::Type::Tclunk };
  596. clunk_request << fid();
  597. // FIXME: Should we observe this error somehow?
  598. [[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(clunk_request);
  599. }
  600. KResult Plan9FSInode::ensure_open_for_mode(int mode)
  601. {
  602. bool use_lopen = fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L;
  603. u32 l_mode = 0;
  604. u8 p9_mode = 0;
  605. {
  606. MutexLocker locker(m_inode_lock);
  607. // If it's already open in this mode, we're done.
  608. if ((m_open_mode & mode) == mode)
  609. return KSuccess;
  610. m_open_mode |= mode;
  611. if ((m_open_mode & O_RDWR) == O_RDWR) {
  612. l_mode |= 2;
  613. p9_mode |= 2;
  614. } else if (m_open_mode & O_WRONLY) {
  615. l_mode |= 1;
  616. p9_mode |= 1;
  617. } else if (m_open_mode & O_RDONLY) {
  618. // Leave the values at 0.
  619. }
  620. }
  621. if (use_lopen) {
  622. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tlopen };
  623. message << fid() << l_mode;
  624. return fs().post_message_and_wait_for_a_reply(message);
  625. } else {
  626. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Topen };
  627. message << fid() << p9_mode;
  628. return fs().post_message_and_wait_for_a_reply(message);
  629. }
  630. }
  631. KResultOr<size_t> Plan9FSInode::read_bytes(off_t offset, size_t size, UserOrKernelBuffer& buffer, FileDescription*) const
  632. {
  633. auto result = const_cast<Plan9FSInode&>(*this).ensure_open_for_mode(O_RDONLY);
  634. if (result.is_error())
  635. return result;
  636. size = fs().adjust_buffer_size(size);
  637. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Treadlink };
  638. StringView data;
  639. // Try readlink first.
  640. bool readlink_succeded = false;
  641. if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L && offset == 0) {
  642. message << fid();
  643. result = fs().post_message_and_wait_for_a_reply(message);
  644. if (result.is_success()) {
  645. readlink_succeded = true;
  646. message >> data;
  647. }
  648. }
  649. if (!readlink_succeded) {
  650. message = Plan9FS::Message { fs(), Plan9FS::Message::Type::Tread };
  651. message << fid() << (u64)offset << (u32)size;
  652. result = fs().post_message_and_wait_for_a_reply(message);
  653. if (result.is_error())
  654. return result.error();
  655. data = message.read_data();
  656. }
  657. // Guard against the server returning more data than requested.
  658. size_t nread = min(data.length(), size);
  659. if (!buffer.write(data.characters_without_null_termination(), nread))
  660. return EFAULT;
  661. return nread;
  662. }
  663. KResultOr<size_t> Plan9FSInode::write_bytes(off_t offset, size_t size, const UserOrKernelBuffer& data, FileDescription*)
  664. {
  665. auto result = ensure_open_for_mode(O_WRONLY);
  666. if (result.is_error())
  667. return result.error();
  668. size = fs().adjust_buffer_size(size);
  669. auto data_copy = data.copy_into_string(size); // FIXME: this seems ugly
  670. if (data_copy.is_null())
  671. return EFAULT;
  672. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Twrite };
  673. message << fid() << (u64)offset;
  674. message.append_data(data_copy);
  675. result = fs().post_message_and_wait_for_a_reply(message);
  676. if (result.is_error())
  677. return result.error();
  678. u32 nwritten;
  679. message >> nwritten;
  680. return nwritten;
  681. }
  682. InodeMetadata Plan9FSInode::metadata() const
  683. {
  684. InodeMetadata metadata;
  685. metadata.inode = identifier();
  686. // 9P2000.L; TODO: 9P2000 & 9P2000.u
  687. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tgetattr };
  688. message << fid() << (u64)GetAttrMask::Basic;
  689. auto result = fs().post_message_and_wait_for_a_reply(message);
  690. if (result.is_error()) {
  691. // Just return blank metadata; hopefully that's enough to result in an
  692. // error at some upper layer. Ideally, there would be a way for
  693. // Inode::metadata() to return failure.
  694. return metadata;
  695. }
  696. u64 valid;
  697. Plan9FS::qid qid;
  698. u32 mode;
  699. u32 uid;
  700. u32 gid;
  701. u64 nlink;
  702. u64 rdev;
  703. u64 size;
  704. u64 blksize;
  705. u64 blocks;
  706. message >> valid >> qid >> mode >> uid >> gid >> nlink >> rdev >> size >> blksize >> blocks;
  707. // TODO: times...
  708. if (valid & (u64)GetAttrMask::Mode)
  709. metadata.mode = mode;
  710. if (valid & (u64)GetAttrMask::NLink)
  711. metadata.link_count = nlink;
  712. #if 0
  713. // FIXME: Map UID/GID somehow? Or what do we do?
  714. if (valid & (u64)GetAttrMask::UID)
  715. metadata.uid = uid;
  716. if (valid & (u64)GetAttrMask::GID)
  717. metadata.uid = gid;
  718. // FIXME: What about device nodes?
  719. if (valid & (u64)GetAttrMask::RDev)
  720. metadata.encoded_device = 0; // TODO
  721. #endif
  722. if (valid & (u64)GetAttrMask::Size)
  723. metadata.size = size;
  724. if (valid & (u64)GetAttrMask::Blocks) {
  725. metadata.block_size = blksize;
  726. metadata.block_count = blocks;
  727. }
  728. return metadata;
  729. }
  730. void Plan9FSInode::flush_metadata()
  731. {
  732. // Do nothing.
  733. }
  734. KResult Plan9FSInode::traverse_as_directory(Function<bool(FileSystem::DirectoryEntryView const&)> callback) const
  735. {
  736. KResult result = KSuccess;
  737. // TODO: Should we synthesize "." and ".." here?
  738. if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L) {
  739. // Start by cloning the fid and opening it.
  740. auto clone_fid = fs().allocate_fid();
  741. {
  742. Plan9FS::Message clone_message { fs(), Plan9FS::Message::Type::Twalk };
  743. clone_message << fid() << clone_fid << (u16)0;
  744. result = fs().post_message_and_wait_for_a_reply(clone_message);
  745. if (result.is_error())
  746. return result;
  747. Plan9FS::Message open_message { fs(), Plan9FS::Message::Type::Tlopen };
  748. open_message << clone_fid << (u32)0;
  749. result = fs().post_message_and_wait_for_a_reply(open_message);
  750. if (result.is_error()) {
  751. Plan9FS::Message close_message { fs(), Plan9FS::Message::Type::Tclunk };
  752. close_message << clone_fid;
  753. // FIXME: Should we observe this error?
  754. [[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(close_message);
  755. return result;
  756. }
  757. }
  758. u64 offset = 0;
  759. u32 count = fs().adjust_buffer_size(8 * MiB);
  760. while (true) {
  761. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Treaddir };
  762. message << clone_fid << offset << count;
  763. result = fs().post_message_and_wait_for_a_reply(message);
  764. if (result.is_error())
  765. break;
  766. StringView data = message.read_data();
  767. if (data.is_empty()) {
  768. // We've reached the end.
  769. break;
  770. }
  771. for (Plan9FS::Message::Decoder decoder { data }; decoder.has_more_data();) {
  772. Plan9FS::qid qid;
  773. u8 type;
  774. StringView name;
  775. decoder >> qid >> offset >> type >> name;
  776. callback({ name, { fsid(), fs().allocate_fid() }, 0 });
  777. }
  778. }
  779. Plan9FS::Message close_message { fs(), Plan9FS::Message::Type::Tclunk };
  780. close_message << clone_fid;
  781. // FIXME: Should we observe this error?
  782. [[maybe_unused]] auto rc = fs().post_message_and_explicitly_ignore_reply(close_message);
  783. return result;
  784. } else {
  785. // TODO
  786. return ENOTIMPL;
  787. }
  788. }
  789. KResultOr<NonnullRefPtr<Inode>> Plan9FSInode::lookup(StringView name)
  790. {
  791. u32 newfid = fs().allocate_fid();
  792. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Twalk };
  793. message << fid() << newfid << (u16)1 << name;
  794. auto result = fs().post_message_and_wait_for_a_reply(message);
  795. if (result.is_error())
  796. return result;
  797. return Plan9FSInode::create(fs(), newfid);
  798. }
  799. KResultOr<NonnullRefPtr<Inode>> Plan9FSInode::create_child(StringView, mode_t, dev_t, UserID, GroupID)
  800. {
  801. // TODO
  802. return ENOTIMPL;
  803. }
  804. KResult Plan9FSInode::add_child(Inode&, const StringView&, mode_t)
  805. {
  806. // TODO
  807. return ENOTIMPL;
  808. }
  809. KResult Plan9FSInode::remove_child(const StringView&)
  810. {
  811. // TODO
  812. return ENOTIMPL;
  813. }
  814. KResult Plan9FSInode::chmod(mode_t)
  815. {
  816. // TODO
  817. return ENOTIMPL;
  818. }
  819. KResult Plan9FSInode::chown(UserID, GroupID)
  820. {
  821. // TODO
  822. return ENOTIMPL;
  823. }
  824. KResult Plan9FSInode::truncate(u64 new_size)
  825. {
  826. if (fs().m_remote_protocol_version >= Plan9FS::ProtocolVersion::v9P2000L) {
  827. Plan9FS::Message message { fs(), Plan9FS::Message::Type::Tsetattr };
  828. SetAttrMask valid = SetAttrMask::Size;
  829. u32 mode = 0;
  830. u32 uid = 0;
  831. u32 gid = 0;
  832. u64 atime_sec = 0;
  833. u64 atime_nsec = 0;
  834. u64 mtime_sec = 0;
  835. u64 mtime_nsec = 0;
  836. message << fid() << (u64)valid << mode << uid << gid << new_size << atime_sec << atime_nsec << mtime_sec << mtime_nsec;
  837. return fs().post_message_and_wait_for_a_reply(message);
  838. } else {
  839. // TODO: wstat version
  840. return KSuccess;
  841. }
  842. }
  843. }