blob: 2053a42ea6d98a24a8745748a597235edd0dae64 [file] [log] [blame]
// Copyright 2018 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 LIB_FIDL_WALKER_H_
#define LIB_FIDL_WALKER_H_
#include <lib/fidl/coding.h>
#include <lib/fidl/internal.h>
#include <lib/fidl/visitor.h>
#include <stdalign.h>
#include <zircon/assert.h>
#include <zircon/compiler.h>
#include <cstdint>
#include <cstdlib>
#include <limits>
#include <type_traits>
namespace fidl {
namespace internal {
// Some assumptions about data type layout.
static_assert(offsetof(fidl_string_t, size) == 0u, "fidl_string_t layout");
static_assert(offsetof(fidl_string_t, data) == 8u, "fidl_string_t layout");
static_assert(sizeof(fidl_string_t) == 16u, "fidl_string_t layout");
static_assert(offsetof(fidl_vector_t, count) == 0u, "fidl_vector_t layout");
static_assert(offsetof(fidl_vector_t, data) == 8u, "fidl_vector_t layout");
static_assert(sizeof(fidl_vector_t) == 16u, "fidl_vector_t layout");
static_assert(offsetof(fidl_envelope_t, num_bytes) == 0u, "fidl_envelope_t layout");
static_assert(offsetof(fidl_envelope_t, num_handles) == 4u, "fidl_envelope_t layout");
static_assert(offsetof(fidl_envelope_t, data) == 8u, "fidl_envelope_t layout");
static_assert(sizeof(fidl_envelope_t) == 16u, "fidl_envelope_t layout");
static_assert(ZX_HANDLE_INVALID == FIDL_HANDLE_ABSENT, "invalid handle equals absence marker");
constexpr uint32_t PrimitiveSize(const FidlCodedPrimitive primitive) {
switch (primitive) {
case FidlCodedPrimitive::kBool:
case FidlCodedPrimitive::kInt8:
case FidlCodedPrimitive::kUint8:
return 1;
case FidlCodedPrimitive::kInt16:
case FidlCodedPrimitive::kUint16:
return 2;
case FidlCodedPrimitive::kInt32:
case FidlCodedPrimitive::kUint32:
case FidlCodedPrimitive::kFloat32:
return 4;
case FidlCodedPrimitive::kInt64:
case FidlCodedPrimitive::kUint64:
case FidlCodedPrimitive::kFloat64:
return 8;
}
__builtin_unreachable();
}
constexpr uint32_t TypeSize(const fidl_type_t* type) {
switch (type->type_tag) {
case fidl::kFidlTypePrimitive:
return PrimitiveSize(type->coded_primitive);
case fidl::kFidlTypeEnum:
return PrimitiveSize(type->coded_enum.underlying_type);
case fidl::kFidlTypeBits:
return PrimitiveSize(type->coded_bits.underlying_type);
case fidl::kFidlTypeStructPointer:
case fidl::kFidlTypeUnionPointer:
return sizeof(uint64_t);
case fidl::kFidlTypeHandle:
return sizeof(zx_handle_t);
case fidl::kFidlTypeStruct:
return type->coded_struct.size;
case fidl::kFidlTypeTable:
return sizeof(fidl_vector_t);
case fidl::kFidlTypeUnion:
return type->coded_union.size;
case fidl::kFidlTypeXUnion:
return sizeof(fidl_xunion_t);
case fidl::kFidlTypeString:
return sizeof(fidl_string_t);
case fidl::kFidlTypeArray:
return type->coded_array.array_size;
case fidl::kFidlTypeVector:
return sizeof(fidl_vector_t);
}
__builtin_unreachable();
}
constexpr bool IsPrimitive(const fidl_type_t* type) {
switch (type->type_tag) {
case fidl::kFidlTypePrimitive:
return true;
default:
return false;
}
}
// The Walker class traverses through a FIDL message by following its coding table and
// calling the visitor implementation. VisitorImpl must be a concrete implementation of the
// fidl::Visitor interface. The concrete type is used to eliminate dynamic dispatch.
template <typename VisitorImpl>
class Walker final {
private:
using MutationTrait = typename VisitorImpl::MutationTrait;
using StartingPoint = typename VisitorImpl::StartingPoint;
using Position = typename VisitorImpl::Position;
using VisitorSuper = Visitor<MutationTrait, StartingPoint, Position>;
using Status = typename VisitorSuper::Status;
static_assert(CheckVisitorInterface<VisitorSuper, VisitorImpl>(), "");
public:
Walker(const fidl_type_t* type, StartingPoint start) : type_(type), start_(start) {}
// Walk the object/buffer located at |start_|.
void Walk(VisitorImpl& visitor);
private:
// Optionally uses non-const pointers depending on if the visitor
// is declared as mutating or not.
template <typename T>
using Ptr = typename VisitorImpl::template Ptr<T>;
// Wrapper around Position::Get with friendlier syntax.
template <typename T>
Ptr<T> PtrTo(Position position) {
return position.template Get<T>(start_);
}
// Functions that manipulate the coding stack frames.
struct Frame {
Frame(const fidl_type_t* fidl_type, Position position) : position(position) {
switch (fidl_type->type_tag) {
case fidl::kFidlTypeEnum:
state = kStateEnum;
enum_state.underlying_type = fidl_type->coded_enum.underlying_type;
enum_state.validate = fidl_type->coded_enum.validate;
break;
case fidl::kFidlTypeBits:
state = kStateBits;
bits_state.underlying_type = fidl_type->coded_bits.underlying_type;
bits_state.mask = fidl_type->coded_bits.mask;
break;
case fidl::kFidlTypeStruct:
state = kStateStruct;
struct_state.fields = fidl_type->coded_struct.fields;
struct_state.field_count = fidl_type->coded_struct.field_count;
struct_state.field = 0;
struct_state.struct_size = fidl_type->coded_struct.size;
break;
case fidl::kFidlTypeStructPointer:
state = kStateStructPointer;
struct_pointer_state.struct_type = fidl_type->coded_struct_pointer.struct_type;
break;
case fidl::kFidlTypeTable:
state = kStateTable;
table_state.field = fidl_type->coded_table.fields;
table_state.remaining_fields = fidl_type->coded_table.field_count;
table_state.present_count = 0;
table_state.ordinal = 0;
break;
case fidl::kFidlTypeUnion:
state = kStateUnion;
union_state.fields = fidl_type->coded_union.fields;
union_state.field_count = fidl_type->coded_union.field_count;
union_state.data_offset = fidl_type->coded_union.data_offset;
union_state.union_size = fidl_type->coded_union.size;
break;
case fidl::kFidlTypeUnionPointer:
state = kStateUnionPointer;
union_pointer_state.union_type = fidl_type->coded_union_pointer.union_type;
break;
case fidl::kFidlTypeXUnion:
state = kStateXUnion;
xunion_state.fields = fidl_type->coded_xunion.fields;
xunion_state.field_count = fidl_type->coded_xunion.field_count;
xunion_state.inside_envelope = false;
xunion_state.nullable = fidl_type->coded_xunion.nullable;
xunion_state.strictness = fidl_type->coded_xunion.strictness;
break;
case fidl::kFidlTypeArray:
state = kStateArray;
array_state.element = fidl_type->coded_array.element;
array_state.array_size = fidl_type->coded_array.array_size;
array_state.element_size = fidl_type->coded_array.element_size;
array_state.element_offset = 0;
break;
case fidl::kFidlTypeString:
state = kStateString;
string_state.max_size = fidl_type->coded_string.max_size;
string_state.nullable = fidl_type->coded_string.nullable;
break;
case fidl::kFidlTypeHandle:
state = kStateHandle;
handle_state.nullable = fidl_type->coded_handle.nullable;
break;
case fidl::kFidlTypeVector:
state = kStateVector;
vector_state.element = fidl_type->coded_vector.element;
vector_state.max_count = fidl_type->coded_vector.max_count;
vector_state.element_size = fidl_type->coded_vector.element_size;
vector_state.nullable = fidl_type->coded_vector.nullable;
break;
case fidl::kFidlTypePrimitive:
// Walker will never recurse into a primitive. Note: this branch must be
// implemented should we decide to support table-driven ToString in the future.
__builtin_unreachable();
}
}
Frame(const fidl::FidlCodedStruct* coded_struct, Position position) : position(position) {
state = kStateStruct;
struct_state.fields = coded_struct->fields;
struct_state.field_count = coded_struct->field_count;
struct_state.field = 0;
struct_state.struct_size = coded_struct->size;
}
Frame(const fidl::FidlCodedTable* coded_table, Position position) : position(position) {
state = kStateStruct;
table_state.field = coded_table->fields;
table_state.remaining_fields = coded_table->field_count;
table_state.present_count = 0;
table_state.ordinal = 0;
}
Frame(const fidl::FidlCodedUnion* coded_union, Position position) : position(position) {
state = kStateUnion;
union_state.fields = coded_union->fields;
union_state.field_count = coded_union->field_count;
union_state.data_offset = coded_union->data_offset;
union_state.union_size = coded_union->size;
}
Frame(const fidl::FidlCodedXUnion* coded_xunion, Position position)
: state(kStateXUnion), position(position) {
// This initialization is done in the ctor body instead of in an
// initialization list since we need to set fields in unions, which
// is much more involved in a ctor initialization list.
xunion_state.fields = coded_xunion->fields;
xunion_state.field_count = coded_xunion->field_count;
xunion_state.inside_envelope = false;
xunion_state.nullable = coded_xunion->nullable;
xunion_state.strictness = coded_xunion->strictness;
}
Frame(const fidl_type_t* element, uint32_t array_size, uint32_t element_size, Position position)
: position(position) {
state = kStateArray;
array_state.element = element;
array_state.array_size = array_size;
array_state.element_size = element_size;
array_state.element_offset = 0;
}
// The default constructor does nothing when initializing the stack of frames.
Frame() = default;
static Frame DoneSentinel() {
Frame frame;
frame.state = kStateDone;
return frame;
}
uint32_t NextStructField() {
ZX_DEBUG_ASSERT(state == kStateStruct);
uint32_t current = struct_state.field;
struct_state.field++;
return current;
}
uint32_t NextArrayOffset() {
ZX_DEBUG_ASSERT(state == kStateArray);
uint32_t current = array_state.element_offset;
array_state.element_offset += array_state.element_size;
return current;
}
enum : int {
kStateEnum,
kStateBits,
kStateStruct,
kStateStructPointer,
kStateTable,
kStateUnion,
kStateUnionPointer,
kStateXUnion,
kStateArray,
kStateString,
kStateHandle,
kStateVector,
kStateDone,
} state;
// Position into the message.
Position position;
// This is a subset of the information recorded in the
// fidl_type structures needed for coding state. For
// example, struct sizes do not need to be present here.
union {
struct {
fidl::FidlCodedPrimitive underlying_type;
fidl::EnumValidationPredicate validate;
} enum_state;
struct {
fidl::FidlCodedPrimitive underlying_type;
uint64_t mask;
} bits_state;
struct {
const fidl::FidlStructField* fields;
uint32_t field_count;
// Index of the currently processing field.
uint32_t field;
// Size of the entire struct.
uint32_t struct_size;
} struct_state;
struct {
const fidl::FidlCodedStruct* struct_type;
} struct_pointer_state;
struct {
// Sparse (but monotonically increasing) coding table array for fields;
// advance the |field| pointer on every matched ordinal to save space
const fidl::FidlTableField* field;
// Number of unseen fields in the coding table
uint32_t remaining_fields;
// How many fields are stored in the message
uint32_t present_count;
// Current ordinal (valid ordinals start at 1)
uint32_t ordinal;
// When true, the walker is currently working within an envelope, or equivalently,
// |EnterEnvelope| was successful.
bool inside_envelope;
} table_state;
struct {
// Array of coding table corresponding to each union variant.
// The union tag counts upwards from 0 without breaks; hence it can be used to
// index into the |fields| array.
const fidl::FidlUnionField* fields;
// Size of the |fields| array. Equal to the number of tags.
uint32_t field_count;
// Offset of the payload in the wire format (size of tag + padding).
uint32_t data_offset;
// Size of the entire union.
uint32_t union_size;
} union_state;
struct {
const fidl::FidlCodedUnion* union_type;
} union_pointer_state;
struct {
const fidl::FidlXUnionField* fields;
// Number of known ordinals declared in the coding table
uint32_t field_count;
// When true, the walker is currently working within an envelope, or equivalently,
// |EnterEnvelope| was successful.
bool inside_envelope;
fidl::FidlNullability nullable;
fidl::FidlStrictness strictness;
} xunion_state;
struct {
const fidl_type_t* element;
// Size of the entire array in bytes
uint32_t array_size;
// Size of a single element in bytes
uint32_t element_size;
// Byte offset of the current element being processed
uint32_t element_offset;
} array_state;
struct {
uint32_t max_size;
bool nullable;
} string_state;
struct {
bool nullable;
} handle_state;
struct {
const fidl_type_t* element;
// Upperbound on number of elements.
uint32_t max_count;
// Size of a single element in bytes
uint32_t element_size;
bool nullable;
} vector_state;
};
};
// Returns true on success and false on recursion overflow.
bool Push(Frame frame) {
if (depth_ == FIDL_RECURSION_DEPTH) {
return false;
}
coding_frames_[depth_] = frame;
++depth_;
return true;
}
void Pop() {
ZX_DEBUG_ASSERT(depth_ != 0u);
--depth_;
}
Frame* Peek() {
ZX_DEBUG_ASSERT(depth_ != 0u);
return &coding_frames_[depth_ - 1];
}
const fidl_type_t* const type_;
const StartingPoint start_;
// Decoding stack state.
uint32_t depth_ = 0u;
Frame coding_frames_[FIDL_RECURSION_DEPTH];
};
template <typename VisitorImpl>
void Walker<VisitorImpl>::Walk(VisitorImpl& visitor) {
Push(Frame::DoneSentinel());
Push(Frame(type_, start_.ToPosition()));
// Macro to insert the relevant goop required to support two control flows here in case of error:
// one where we keep reading after error, and another where we return immediately.
#define FIDL_STATUS_GUARD_IMPL(status, pop) \
switch ((status)) { \
case Status::kSuccess: \
break; \
case Status::kConstraintViolationError: \
if (VisitorImpl::kContinueAfterConstraintViolation) { \
if ((pop)) { \
Pop(); \
} \
continue; \
} else { \
return; \
} \
case Status::kMemoryError: \
return; \
}
#define FIDL_STATUS_GUARD(status) FIDL_STATUS_GUARD_IMPL(status, true)
#define FIDL_STATUS_GUARD_NO_POP(status) FIDL_STATUS_GUARD_IMPL(status, false)
for (;;) {
Frame* frame = Peek();
switch (frame->state) {
case Frame::kStateEnum: {
uint64_t value;
switch (frame->enum_state.underlying_type) {
case FidlCodedPrimitive::kUint8:
value = *PtrTo<uint8_t>(frame->position);
break;
case FidlCodedPrimitive::kUint16:
value = *PtrTo<uint16_t>(frame->position);
break;
case FidlCodedPrimitive::kUint32:
value = *PtrTo<uint32_t>(frame->position);
break;
case FidlCodedPrimitive::kUint64:
value = *PtrTo<uint64_t>(frame->position);
break;
case FidlCodedPrimitive::kInt8:
value = static_cast<uint64_t>(*PtrTo<int8_t>(frame->position));
break;
case FidlCodedPrimitive::kInt16:
value = static_cast<uint64_t>(*PtrTo<int16_t>(frame->position));
break;
case FidlCodedPrimitive::kInt32:
value = static_cast<uint64_t>(*PtrTo<int32_t>(frame->position));
break;
case FidlCodedPrimitive::kInt64:
value = static_cast<uint64_t>(*PtrTo<int64_t>(frame->position));
break;
default:
__builtin_unreachable();
}
if (!frame->enum_state.validate(value)) {
// TODO(FIDL-523): Make this strictness dependent.
visitor.OnError("not a valid enum member");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
Pop();
continue;
}
case Frame::kStateBits: {
uint64_t value;
switch (frame->bits_state.underlying_type) {
case FidlCodedPrimitive::kUint8:
value = *PtrTo<uint8_t>(frame->position);
break;
case FidlCodedPrimitive::kUint16:
value = *PtrTo<uint16_t>(frame->position);
break;
case FidlCodedPrimitive::kUint32:
value = *PtrTo<uint32_t>(frame->position);
break;
case FidlCodedPrimitive::kUint64:
value = *PtrTo<uint64_t>(frame->position);
break;
default:
__builtin_unreachable();
}
if (value & ~frame->bits_state.mask) {
visitor.OnError("not a valid bits member");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
Pop();
continue;
}
case Frame::kStateStruct: {
const uint32_t field_index = frame->NextStructField();
if (field_index == frame->struct_state.field_count) {
Pop();
continue;
}
const fidl::FidlStructField& field = frame->struct_state.fields[field_index];
const fidl_type_t* field_type = field.type;
Position field_position = frame->position + field.offset;
if (field.padding > 0) {
Position padding_position;
if (field_type) {
padding_position = field_position + TypeSize(field_type);
} else {
// Current type does not have coding information. |field.offset| stores the
// offset of the padding.
padding_position = field_position;
}
auto status = visitor.VisitInternalPadding(padding_position, field.padding);
FIDL_STATUS_GUARD(status);
}
if (!field_type) {
// Skip fields that do not contain codable types.
// Such fields only serve to provide padding information.
continue;
}
if (!Push(Frame(field_type, field_position))) {
visitor.OnError("recursion depth exceeded processing struct");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
continue;
}
case Frame::kStateStructPointer: {
if (*PtrTo<Ptr<void>>(frame->position) == nullptr) {
Pop();
continue;
}
auto status =
visitor.VisitPointer(frame->position, PtrTo<Ptr<void>>(frame->position),
frame->struct_pointer_state.struct_type->size, &frame->position);
FIDL_STATUS_GUARD(status);
const fidl::FidlCodedStruct* coded_struct = frame->struct_pointer_state.struct_type;
*frame = Frame(coded_struct, frame->position);
continue;
}
case Frame::kStateTable: {
auto& table_frame = frame->table_state;
// Utility to locate the position of the Nth-ordinal envelope header
auto envelope_position = [&frame](uint32_t ordinal) -> Position {
return frame->position + (ordinal - 1) * static_cast<uint32_t>(sizeof(fidl_envelope_t));
};
if (table_frame.ordinal == 0) {
// Process the vector part of the table
auto envelope_vector_ptr = PtrTo<fidl_vector_t>(frame->position);
if (envelope_vector_ptr->data == nullptr) {
visitor.OnError("Table data cannot be absent");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
uint32_t size;
if (mul_overflow(envelope_vector_ptr->count, sizeof(fidl_envelope_t), &size)) {
visitor.OnError("integer overflow calculating table size");
return;
}
auto status = visitor.VisitPointer(frame->position, &envelope_vector_ptr->data, size,
&frame->position);
FIDL_STATUS_GUARD(status);
table_frame.ordinal = 1;
table_frame.present_count = static_cast<uint32_t>(envelope_vector_ptr->count);
table_frame.inside_envelope = false;
continue;
}
if (table_frame.inside_envelope) {
// Leave the envelope that was entered during the last iteration
uint32_t last_ordinal = table_frame.ordinal - 1;
ZX_DEBUG_ASSERT(last_ordinal >= 1);
Position envelope_pos = envelope_position(last_ordinal);
auto envelope_ptr = PtrTo<fidl_envelope_t>(envelope_pos);
table_frame.inside_envelope = false;
auto status = visitor.LeaveEnvelope(envelope_pos, envelope_ptr);
FIDL_STATUS_GUARD(status);
}
if (table_frame.ordinal > table_frame.present_count) {
// Processed last stored field in table. Done with this table.
Pop();
continue;
}
const fidl::FidlTableField* known_field = nullptr;
if (table_frame.remaining_fields > 0) {
const fidl::FidlTableField* field = table_frame.field;
if (field->ordinal == table_frame.ordinal) {
known_field = field;
table_frame.field++;
table_frame.remaining_fields--;
}
}
Position envelope_pos = envelope_position(table_frame.ordinal);
auto envelope_ptr = PtrTo<fidl_envelope_t>(envelope_pos);
// Process the next ordinal in the following state machine iteration
table_frame.ordinal++;
// Make sure we don't process a malformed envelope
const fidl_type_t* payload_type = known_field ? known_field->type : nullptr;
auto status = visitor.EnterEnvelope(envelope_pos, envelope_ptr, payload_type);
FIDL_STATUS_GUARD(status);
table_frame.inside_envelope = true;
// Skip empty envelopes
if (envelope_ptr->data == nullptr) {
continue;
}
uint32_t num_bytes =
payload_type != nullptr ? TypeSize(payload_type) : envelope_ptr->num_bytes;
Position position;
status =
visitor.VisitPointer(frame->position,
// casting since |envelope_ptr->data| is always void*
&const_cast<Ptr<void>&>(envelope_ptr->data), num_bytes, &position);
// Do not pop the table frame, to guarantee calling |LeaveEnvelope|
FIDL_STATUS_GUARD_NO_POP(status);
if (payload_type != nullptr && !IsPrimitive(payload_type)) {
if (!Push(Frame(payload_type, position))) {
visitor.OnError("recursion depth exceeded processing table");
FIDL_STATUS_GUARD_NO_POP(Status::kConstraintViolationError);
}
}
continue;
}
case Frame::kStateUnion: {
auto union_tag = *PtrTo<fidl_union_tag_t>(frame->position);
if (union_tag >= frame->union_state.field_count) {
visitor.OnError("Bad union discriminant");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
auto variant = frame->union_state.fields[union_tag];
if (variant.padding > 0) {
Position padding_position =
frame->position + (frame->union_state.union_size - variant.padding);
auto status = visitor.VisitInternalPadding(padding_position, variant.padding);
FIDL_STATUS_GUARD(status);
}
auto data_offset = frame->union_state.data_offset;
ZX_DEBUG_ASSERT(data_offset == 4 || data_offset == 8);
if (data_offset == 8) {
// There is an additional 4 byte of padding after the tag.
auto status = visitor.VisitInternalPadding(frame->position + 4, 4);
FIDL_STATUS_GUARD(status);
}
const fidl_type_t* member = variant.type;
if (!member) {
Pop();
continue;
}
frame->position += data_offset;
*frame = Frame(member, frame->position);
continue;
}
case Frame::kStateUnionPointer: {
if (*PtrTo<Ptr<fidl_union_tag_t>>(frame->position) == nullptr) {
Pop();
continue;
}
auto status =
visitor.VisitPointer(frame->position, PtrTo<Ptr<void>>(frame->position),
frame->union_pointer_state.union_type->size, &frame->position);
FIDL_STATUS_GUARD(status);
const fidl::FidlCodedUnion* coded_union = frame->union_pointer_state.union_type;
*frame = Frame(coded_union, frame->position);
continue;
}
case Frame::kStateXUnion: {
auto xunion = PtrTo<fidl_xunion_t>(frame->position);
const auto envelope_pos = frame->position + offsetof(fidl_xunion_t, envelope);
auto envelope_ptr = &xunion->envelope;
// |inside_envelope| is always false when first encountering an xunion.
if (frame->xunion_state.inside_envelope) {
// Finished processing the xunion field, and is in clean-up state
auto status = visitor.LeaveEnvelope(envelope_pos, envelope_ptr);
FIDL_STATUS_GUARD(status);
Pop();
continue;
}
const auto padding_position = frame->position + offsetof(fidl_xunion_t, padding);
auto status = visitor.VisitInternalPadding(padding_position, sizeof(xunion->padding));
FIDL_STATUS_GUARD(status);
// Validate zero-ordinal invariants
if (xunion->tag == 0) {
if (envelope_ptr->data != nullptr || envelope_ptr->num_bytes != 0 ||
envelope_ptr->num_handles != 0) {
visitor.OnError("xunion with zero as ordinal must be empty");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
if (!frame->xunion_state.nullable) {
visitor.OnError("non-nullable xunion is absent");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
Pop();
continue;
}
// Find coding table corresponding to the ordinal via linear search
const FidlXUnionField* known_field = nullptr;
for (size_t i = 0; i < frame->xunion_state.field_count; i++) {
const auto field = frame->xunion_state.fields + i;
if (field->ordinal == xunion->tag) {
known_field = field;
break;
}
}
if (!known_field && frame->xunion_state.strictness == fidl::FidlStrictness::kStrict) {
visitor.OnError("strict xunion has unknown ordinal");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
// Make sure we don't process a malformed envelope
const fidl_type_t* payload_type = known_field ? known_field->type : nullptr;
status = visitor.EnterEnvelope(envelope_pos, envelope_ptr, payload_type);
FIDL_STATUS_GUARD(status);
frame->xunion_state.inside_envelope = true;
// Skip empty envelopes
if (envelope_ptr->data == nullptr) {
if (xunion->tag != 0) {
visitor.OnError("empty xunion must have zero as ordinal");
FIDL_STATUS_GUARD_NO_POP(Status::kConstraintViolationError);
}
continue;
}
uint32_t num_bytes =
payload_type != nullptr ? TypeSize(payload_type) : envelope_ptr->num_bytes;
Position position;
status = visitor.VisitPointer(frame->position, &const_cast<Ptr<void>&>(envelope_ptr->data),
num_bytes, &position);
FIDL_STATUS_GUARD_NO_POP(status);
if (payload_type != nullptr && !IsPrimitive(payload_type)) {
if (!Push(Frame(payload_type, position))) {
visitor.OnError("recursion depth exceeded processing xunion");
FIDL_STATUS_GUARD_NO_POP(Status::kConstraintViolationError);
}
}
continue;
}
case Frame::kStateArray: {
const uint32_t element_offset = frame->NextArrayOffset();
if (element_offset == frame->array_state.array_size) {
Pop();
continue;
}
const fidl_type_t* element_type = frame->array_state.element;
if (element_type) {
Position position = frame->position + element_offset;
if (!Push(Frame(element_type, position))) {
visitor.OnError("recursion depth exceeded processing array");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
} else {
// If there is no element type pointer, the array contents
// do not need extra processing, but the array coding table
// is present to provide size information when linearizing
// envelopes. Just continue.
Pop();
}
continue;
}
case Frame::kStateString: {
auto string_ptr = PtrTo<fidl_string_t>(frame->position);
if (string_ptr->data == nullptr) {
if (!frame->string_state.nullable &&
!VisitorImpl::kAllowNonNullableCollectionsToBeAbsent) {
visitor.OnError("non-nullable string is absent");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
if (string_ptr->size == 0) {
if (frame->string_state.nullable ||
!VisitorImpl::kAllowNonNullableCollectionsToBeAbsent) {
Pop();
continue;
}
} else {
visitor.OnError("string is absent but length is not zero");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
}
uint64_t bound = frame->string_state.max_size;
uint64_t size = string_ptr->size;
if (size > std::numeric_limits<uint32_t>::max()) {
visitor.OnError("string size overflows 32 bits");
FIDL_STATUS_GUARD(Status::kMemoryError);
}
if (size > bound) {
visitor.OnError("message tried to access too large of a bounded string");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
Position position;
auto status = visitor.VisitPointer(
position, &reinterpret_cast<Ptr<void>&>(const_cast<Ptr<char>&>(string_ptr->data)),
static_cast<uint32_t>(size), &position);
FIDL_STATUS_GUARD(status);
Pop();
continue;
}
case Frame::kStateHandle: {
auto handle_ptr = PtrTo<zx_handle_t>(frame->position);
if (*handle_ptr == ZX_HANDLE_INVALID) {
if (!frame->handle_state.nullable) {
visitor.OnError("message is missing a non-nullable handle");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
Pop();
continue;
}
auto status = visitor.VisitHandle(frame->position, handle_ptr);
FIDL_STATUS_GUARD(status);
Pop();
continue;
}
case Frame::kStateVector: {
auto vector_ptr = PtrTo<fidl_vector_t>(frame->position);
if (vector_ptr->data == nullptr) {
if (!frame->vector_state.nullable &&
!VisitorImpl::kAllowNonNullableCollectionsToBeAbsent) {
visitor.OnError("non-nullable vector is absent");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
if (vector_ptr->count == 0) {
if (frame->vector_state.nullable ||
!VisitorImpl::kAllowNonNullableCollectionsToBeAbsent) {
Pop();
continue;
}
} else {
visitor.OnError("absent vector of non-zero elements");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
}
if (vector_ptr->count > frame->vector_state.max_count) {
visitor.OnError("message tried to access too large of a bounded vector");
FIDL_STATUS_GUARD(Status::kConstraintViolationError);
}
uint32_t size;
if (mul_overflow(vector_ptr->count, frame->vector_state.element_size, &size)) {
visitor.OnError("integer overflow calculating vector size");
return;
}
auto status =
visitor.VisitPointer(frame->position, &vector_ptr->data, size, &frame->position);
FIDL_STATUS_GUARD(status);
if (frame->vector_state.element) {
// Continue by visiting the vector elements as an array.
*frame = Frame(frame->vector_state.element, size, frame->vector_state.element_size,
frame->position);
} else {
// If there is no element type pointer, there is
// nothing to process in the vector secondary
// payload. So just continue.
Pop();
}
continue;
}
case Frame::kStateDone: {
return;
}
}
}
}
} // namespace internal
// Walks the FIDL message, calling hooks in the concrete VisitorImpl.
//
// |visitor| is an implementation of the fidl::Visitor interface.
// |type| is the coding table for the FIDL type. It cannot be null.
// |start| is the starting point for the walk.
template <typename VisitorImpl>
void Walk(VisitorImpl& visitor, const fidl_type_t* type,
typename VisitorImpl::StartingPoint start) {
internal::Walker<VisitorImpl> walker(type, start);
walker.Walk(visitor);
}
// Infer the size of the primary object, from the coding table in |type|.
// Ensures that the primary object is of one of the expected types.
//
// An error is returned if:
// - |type| is null
// - The primary object is neither a struct nor a table.
zx_status_t PrimaryObjectSize(const fidl_type_t* type, size_t* out_size, const char** out_error);
// Calculate the offset of the first out-of-line object, from the coding table in |type|.
// Ensures that the primary object is of one of the expected types, and the offset falls within the
// |buffer_size| constraints.
//
// An error is returned if:
// - |type| is null
// - The primary object is neither a struct nor a table.
// - The offset overflows, or is larger than |buffer_size|.
zx_status_t StartingOutOfLineOffset(const fidl_type_t* type, uint32_t buffer_size,
uint32_t* out_first_out_of_line, const char** out_error);
} // namespace fidl
#endif // LIB_FIDL_WALKER_H_