blob: 597bf21a12f75b331dbd014e03df6b95345be9a0 [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_TYPES_H_
#define SRC_LIB_FIDL_CODEC_WIRE_TYPES_H_
#include <lib/fit/function.h>
#include <zircon/types.h>
#include <sstream>
#include <string>
#include <vector>
#include <rapidjson/document.h>
#include "src/lib/fidl_codec/memory_helpers.h"
#include "src/lib/fidl_codec/message_decoder.h"
#include "src/lib/fidl_codec/wire_object.h"
namespace fidl_codec {
class LibraryLoader;
class StructType;
class TableType;
class TypeVisitor;
class UnionType;
// A FIDL type. Provides methods for generating instances of this type.
class Type {
friend class InterfaceMethodParameter;
friend class Library;
public:
Type() = default;
virtual ~Type() = default;
// Returns a detailed representation of the type.
std::string ToString(bool expand = false) const;
virtual const UnionType* AsUnionType() const { return nullptr; }
virtual const StructType* AsStructType() const { return nullptr; }
virtual const TableType* AsTableType() const { return nullptr; }
// Returns true if the type is a ArrayType.
virtual bool IsArray() const { return false; }
// Returns a readable representation of the type.
virtual std::string Name() const = 0;
// Returns C++ name of the type.
virtual std::string CppName() const { return this->Name(); }
// Prints a colorized representation of the type.
virtual void PrettyPrint(PrettyPrinter& printer) const {
printer << Green << Name() << ResetColor;
};
// Returns the size of this type when embedded in another object.
virtual size_t InlineSize() const = 0;
// Whether this is a nullable type.
virtual bool Nullable() const { return false; }
// For vectors and arrays, give the type of the components (members).
virtual const Type* GetComponentType() const { return nullptr; }
// Decodes the type's inline part. It generates a Value and, eventually,
// registers the field for further decoding (secondary objects).
virtual std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const = 0;
// Pretty prints the value for this type. This is used to print numerical values.
virtual void PrettyPrint(const Value* value, PrettyPrinter& printer) const;
// Use a visitor on this value;
virtual void Visit(TypeVisitor* visitor) const = 0;
// Gets a Type object representing the |type_name|. |type| is a string that
// represents a scalar type (e.g., "float64", "uint32").
static std::unique_ptr<Type> ScalarTypeFromName(const std::string& type_name);
// Gets a Type object representing the |type|. |type| is a JSON object with a
// "subtype" field that represents a scalar type (e.g., "float64", "uint32")
static std::unique_ptr<Type> TypeFromPrimitive(const rapidjson::Value& type);
// Gets a Type object representing the |type|. |type| is a JSON object a
// field "kind" that states the type. "kind" is an identifier
// (e.g.,"foo.bar/Baz"). |loader| is the set of libraries to use to lookup
// that identifier.
static std::unique_ptr<Type> TypeFromIdentifier(LibraryLoader* loader,
const rapidjson::Value& type);
// Gets a Type object representing the |type|. |type| is a JSON object a
// field "kind" that states the type (e.g., "array", "vector", "foo.bar/Baz").
// |loader| is the set of libraries to use to find types that need to be given
// by identifier (e.g., "foo.bar/Baz").
static std::unique_ptr<Type> GetType(LibraryLoader* loader, const rapidjson::Value& type);
Type& operator=(const Type& other) = default;
Type(const Type& other) = default;
};
// An instance of this class is created when the system can't determine the real
// class (e.g., in cases of corrupted metadata).
class InvalidType : public Type {
public:
InvalidType() = default;
std::string Name() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
class BoolType : public Type {
public:
BoolType() = default;
std::string Name() const override { return "bool"; }
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
// A generic type that can be used for any integer numeric value that corresponds to a
// C++ integral value.
template <typename T>
class IntegralType : public Type {
static_assert(std::is_integral<T>::value && !std::is_same<T, bool>::value,
"IntegralType can only be used for integers");
public:
IntegralType() = default;
size_t InlineSize() const override { return sizeof(T); }
std::string CppName() const override { return this->Name() + "_t"; }
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override {
auto got = decoder->GetAddress(offset, sizeof(T));
if (got == nullptr) {
return std::make_unique<InvalidValue>();
}
T value = *reinterpret_cast<const T*>(got);
if (value < 0) {
// Sign-extend to 64 bits so the cast to uint64_t is a no-op.
int64_t tmp = value;
// Cast to uint64 before the negation to avoid a int64 overflow (-min_int64 can't be
// represented with a int64).
return std::make_unique<IntegerValue>(-static_cast<uint64_t>(tmp), true);
}
return std::make_unique<IntegerValue>(value, false);
}
};
class Int8Type : public IntegralType<int8_t> {
public:
enum class Kind { kChar, kDecimal };
explicit Int8Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Int16Type : public IntegralType<int16_t> {
public:
enum class Kind { kDecimal };
explicit Int16Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Int32Type : public IntegralType<int32_t> {
public:
enum class Kind { kDecimal };
explicit Int32Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Int64Type : public IntegralType<int64_t> {
public:
enum class Kind { kDecimal, kDuration, kTime };
explicit Int64Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Uint8Type : public IntegralType<uint8_t> {
public:
enum class Kind { kDecimal, kHexaDecimal, kPacketGuestVcpuType };
explicit Uint8Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Uint16Type : public IntegralType<uint16_t> {
public:
enum class Kind { kDecimal, kHexaDecimal, kPacketPageRequestCommand };
explicit Uint16Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Uint32Type : public IntegralType<uint32_t> {
public:
enum class Kind {
kBtiPerm,
kCachePolicy,
kClock,
kDecimal,
kExceptionState,
kHexaDecimal,
kObjectInfoTopic,
kPciBarType,
kProfileInfoFlags,
kPropType,
kPortPacketType,
kRights,
kSignals,
kStatus
};
explicit Uint32Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
class Uint64Type : public IntegralType<uint64_t> {
public:
enum class Kind { kDecimal, kHexaDecimal, kVaddr, kSize, kPaddr, kGpAddr, kUintptr };
explicit Uint64Type(Kind kind = Kind::kDecimal) : kind_(kind) {}
std::string Name() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
Kind kind_;
};
// A generic type that can be used for any numeric value that corresponds to a
// C++ arithmetic value.
template <typename T>
class NumericType : public Type {
static_assert(std::is_arithmetic<T>::value && !std::is_same<T, bool>::value,
"NumericType can only be used for numerics");
public:
size_t InlineSize() const override { return sizeof(T); }
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override {
auto got = decoder->GetAddress(offset, sizeof(T));
if (got == nullptr) {
return std::make_unique<InvalidValue>();
}
return std::make_unique<DoubleValue>(*reinterpret_cast<const T*>(got));
}
};
class Float32Type : public NumericType<float> {
public:
Float32Type() = default;
std::string Name() const override;
std::string CppName() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override {
double result;
if (!value->GetDoubleValue(&result)) {
printer << Red << "invalid" << ResetColor;
} else {
printer << Blue << std::to_string(static_cast<float>(result)) << ResetColor;
}
}
void Visit(TypeVisitor* visitor) const override;
};
class Float64Type : public NumericType<double> {
public:
Float64Type() = default;
std::string Name() const override;
std::string CppName() const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override {
double result;
if (!value->GetDoubleValue(&result)) {
printer << Red << "invalid" << ResetColor;
} else {
printer << Blue << std::to_string(result) << ResetColor;
}
}
void Visit(TypeVisitor* visitor) const override;
};
class StringType : public Type {
public:
StringType() = default;
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
bool Nullable() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
class HandleType : public Type {
public:
HandleType() = default;
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
class EnumType : public Type {
public:
explicit EnumType(const Enum& enum_definition) : enum_definition_(enum_definition) {}
const Enum& enum_definition() const { return enum_definition_; }
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
const Enum& enum_definition_;
};
class BitsType : public Type {
public:
explicit BitsType(const Bits& bits_definition) : bits_definition_(bits_definition) {}
const Bits& bits_definition() const { return bits_definition_; }
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void PrettyPrint(const Value* value, PrettyPrinter& printer) const override;
void Visit(TypeVisitor* visitor) const override;
private:
const Bits& bits_definition_;
};
class UnionType : public Type {
public:
UnionType(const Union& union_definition, bool nullable)
: union_definition_(union_definition), nullable_(nullable) {}
const Union& union_definition() const { return union_definition_; }
const UnionType* AsUnionType() const override { return this; }
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
bool Nullable() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
private:
const Union& union_definition_;
const bool nullable_;
};
class StructType : public Type {
public:
StructType(const Struct& struct_definition, bool nullable)
: struct_definition_(struct_definition), nullable_(nullable) {}
const Struct& struct_definition() const { return struct_definition_; }
const StructType* AsStructType() const override { return this; }
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
bool Nullable() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
private:
const Struct& struct_definition_;
const bool nullable_;
};
class ElementSequenceType : public Type {
public:
explicit ElementSequenceType(std::unique_ptr<Type> component_type)
: component_type_(std::move(component_type)) {
FX_DCHECK(component_type_.get() != nullptr);
}
const Type* component_type() const { return component_type_.get(); }
const Type* GetComponentType() const override;
void Visit(TypeVisitor* visitor) const override;
protected:
std::unique_ptr<Type> component_type_;
};
class ArrayType : public ElementSequenceType {
public:
ArrayType(std::unique_ptr<Type> component_type, uint32_t count)
: ElementSequenceType(std::move(component_type)), count_(count) {}
uint32_t count() const { return count_; }
bool IsArray() const override;
std::string Name() const override;
std::string CppName() const override;
void PrettyPrint(PrettyPrinter& printer) const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
private:
uint32_t count_;
};
class VectorType : public ElementSequenceType {
public:
explicit VectorType(std::unique_ptr<Type> component_type)
: ElementSequenceType(std::move(component_type)) {}
std::string Name() const override;
std::string CppName() const override;
void PrettyPrint(PrettyPrinter& printer) const override;
size_t InlineSize() const override;
bool Nullable() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
class TableType : public Type {
public:
explicit TableType(const Table& table_definition) : table_definition_(table_definition) {}
const Table& table_definition() const { return table_definition_; }
const TableType* AsTableType() const override { return this; }
std::string Name() const override;
std::string CppName() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
private:
const Table& table_definition_;
};
class FidlMessageType : public Type {
public:
FidlMessageType() = default;
std::string Name() const override;
size_t InlineSize() const override;
std::unique_ptr<Value> Decode(MessageDecoder* decoder, uint64_t offset) const override;
void Visit(TypeVisitor* visitor) const override;
};
} // namespace fidl_codec
#endif // SRC_LIB_FIDL_CODEC_WIRE_TYPES_H_