blob: 7a35b156c8aa3e7f08d76dff890d8cdcbc1f30a4 [file] [log] [blame]
// 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 <fuchsia/io/llcpp/fidl.h>
#include <lib/fidl/llcpp/transaction.h>
#include <lib/fs-pty/tty-connection-internal.h>
#include <lib/zx/channel.h>
#include <lib/zx/eventpair.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <threads.h>
#include <zircon/status.h>
#include <zircon/types.h>
#include <thread>
#include <fbl/algorithm.h>
namespace {
// This adapter is necessary for going between the llcpp-style Transaction and
// the C binding style fidl_txn_t.
class Transaction : public fidl::Transaction {
public:
explicit Transaction(fidl_txn_t* txn) : txn_(txn) {}
~Transaction() {
ZX_ASSERT_MSG(status_called_,
"Transaction must have it's Status() method used. This provides "
"HandleFsSpecificMessage with the correct status value.\n");
}
/// Status() return the internal state of the DDK transaction. This MUST be called
/// to bridge the Transaction and DDK dispatcher.
zx_status_t Status() __WARN_UNUSED_RESULT {
status_called_ = true;
return status_;
}
protected:
void Reply(fidl::Message msg) final {
ZX_ASSERT(txn_);
const fidl_msg_t fidl_msg{
.bytes = msg.bytes().data(),
.handles = msg.handles().data(),
.num_bytes = static_cast<uint32_t>(msg.bytes().size()),
.num_handles = static_cast<uint32_t>(msg.handles().size()),
};
status_ = txn_->reply(txn_, &fidl_msg);
}
void Close(zx_status_t close_status) final { status_ = close_status; }
std::unique_ptr<fidl::Transaction> TakeOwnership() final {
ZX_ASSERT_MSG(false, "DdkTransaction cannot take ownership of the transaction.\n");
}
private:
fidl_txn_t* txn_;
zx_status_t status_ = ZX_OK;
bool status_called_ = false;
};
} // namespace
namespace fs_pty::internal {
zx_status_t TtyConnectionImpl::HandleFsSpecificMessage(fidl_msg_t* msg, fidl_txn_t* txn) {
Transaction transaction{txn};
if (!::llcpp::fuchsia::hardware::pty::Device::TryDispatch(this, msg, &transaction)) {
__UNUSED auto ignore = transaction.Status();
return ZX_ERR_NOT_SUPPORTED;
}
return transaction.Status();
}
// Return ZX_ERR_NOT_SUPPORTED for all of the PTY things we don't actually support
void TtyConnectionImpl::OpenClient(uint32_t id, zx::channel client,
OpenClientCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::OpenClientResponse> buf;
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED);
}
void TtyConnectionImpl::ClrSetFeature(uint32_t clr, uint32_t set,
ClrSetFeatureCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::ClrSetFeatureResponse> buf;
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED, 0);
}
void TtyConnectionImpl::GetWindowSize(GetWindowSizeCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::GetWindowSizeResponse> buf;
::llcpp::fuchsia::hardware::pty::WindowSize wsz = {.width = 0, .height = 0};
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED, wsz);
}
void TtyConnectionImpl::MakeActive(uint32_t client_pty_id, MakeActiveCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::MakeActiveResponse> buf;
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED);
}
void TtyConnectionImpl::ReadEvents(ReadEventsCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::ReadEventsResponse> buf;
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED, 0);
}
void TtyConnectionImpl::SetWindowSize(::llcpp::fuchsia::hardware::pty::WindowSize size,
SetWindowSizeCompleter::Sync completer) {
fidl::Buffer<::llcpp::fuchsia::hardware::pty::Device::SetWindowSizeResponse> buf;
completer.Reply(buf.view(), ZX_ERR_NOT_SUPPORTED);
}
// Assert in all of these, since these should be handled by fs::Connection before our
// HandleFsSpecificMessage() is called.
void TtyConnectionImpl::Read(uint64_t count, ReadCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::Write(fidl::VectorView<uint8_t> data, WriteCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Clone(uint32_t flags, zx::channel node, CloneCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Close(CloseCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::Describe(DescribeCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::GetAttr(GetAttrCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::GetFlags(GetFlagsCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::ReadAt(uint64_t count, uint64_t offset, ReadAtCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::WriteAt(fidl::VectorView<uint8_t> data, uint64_t offset,
WriteAtCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Seek(int64_t offset, ::llcpp::fuchsia::io::SeekOrigin start,
SeekCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Truncate(uint64_t length, TruncateCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::SetFlags(uint32_t flags, SetFlagsCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::GetBuffer(uint32_t flags, GetBufferCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Sync(SyncCompleter::Sync completer) { ZX_ASSERT(false); }
void TtyConnectionImpl::SetAttr(uint32_t flags, ::llcpp::fuchsia::io::NodeAttributes attributes,
SetAttrCompleter::Sync completer) {
ZX_ASSERT(false);
}
void TtyConnectionImpl::Ioctl(uint32_t opcode, uint64_t max_out,
fidl::VectorView<zx::handle> handles, fidl::VectorView<uint8_t> in,
IoctlCompleter::Sync completer) {
ZX_ASSERT(false);
}
} // namespace fs_pty::internal