|  | // Copyright 2019 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 <lib/sys/cpp/outgoing_directory.h> | 
|  | #include <zircon/process.h> | 
|  | #include <zircon/processargs.h> | 
|  |  | 
|  | #include <utility> | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // Adds a new empty directory |name| to |dir| and returns pointer to new | 
|  | // directory. Will fail silently if directory with that name already exists. | 
|  | vfs::PseudoDir* AddNewEmptyDirectory(vfs::PseudoDir* dir, std::string name) { | 
|  | auto subdir = std::make_unique<vfs::PseudoDir>(); | 
|  | auto ptr = subdir.get(); | 
|  | dir->AddEntry(std::move(name), std::move(subdir)); | 
|  | return ptr; | 
|  | } | 
|  |  | 
|  | vfs::PseudoDir* GetOrCreateDirectory(vfs::PseudoDir* dir, std::string name) { | 
|  | vfs::internal::Node* node; | 
|  | zx_status_t status = dir->Lookup(name, &node); | 
|  | if (status != ZX_OK) { | 
|  | return AddNewEmptyDirectory(dir, std::move(name)); | 
|  | } | 
|  | return static_cast<vfs::PseudoDir*>(node); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | namespace sys { | 
|  |  | 
|  | OutgoingDirectory::OutgoingDirectory() | 
|  | : root_(std::make_unique<vfs::PseudoDir>()), | 
|  | svc_(GetOrCreateDirectory("svc")), | 
|  | debug_(GetOrCreateDirectory("debug")) {} | 
|  |  | 
|  | OutgoingDirectory::~OutgoingDirectory() = default; | 
|  |  | 
|  | zx_status_t OutgoingDirectory::Serve(zx::channel directory_request, | 
|  | async_dispatcher_t* dispatcher) { | 
|  | return root_->Serve(fuchsia::io::OPEN_RIGHT_READABLE | fuchsia::io::OPEN_RIGHT_WRITABLE, | 
|  | std::move(directory_request), dispatcher); | 
|  | } | 
|  |  | 
|  | zx_status_t OutgoingDirectory::ServeFromStartupInfo(async_dispatcher_t* dispatcher) { | 
|  | return Serve(zx::channel(zx_take_startup_handle(PA_DIRECTORY_REQUEST)), dispatcher); | 
|  | } | 
|  |  | 
|  | vfs::PseudoDir* OutgoingDirectory::GetOrCreateDirectory(const std::string& name) { | 
|  | return ::GetOrCreateDirectory(root_.get(), name); | 
|  | } | 
|  |  | 
|  | zx_status_t OutgoingDirectory::AddPublicService(std::unique_ptr<vfs::Service> service, | 
|  | std::string service_name) const { | 
|  | return svc_->AddEntry(std::move(service_name), std::move(service)); | 
|  | } | 
|  |  | 
|  | zx_status_t OutgoingDirectory::AddNamedService(ServiceHandler handler, std::string service, | 
|  | std::string instance) const { | 
|  | auto dir = ::GetOrCreateDirectory(svc_, std::move(service)); | 
|  | return dir->AddEntry(std::move(instance), handler.TakeDirectory()); | 
|  | } | 
|  |  | 
|  | zx_status_t OutgoingDirectory::RemoveNamedService(const std::string& service, | 
|  | const std::string& instance) const { | 
|  | vfs::internal::Node* node; | 
|  | zx_status_t status = svc_->Lookup(instance, &node); | 
|  | if (status != ZX_OK) { | 
|  | return ZX_OK; | 
|  | } | 
|  | return static_cast<vfs::PseudoDir*>(node)->RemoveEntry(service); | 
|  | } | 
|  |  | 
|  | }  // namespace sys |