blob: 8ac35c4bde361182809de9e3a6bdc375944200a2 [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.
#ifndef SRC_LIB_FIDL_CODEC_WIRE_OBJECT_H_
#define SRC_LIB_FIDL_CODEC_WIRE_OBJECT_H_
#include <lib/fidl/cpp/message.h>
#include <lib/syslog/cpp/macros.h>
#include <map>
#include <memory>
#include <string_view>
#include <vector>
#include "src/lib/fidl_codec/library_loader.h"
#include "src/lib/fidl_codec/message_decoder.h"
namespace fidl_codec {
class FidlMessageValue;
class HandleValue;
class StringValue;
class StructValue;
class VectorValue;
class Visitor;
// Base class for all the values we can find within a message.
class Value {
public:
Value() = default;
virtual ~Value() = default;
virtual bool IsNull() const { return false; }
// Returns the uint8_t value of the value. If the value is not a uint8_t value this returns zero.
// This is used to eventually display a vector of uint8_t values as a string.
virtual uint8_t GetUint8Value() const { return 0; }
// Gets the integer value of the value. Returns false if the node can't compute an integer value.
// For floating point values, the floating point value is converted to the nearest integer
// value.
virtual bool GetIntegerValue(uint64_t* absolute, bool* negative) const { return false; }
// Gets the floating point value of the value. Returns false if the node can't compute a floating
// point value. For integer values, we can lost precision during the conversion.
virtual bool GetDoubleValue(double* result) const { return false; }
// Methods to downcast a value.
virtual const StringValue* AsStringValue() const { return nullptr; }
virtual const HandleValue* AsHandleValue() const { return nullptr; }
virtual StructValue* AsStructValue() { return nullptr; }
virtual const StructValue* AsStructValue() const { return nullptr; }
virtual const VectorValue* AsVectorValue() const { return nullptr; }
virtual const FidlMessageValue* AsFidlMessageValue() const { return nullptr; }
// Returns true if we need to load information about the handle (call to zx_object_get_info with
// ZX_INFO_HANDLE_TABLE). We need to load information about the handle if one of the handles of
// the value has an unknown koid.
virtual bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const {
return false;
}
// Returns the size needed to display the value. If the needed size is
// greater than |remaining_size|, the return value can be anything greater
// than |remaining_size| and the only useful information is that the value
// can't fit in |remaining_size|.
// Remaining size is just an optimization parameter. It avoids to compute the
// whole display size for an object: the computation is stopped as soon as we
// find that the object doesn't fit.
virtual size_t DisplaySize(const Type* for_type, size_t remaining_size) const = 0;
// Pretty print of the value.
virtual void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const = 0;
// Use a visitor on this value;
virtual void Visit(Visitor* visitor, const Type* for_type) const = 0;
};
// An invalid value. This value can't be present in a valid object.
// It can only be found if we had an error while decoding a message.
class InvalidValue : public Value {
public:
InvalidValue() = default;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override {
constexpr int kInvalidSize = 7;
return kInvalidSize; // length of "invalid"
}
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override {
printer << Red << "invalid" << ResetColor;
}
void Visit(Visitor* visitor, const Type* for_type) const override;
};
// A null value.
class NullValue : public Value {
public:
NullValue() = default;
bool IsNull() const override { return true; }
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override {
constexpr int kNullSize = 4;
return kNullSize; // length of "null"
}
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override {
printer << Red << "null" << ResetColor;
}
void Visit(Visitor* visitor, const Type* for_type) const override;
};
// A value with no known representation (we only print the raw data).
class RawValue : public Value {
public:
RawValue(const uint8_t* data, size_t size) : data_(data, data + size) {}
const std::vector<uint8_t>& data() const { return data_; }
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const std::vector<uint8_t> data_;
};
// A Boolean value.
class BoolValue : public Value {
public:
explicit BoolValue(uint8_t value) : value_(value) {}
uint8_t value() const { return value_; }
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const uint8_t value_;
};
class IntegerValue : public Value {
public:
IntegerValue(uint64_t absolute_value, bool negative)
: absolute_value_(absolute_value), negative_(negative) {}
explicit IntegerValue(int64_t value)
: absolute_value_((value < 0) ? -static_cast<uint64_t>(value) : value),
negative_(value < 0) {}
explicit IntegerValue(uint64_t value) : absolute_value_(value), negative_(false) {}
uint64_t absolute_value() const { return absolute_value_; }
bool negative() const { return negative_; }
uint8_t GetUint8Value() const override;
bool GetIntegerValue(uint64_t* absolute, bool* negative) const override {
*absolute = absolute_value_;
*negative = negative_;
return true;
}
bool GetDoubleValue(double* result) const override {
*result = static_cast<double>(absolute_value_);
if (negative_) {
*result = -(*result);
}
return true;
}
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const uint64_t absolute_value_;
const bool negative_;
};
class DoubleValue : public Value {
public:
explicit DoubleValue(double value) : value_(value) {}
double value() const { return value_; }
bool GetIntegerValue(uint64_t* absolute, bool* negative) const override {
if (value_ < 0) {
*absolute = static_cast<uint64_t>(-value_);
*negative = true;
} else {
*absolute = static_cast<uint64_t>(value_);
*negative = false;
}
return true;
}
bool GetDoubleValue(double* result) const override {
*result = value_;
return true;
}
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const double value_;
};
// A string value.
class StringValue : public Value {
public:
explicit StringValue(std::string_view string) : string_(string) {}
const std::string& string() const { return string_; }
const StringValue* AsStringValue() const override { return this; }
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const std::string string_;
};
// A handle.
class HandleValue : public Value {
public:
explicit HandleValue(const zx_handle_disposition_t& handle) : handle_(handle) {}
const zx_handle_disposition_t& handle() const { return handle_; }
const HandleValue* AsHandleValue() const override { return this; }
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const zx_handle_disposition_t handle_;
};
// An union.
class UnionValue : public Value {
public:
UnionValue(const UnionMember& member, std::unique_ptr<Value> value)
: member_(member), value_(std::move(value)) {}
const UnionMember& member() const { return member_; }
const std::unique_ptr<Value>& value() const { return value_; }
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const UnionMember& member_;
const std::unique_ptr<Value> value_;
};
// An instance of a Struct. This includes requests and responses which are also structs.
class StructValue : public Value {
public:
explicit StructValue(const Struct& struct_definition) : struct_definition_(struct_definition) {}
const Struct& struct_definition() const { return struct_definition_; }
const std::map<const StructMember*, std::unique_ptr<Value>>& fields() const { return fields_; }
void AddField(const StructMember* member, std::unique_ptr<Value> value) {
fields_.emplace(std::make_pair(member, std::move(value)));
}
void inline AddField(std::string_view name, std::unique_ptr<Value> value) {
AddField(name, 0, std::move(value));
}
void AddField(std::string_view name, uint32_t id, std::unique_ptr<Value> value);
StructValue* AsStructValue() override { return this; }
const StructValue* AsStructValue() const override { return this; }
std::pair<const Type*, const Value*> GetFieldValue(std::string_view field_name) const;
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
// Extract the JSON for this object.
void ExtractJson(rapidjson::Document::AllocatorType& allocator, rapidjson::Value& result) const;
private:
const Struct& struct_definition_;
std::map<const StructMember*, std::unique_ptr<Value>> fields_;
};
// A vector.
class VectorValue : public Value {
public:
VectorValue() = default;
const std::vector<std::unique_ptr<Value>>& values() const { return values_; }
void AddValue(std::unique_ptr<Value> value) {
if (value == nullptr) {
is_string_ = false;
} else {
uint8_t uvalue = value->GetUint8Value();
if (!std::isprint(uvalue)) {
if ((uvalue == '\r') || (uvalue == '\n')) {
has_new_line_ = true;
} else {
is_string_ = false;
}
}
}
values_.push_back(std::move(value));
}
const VectorValue* AsVectorValue() const override { return this; }
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
std::vector<std::unique_ptr<Value>> values_;
bool is_string_ = true;
bool has_new_line_ = false;
};
// A table.
class TableValue : public Value {
public:
explicit TableValue(const Table& table_definition) : table_definition_(table_definition) {}
const Table& table_definition() const { return table_definition_; }
const std::map<const TableMember*, std::unique_ptr<Value>>& members() const { return members_; }
Ordinal32 highest_member() const { return highest_member_; }
void AddMember(const TableMember* member, std::unique_ptr<Value> value) {
members_.emplace(std::make_pair(member, std::move(value)));
if (member->ordinal() > highest_member_) {
highest_member_ = member->ordinal();
}
}
bool AddMember(std::string_view name, std::unique_ptr<Value> value);
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
const Table& table_definition_;
std::map<const TableMember*, std::unique_ptr<Value>> members_;
Ordinal32 highest_member_ = 0;
};
// An instance of a FIDL message.
class FidlMessageValue : public Value {
public:
FidlMessageValue(fidl_codec::DecodedMessage* message, std::string global_errors,
const uint8_t* bytes, uint32_t num_bytes, const zx_handle_disposition_t* handles,
uint32_t num_handles);
FidlMessageValue(zx_txid_t txid, uint64_t ordinal, const std::string& global_errors,
const std::string& epitaph_error, bool received, bool is_request,
bool unknown_direction, const fidl_codec::InterfaceMethod* method,
const uint8_t* bytes, size_t byte_size, const std::string& request_errors,
const std::string& response_errors)
: txid_(txid),
ordinal_(ordinal),
global_errors_(global_errors),
epitaph_error_(epitaph_error),
received_(received),
is_request_(is_request),
unknown_direction_(unknown_direction),
method_(method),
bytes_(bytes, bytes + byte_size),
request_errors_(request_errors),
response_errors_(response_errors) {}
zx_txid_t txid() const { return txid_; }
uint64_t ordinal() const { return ordinal_; }
const std::string& global_errors() const { return global_errors_; }
const std::string& epitaph_error() const { return epitaph_error_; }
bool received() const { return received_; }
bool is_request() const { return is_request_; }
bool unknown_direction() const { return unknown_direction_; }
const fidl_codec::InterfaceMethod* method() const { return method_; }
const std::vector<uint8_t>& bytes() const { return bytes_; }
const std::vector<zx_handle_disposition_t>& handles() const { return handles_; }
void add_handle(const zx_handle_disposition_t& handle) { handles_.emplace_back(handle); }
const StructValue* decoded_request() const { return decoded_request_.get(); }
void set_decoded_request(std::unique_ptr<StructValue> decoded_request) {
decoded_request_ = std::move(decoded_request);
}
const std::string& request_errors() const { return request_errors_; }
const StructValue* decoded_response() const { return decoded_response_.get(); }
void set_decoded_response(std::unique_ptr<StructValue> decoded_response) {
decoded_response_ = std::move(decoded_response);
}
const std::string& response_errors() const { return response_errors_; }
bool matched_request() const { return (decoded_request_ != nullptr) && request_errors_.empty(); }
bool matched_response() const {
return (decoded_response_ != nullptr) && response_errors_.empty();
}
const FidlMessageValue* AsFidlMessageValue() const override { return this; }
bool NeedsToLoadHandleInfo(zx_koid_t tid,
semantic::HandleSemantic* handle_semantic) const override;
size_t DisplaySize(const Type* for_type, size_t remaining_size) const override;
void PrettyPrint(const Type* for_type, PrettyPrinter& printer) const override;
void PrintMessage(PrettyPrinter& printer) const;
void PrintMessageBody(PrettyPrinter& printer) const;
void DumpMessage(PrettyPrinter& printer) const;
void Visit(Visitor* visitor, const Type* for_type) const override;
private:
// The transfer ID of the mesage.
const zx_txid_t txid_;
// The ordinal of the message.
const uint64_t ordinal_;
// Global errors for the message (errors before we can start decoding anything).
const std::string global_errors_;
// Text value of the error status of the epitaph.
const std::string epitaph_error_;
// True if the message was received.
const bool received_;
// True if the message is a request. False if the message is a response.
const bool is_request_;
// True if we haven't been able to select a request of a response (case where both can be
// decoded).
const bool unknown_direction_;
// The method associated with the ordinal.
const fidl_codec::InterfaceMethod* const method_;
// All the bytes of the message.
std::vector<uint8_t> bytes_;
// All the handles of the message.
std::vector<zx_handle_disposition_t> handles_;
// Value of the request we have been able to decode.
std::unique_ptr<StructValue> decoded_request_;
// Errors generated during the decoding of the request. If not empty, decoded_request_ holds only
// a partial result.
const std::string request_errors_;
// Value of the response we have been able to decode.
std::unique_ptr<StructValue> decoded_response_;
// Errors generated during the decoding of the response. If not empty, decoded_response_ holds
// only a partial result.
const std::string response_errors_;
};
} // namespace fidl_codec
#endif // SRC_LIB_FIDL_CODEC_WIRE_OBJECT_H_