blob: c4ed05cc354775dfd5868ca964ee820142884224 [file] [log] [blame] [edit]
// 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 "src/lib/fidl_codec/message_decoder.h"
#include <fuchsia/sys/cpp/fidl.h>
#include <lib/async-loop/cpp/loop.h>
#include <lib/async-loop/default.h>
#include <lib/fidl/cpp/test/frobinator_impl.h>
#include <iostream>
#include <memory>
#include <sstream>
#include <string>
#include <vector>
#include <gtest/gtest.h>
#include <test/fidlcodec/examples/cpp/fidl.h>
#include "src/lib/fidl_codec/fidl_codec_test.h"
#include "src/lib/fidl_codec/library_loader.h"
#include "src/lib/fidl_codec/library_loader_test_data.h"
#include "src/lib/fidl_codec/message_decoder.h"
#include "src/lib/fidl_codec/wire_object.h"
#include "src/lib/fidl_codec/wire_parser.h"
using test::fidlcodec::examples::Echo;
using test::fidlcodec::examples::FidlCodecTestInterface;
namespace fidl_codec {
constexpr int kColumns = 80;
constexpr uint64_t kProcessKoid = 0x1234;
class MessageDecoderTest : public ::testing::Test {
protected:
void SetUp() override {
loader_ = GetLoader();
ASSERT_NE(loader_, nullptr);
display_options_.pretty_print = true;
display_options_.columns = kColumns;
decoder_ = std::make_unique<MessageDecoderDispatcher>(loader_, display_options_);
}
// Intercepts the caller's method call on a FIDL InterfacePtr and returns the bytes
// sent over the channel.
template <class T>
fidl::HLCPPIncomingMessage InvokeAndIntercept(
std::function<void(fidl::InterfacePtr<T>&)> invoker) {
fidl::HLCPPIncomingMessage message = buffer_.CreateEmptyIncomingMessage();
InterceptRequest<T>(message, invoker);
return message;
}
// Simulates a server sending an epitaph and returns the bytes sent over the channel.
fidl::HLCPPIncomingMessage InvokeAndReceiveEpitaph(zx_status_t epitaph) {
fidl::HLCPPIncomingMessage message = buffer_.CreateEmptyIncomingMessage();
// The protocol doesn't matter, no methods are actually called.
InterceptEpitaphResponse<FidlCodecTestInterface>(message, epitaph);
return message;
}
// Asserts that the decoded and FIDL message matches the expected display output.
// `syscall_type` interprets the FIDL message as received or sent.
void AssertDecoded(const fidl::HLCPPIncomingMessage& message, SyscallFidlType syscall_type,
const char* expected) {
std::unique_ptr<zx_handle_info_t[]> handle_infos;
if (message.handles().size() > 0) {
handle_infos = std::make_unique<zx_handle_info_t[]>(message.handles().size());
for (uint32_t i = 0; i < message.handles().size(); ++i) {
handle_infos[i].handle = message.handles().data()[i];
handle_infos[i].type = ZX_OBJ_TYPE_NONE;
handle_infos[i].rights = 0;
}
}
DecodedMessage decoded_message;
std::stringstream error_stream;
decoded_message.DecodeMessage(decoder(), process_koid(), ZX_HANDLE_INVALID,
message.bytes().data(), message.bytes().size(), nullptr, 0,
syscall_type, error_stream);
auto result = std::make_unique<fidl_codec::FidlMessageValue>(
&decoded_message, error_stream.str(), message.bytes().data(), message.bytes().size(),
nullptr, 0);
if (!result->unknown_direction()) {
// When the direction is known, only one of request/response must be set.
ASSERT_TRUE((result->decoded_request() == nullptr) ||
(result->decoded_response() == nullptr));
}
std::stringstream output;
PrettyPrinter printer(output, decoder()->colors(), /*pretty_print=*/true, /*line_header=*/"",
/*max_line_size=*/kColumns,
/*header_on_every_line=*/false);
result->PrettyPrint(nullptr, printer);
ASSERT_EQ(output.str(), expected) << "expected = " << expected << " actual = " << output.str();
}
MessageDecoderDispatcher* decoder() const { return decoder_.get(); }
uint64_t process_koid() const { return process_koid_; }
private:
fidl::IncomingMessageBuffer buffer_;
LibraryLoader* loader_;
std::unique_ptr<MessageDecoderDispatcher> decoder_;
DisplayOptions display_options_;
uint64_t process_koid_ = kProcessKoid;
};
#define TEST_DECODE_MESSAGE(_interface, _iface, _expected, ...) \
do { \
auto message = InvokeAndIntercept<_interface>( \
[&](fidl::InterfacePtr<_interface>& ptr) { ptr->_iface(__VA_ARGS__); }); \
AssertDecoded(message, SyscallFidlType::kOutputMessage, _expected); \
} while (0)
TEST_F(MessageDecoderTest, TestEmptyLaunched) {
decoder()->AddLaunchedProcess(process_koid());
TEST_DECODE_MESSAGE(FidlCodecTestInterface, Empty,
"sent request test.fidlcodec.examples/FidlCodecTestInterface.Empty = {}\n");
}
TEST_F(MessageDecoderTest, TestStringLaunched) {
decoder()->AddLaunchedProcess(process_koid());
TEST_DECODE_MESSAGE(FidlCodecTestInterface, String,
"sent request test.fidlcodec.examples/FidlCodecTestInterface.String = {\n"
" s: string = \"Hello World\"\n"
"}\n",
"Hello World");
}
TEST_F(MessageDecoderTest, TestStringAttached) {
TEST_DECODE_MESSAGE(FidlCodecTestInterface, String,
"sent request test.fidlcodec.examples/FidlCodecTestInterface.String = {\n"
" s: string = \"Hello World\"\n"
"}\n",
"Hello World");
}
TEST_F(MessageDecoderTest, TestEchoLaunched) {
decoder()->AddLaunchedProcess(process_koid());
TEST_DECODE_MESSAGE(Echo, EchoString,
"sent request test.fidlcodec.examples/Echo.EchoString = {\n"
" value: string = \"Hello World\"\n"
"}\n",
"Hello World", [](const ::fidl::StringPtr&) {});
}
TEST_F(MessageDecoderTest, TestEchoAttached) {
TEST_DECODE_MESSAGE(Echo, EchoString,
"Can't determine request/response. it can be:\n"
" sent request test.fidlcodec.examples/Echo.EchoString = {\n"
" value: string = \"Hello World\"\n"
" }\n"
" sent response test.fidlcodec.examples/Echo.EchoString = {\n"
" response: string = \"Hello World\"\n"
" }\n",
"Hello World", [](const ::fidl::StringPtr&) {});
}
TEST_F(MessageDecoderTest, TestEpitaphReceived) {
auto message = InvokeAndReceiveEpitaph(ZX_ERR_UNAVAILABLE);
AssertDecoded(message, SyscallFidlType::kInputMessage, "received epitaph ZX_ERR_UNAVAILABLE\n");
}
TEST_F(MessageDecoderTest, TestUnknownEpitaphReceived) {
auto message = InvokeAndReceiveEpitaph(1990);
AssertDecoded(message, SyscallFidlType::kInputMessage, "received epitaph status=1990\n");
}
TEST_F(MessageDecoderTest, TestEpitaphSent) {
auto message = InvokeAndReceiveEpitaph(ZX_ERR_INTERNAL);
AssertDecoded(message, SyscallFidlType::kOutputMessage, "sent epitaph ZX_ERR_INTERNAL\n");
}
TEST_F(MessageDecoderTest, TestUnknownEpitaphSent) {
auto message = InvokeAndReceiveEpitaph(1990);
AssertDecoded(message, SyscallFidlType::kOutputMessage, "sent epitaph status=1990\n");
}
} // namespace fidl_codec