blob: 34407600fda7be53b20a0e3f1e1298b5ded85df7 [file] [log] [blame] [edit]
// 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.
// This is separate from test_utils as it is not used in conformance tests and
// can therefore e.g. use handles
#ifndef SRC_LIB_FIDL_LLCPP_TESTS_TYPES_TEST_UTILS_H_
#define SRC_LIB_FIDL_LLCPP_TESTS_TYPES_TEST_UTILS_H_
#include <lib/fidl/llcpp/coding.h>
#include <lib/fidl/llcpp/message.h>
#include <lib/fidl/llcpp/traits.h>
#include <lib/zx/event.h>
#include <zircon/fidl.h>
#include <zircon/status.h>
#include <zircon/types.h>
#include <vector>
#include <gtest/gtest.h>
namespace llcpp_types_test_utils {
class HandleChecker {
public:
HandleChecker() = default;
size_t size() const { return events_.size(); }
void AddEvent(zx_handle_t event);
void CheckEvents();
private:
std::vector<zx::event> events_;
};
// Verifies that:
// - |bytes| and |handles| decodes succesfully as |FidlType|
// - all handles in |handles| are closed
// - the resulting object fails to encode
// Assuming that:
// - FidlType is a transactional message, with a single result field that is
// either a union or a table.
//
// Also runs a checker function on the decoded object, to test any properties.
// This is the intended behavior for all flexible types (unions and tables) in
// LLCPP, regardless of resourceness (since no unknown handles are stored, even on
// resource types).
template <typename FidlType, typename CheckerFunc>
void CannotProxyUnknownEnvelope(std::vector<uint8_t> bytes, std::vector<zx_handle_t> handles,
CheckerFunc check) {
// These flexible tests require the type be a message so that traits like
// HasFlexibleEnvelope exist.
// It also assumes the type under test exists under field .result within the
// wrapper struct
static_assert(fidl::IsFidlMessage<FidlType>::value, "FIDL message type required");
HandleChecker handle_checker;
for (const auto& handle : handles) {
handle_checker.AddEvent(handle);
}
const char* decode_error;
auto status = fidl_decode(FidlType::Type, bytes.data(), bytes.size(), handles.data(),
handles.size(), &decode_error);
ASSERT_EQ(status, ZX_OK) << decode_error;
auto result = reinterpret_cast<FidlType*>(&bytes[0]);
check(result->result);
handle_checker.CheckEvents();
// Here we want to encode a message with unknown data. To be able to encoded all the unknown data,
// we always use a full size buffer.
FIDL_ALIGNDECL
uint8_t encoded_bytes[ZX_CHANNEL_MAX_MSG_BYTES];
fidl::UnownedEncodedMessage<FidlType> encoded(encoded_bytes, ZX_CHANNEL_MAX_MSG_BYTES, result);
ASSERT_EQ(encoded.status(), ZX_ERR_INVALID_ARGS) << encoded.status_string();
EXPECT_STREQ(encoded.error(), "Cannot encode unknown union or table") << encoded.error();
}
} // namespace llcpp_types_test_utils
#endif // SRC_LIB_FIDL_LLCPP_TESTS_TYPES_TEST_UTILS_H_