| // Copyright 2020 The Fuchsia Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "inspect-manager.h" |
| |
| #include <lib/inspect/service/cpp/service.h> |
| #include <lib/syslog/cpp/macros.h> |
| #include <sys/stat.h> |
| |
| #include "src/lib/storage/vfs/cpp/service.h" |
| |
| namespace fio = fuchsia_io; |
| |
| namespace fshost { |
| |
| zx_status_t OpenNode(fidl::UnownedClientEnd<fio::Directory> root, const std::string& path, |
| uint32_t mode, fidl::ClientEnd<fio::Node>* result) { |
| auto dir = fidl::CreateEndpoints<fio::Node>(); |
| if (!dir.is_ok()) { |
| return dir.status_value(); |
| } |
| |
| fidl::StringView path_view(fidl::StringView::FromExternal(path)); |
| zx_status_t status = fidl::WireCall(root) |
| ->Open(fs::VnodeConnectionOptions::ReadOnly().ToIoV1Flags(), mode, |
| std::move(path_view), std::move(dir->server)) |
| .status(); |
| if (status != ZX_OK) { |
| return status; |
| } |
| *result = std::move(dir->client); |
| return ZX_OK; |
| } |
| |
| fbl::RefPtr<fs::PseudoDir> FshostInspectManager::Initialize(async_dispatcher* dispatcher) { |
| auto diagnostics_dir = fbl::MakeRefCounted<fs::PseudoDir>(); |
| diagnostics_dir->AddEntry( |
| fuchsia::inspect::Tree::Name_, |
| fbl::MakeRefCounted<fs::Service>([connector = inspect::MakeTreeHandler( |
| &inspector_, dispatcher)](zx::channel chan) mutable { |
| connector(fidl::InterfaceRequest<fuchsia::inspect::Tree>(std::move(chan))); |
| return ZX_OK; |
| })); |
| return diagnostics_dir; |
| } |
| |
| void FshostInspectManager::ServeStats(std::string name, |
| fidl::ClientEnd<fuchsia_io::Directory> root) { |
| inspector_.GetRoot().CreateLazyNode( |
| name + "_stats", |
| [this, name = std::move(name), root = std::move(root)] { |
| if (!root.is_valid()) { |
| FX_LOGS_FIRST_N(ERROR, 10) |
| << "Cannot serve stats for " << name << ": invalid root channel!"; |
| return fpromise::make_result_promise<inspect::Inspector>(fpromise::error()); |
| } |
| fidl::ClientEnd<fio::Node> root_chan; |
| zx_status_t status = OpenNode(root, ".", S_IFDIR, &root_chan); |
| if (status != ZX_OK) { |
| FX_LOGS_FIRST_N(ERROR, 10) << "Cannot serve stats for " << name |
| << ": failed to open node: " << zx_status_get_string(status); |
| return fpromise::make_result_promise<inspect::Inspector>(fpromise::error()); |
| } |
| // Note: we are unsafely assuming that |root_chan| is a directory |
| // i.e. speaks |fuchsia.io/Directory|. |
| fidl::ClientEnd<fio::Directory> root_dir(root_chan.TakeChannel()); |
| inspect::Inspector insp; |
| FillFileTreeSizes(std::move(root_dir), insp.GetRoot().CreateChild(name), &insp); |
| FillStats(root, &insp); |
| return fpromise::make_ok_promise(std::move(insp)); |
| }, |
| &inspector_); |
| } |
| |
| void FshostInspectManager::FillStats(fidl::UnownedClientEnd<fio::Directory> dir_chan, |
| inspect::Inspector* inspector) { |
| fidl::UnownedClientEnd<fuchsia_io::Directory> dir(dir_chan.channel()); |
| auto result = fidl::WireCall(dir)->QueryFilesystem(); |
| inspect::Node stats = inspector->GetRoot().CreateChild("stats"); |
| if (result.status() == ZX_OK) { |
| fidl::WireResponse<fuchsia_io::Directory::QueryFilesystem>* response = result.Unwrap(); |
| fuchsia_io::wire::FilesystemInfo* info = response->info.get(); |
| if (info != nullptr) { |
| stats.CreateUint("fvm_free_bytes", info->free_shared_pool_bytes, inspector); |
| stats.CreateUint("allocated_inodes", info->total_nodes, inspector); |
| stats.CreateUint("used_inodes", info->used_nodes, inspector); |
| // Total bytes is the size of the partition plus the size it could conceivably grow into. |
| // TODO(fxbug.dev/84626): Remove this misleading metric. |
| stats.CreateUint("total_bytes", info->total_bytes + info->free_shared_pool_bytes, inspector); |
| stats.CreateUint("allocated_bytes", info->total_bytes, inspector); |
| stats.CreateUint("used_bytes", info->used_bytes, inspector); |
| } else { |
| stats.CreateString("error", "Query failed", inspector); |
| } |
| } else { |
| stats.CreateString("error", "Query failed", inspector); |
| } |
| inspector->emplace(std::move(stats)); |
| } |
| |
| void FshostInspectManager::FillFileTreeSizes(fidl::ClientEnd<fio::Directory> current_dir, |
| inspect::Node node, inspect::Inspector* inspector) { |
| struct PendingDirectory { |
| std::unique_ptr<DirectoryEntriesIterator> entries_iterator; |
| inspect::Node node; |
| size_t total_size; |
| }; |
| |
| // Keeps track of entries in the stack, the entry at N+1 will always be a child of the entry at N |
| // to be able to update the parent `total_size` and propagate the sizes up. We use the lazy |
| // iterator to have a single child connection at a time per node. |
| std::vector<PendingDirectory> work_stack; |
| auto current = PendingDirectory{ |
| .entries_iterator = std::make_unique<DirectoryEntriesIterator>(std::move(current_dir)), |
| .node = std::move(node), |
| .total_size = 0, |
| }; |
| work_stack.push_back(std::move(current)); |
| |
| while (!work_stack.empty()) { |
| auto& current = work_stack.back(); |
| |
| // If we have finished with this node then pop it from the stack, save it in inspect and |
| // continue. |
| if (current.entries_iterator->finished()) { |
| // Maintain this node alive in inspect by adding it to the inspector value list and delete the |
| // stack item. |
| current.node.CreateUint("size", current.total_size, inspector); |
| inspector->emplace(std::move(current.node)); |
| size_t size = current.total_size; |
| |
| work_stack.pop_back(); |
| |
| // The next node in the stack is the parent of this node. Increment its size by the total size |
| // of this node. If the work stack is emtpy, then `current` is the root. |
| if (!work_stack.empty()) { |
| work_stack.back().total_size += size; |
| } |
| |
| continue; |
| } |
| |
| // Get the next entry. |
| while (auto entry = current.entries_iterator->GetNext()) { |
| // If the entry is a directory, push it to the stack and continue the stack loop. |
| if (entry->is_dir) { |
| work_stack.push_back(PendingDirectory{ |
| .entries_iterator = std::make_unique<DirectoryEntriesIterator>( |
| fidl::ClientEnd<fuchsia_io::Directory>(entry->node.TakeChannel())), |
| .node = current.node.CreateChild(entry->name), |
| .total_size = 0, |
| }); |
| break; |
| } else { |
| // If the entry is a file, record its size. |
| inspect::Node child_node = current.node.CreateChild(entry->name); |
| child_node.CreateUint("size", entry->size, inspector); |
| inspector->emplace(std::move(child_node)); |
| current.total_size += entry->size; |
| } |
| } |
| } |
| } |
| |
| void FshostInspectManager::LogCorruption(fs_management::DiskFormat format) { |
| if (!corruption_node_.has_value()) { |
| corruption_node_ = inspector_.GetRoot().CreateChild("corruption_events"); |
| } |
| auto found_it = corruption_events_.find(format); |
| if (found_it == corruption_events_.cend()) { |
| found_it = corruption_events_.emplace_hint( |
| found_it, format, corruption_node_->CreateUint(DiskFormatString(format), 0u)); |
| } |
| found_it->second.Add(1u); |
| } |
| |
| // Create a new lazy iterator. |
| DirectoryEntriesIterator::DirectoryEntriesIterator(fidl::ClientEnd<fio::Directory> directory) |
| : directory_(std::move(directory)) {} |
| |
| // Get the next entry. If there's no more entries left, this method will return std::nullopt |
| // forever. |
| std::optional<DirectoryEntry> DirectoryEntriesIterator::GetNext() { |
| // Loop until we can return an entry or there are none left. |
| while (true) { |
| // If we have pending entries to return, take one and return it. If for some reason, we fail |
| // to make a result out of the pending entry (it may not exist anymore) then keep trying until |
| // we can return one. |
| while (!pending_entries_.empty()) { |
| auto entry_name = pending_entries_.front(); |
| pending_entries_.pop(); |
| if (auto result = MaybeMakeEntry(entry_name)) { |
| return result; |
| } |
| } |
| |
| // When there are no pending entries and we have already finished, return. |
| if (finished_) { |
| return std::nullopt; |
| } |
| |
| // Load the next set of dirents. |
| RefreshPendingEntries(); |
| |
| // If we didn't find any pending entries in this batch of dirents, then we have finished. |
| if (pending_entries_.empty()) { |
| finished_ = true; |
| return std::nullopt; |
| } |
| } |
| } |
| |
| std::optional<DirectoryEntry> DirectoryEntriesIterator::MaybeMakeEntry( |
| const std::string& entry_name) { |
| // Open child of the current node with the given entry name. |
| fidl::ClientEnd<fio::Node> child_chan; |
| zx_status_t status = OpenNode(directory_, entry_name, 0, &child_chan); |
| if (status != ZX_OK) { |
| return std::nullopt; |
| } |
| |
| // Get child attributes to know whether the child is a directory or not. |
| auto result = fidl::WireCall(child_chan)->GetAttr(); |
| if (result.status() != ZX_OK) { |
| return std::nullopt; |
| } |
| fidl::WireResponse<fio::Node::GetAttr>* response = result.Unwrap(); |
| |
| bool is_dir = response->attributes.mode & fio::wire::kModeTypeDirectory; |
| return std::optional<DirectoryEntry>{{ |
| .name = entry_name, |
| .node = std::move(child_chan), |
| .size = (is_dir) ? 0 : response->attributes.content_size, |
| .is_dir = is_dir, |
| }}; |
| } |
| |
| // Reads the next set of dirents and loads them into `pending_entries_`. |
| void DirectoryEntriesIterator::RefreshPendingEntries() { |
| auto result = fidl::WireCall(directory_)->ReadDirents(fio::wire::kMaxBuf); |
| if (result.status() != ZX_OK) { |
| return; |
| } |
| fidl::WireResponse<fio::Directory::ReadDirents>* response = result.Unwrap(); |
| if (response->dirents.count() == 0) { |
| return; |
| } |
| |
| size_t offset = 0; |
| auto data_ptr = response->dirents.data(); |
| |
| while (sizeof(vdirent_t) < response->dirents.count() - offset) { |
| const vdirent_t* entry = reinterpret_cast<const vdirent_t*>(data_ptr + offset); |
| std::string entry_name(entry->name, entry->size); |
| offset += sizeof(vdirent_t) + entry->size; |
| if (entry_name == "." || entry_name == "..") { |
| continue; |
| } |
| pending_entries_.push(std::move(entry_name)); |
| } |
| } |
| |
| } // namespace fshost |