blob: 48dc0a4c349da93b5e3024fe4982642b8a8e6c0b [file] [log] [blame]
:: # Copyright 2013, Big Switch Networks, Inc.
:: #
:: # LoxiGen is licensed under the Eclipse Public License, version 1.0 (EPL), with
:: # the following special exception:
:: #
:: # LOXI Exception
:: #
:: # As a special exception to the terms of the EPL, you may distribute libraries
:: # generated by LoxiGen (LoxiGen Libraries) under the terms of your choice, provided
:: # that copyright and licensing notices generated by LoxiGen are not altered or removed
:: # from the LoxiGen Libraries and the notice provided below is (i) included in
:: # the LoxiGen Libraries, if distributed in source code form and (ii) included in any
:: # documentation for the LoxiGen Libraries, if distributed in binary form.
:: #
:: # Notice: "Copyright 2013, Big Switch Networks, Inc. This library was generated by the LoxiGen Compiler."
:: #
:: # You may not use this file except in compliance with the EPL or LOXI Exception. You may obtain
:: # a copy of the EPL at:
:: #
:: # http://www.eclipse.org/legal/epl-v10.html
:: #
:: # Unless required by applicable law or agreed to in writing, software
:: # distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
:: # WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
:: # EPL for the specific language governing permissions and limitations
:: # under the EPL.
::
:: include('_copyright.c')
/****************************************************************
*
* of_object.c
*
* These are the low level object constructor/destructor operators.
*
****************************************************************/
#include "loci_log.h"
#include <loci/loci.h>
#include <loci/loci_validator.h>
/**
* Create a generic new object and possibly underlying wire buffer
* @param bytes The number of bytes to allocate in the underlying buffer
*
* If bytes <= 0, do not allocate a wire buffer.
*
* Note that this is an internal function. The class specific
* new functions should be called to properly initialize and track an
* OF object.
*/
of_object_t *
of_object_new(int bytes)
{
of_object_t *obj;
if ((obj = (of_object_t *)MALLOC(sizeof(*obj))) == NULL) {
return NULL;
}
MEMSET(obj, 0, sizeof(*obj));
if (bytes > 0) {
if ((obj->wbuf = of_wire_buffer_new(bytes)) == NULL) {
FREE(obj);
return NULL;
}
}
return obj;
}
/**
* The delete function for LOCI objects
*
* @param obj Pointer to the object to be deleted
*
* This can be called on any LOCI object; it should not need to be
* overridden.
*/
void
of_object_delete(of_object_t *obj)
{
if (obj == NULL) {
return;
}
if (obj->parent == NULL) {
of_wire_buffer_free(obj->wbuf);
}
FREE(obj);
}
/**
* Duplicate an object
* @param src The object to be duplicated
* @returns Pointer to the duplicate or NULL on error. Caller is responsible
* for freeing the returned object.
*/
of_object_t *
of_object_dup(of_object_t *src)
{
of_object_t *dst;
of_object_init_f init_fn;
if ((dst = (of_object_t *)MALLOC(sizeof(*dst))) == NULL) {
return NULL;
}
MEMSET(dst, 0, sizeof(*dst));
/* Allocate a minimal wire buffer assuming we will not write to it. */
if ((dst->wbuf = of_wire_buffer_new(src->length)) == NULL) {
FREE(dst);
return NULL;
}
init_fn = of_object_init_map[src->object_id];
init_fn(dst, src->version, src->length, 0);
MEMCPY(OF_OBJECT_BUFFER_INDEX(dst, 0),
OF_OBJECT_BUFFER_INDEX(src, 0),
src->length);
return dst;
}
/**
* Generic new from message call
*/
of_object_t *
of_object_new_from_message(of_message_t msg, int len)
{
of_object_id_t object_id;
of_object_t *obj;
of_version_t version;
version = of_message_version_get(msg);
if (!OF_VERSION_OKAY(version)) {
return NULL;
}
if (of_validate_message(msg, len) != 0) {
LOCI_LOG_ERROR("message validation failed\n");
return NULL;
}
if ((obj = of_object_new(-1)) == NULL) {
return NULL;
}
if (of_object_buffer_bind(obj, OF_MESSAGE_TO_BUFFER(msg), len,
OF_MESSAGE_FREE_FUNCTION) < 0) {
FREE(obj);
return NULL;
}
obj->version = version;
of_header_wire_object_id_get(obj, &object_id);
of_object_init_map[object_id](obj, version, len, 0);
return obj;
}
/**
* Parse a message without allocating memory
*
* @param storage Pointer to an uninitialized of_object_storage_t
* @param buf Pointer to the buffer
* @param length Length of buf
* @returns Pointer to an initialized of_object_t
*
* The lifetime of the returned object is the minimum of the lifetimes of
* 'buf' and 'storage'.
*/
of_object_t *
of_object_new_from_message_preallocated(of_object_storage_t *storage,
uint8_t *buf, int len)
{
of_object_t *obj = &storage->obj;
of_wire_buffer_t *wbuf = &storage->wbuf;
of_message_t msg = buf;
of_version_t version;
of_object_id_t object_id;
memset(storage, 0, sizeof(*storage));
version = of_message_version_get(msg);
if (!OF_VERSION_OKAY(version)) {
return NULL;
}
if (of_validate_message(msg, len) != 0) {
LOCI_LOG_ERROR("message validation failed\n");
return NULL;
}
obj->version = version;
obj->wbuf = wbuf;
wbuf->buf = msg;
wbuf->alloc_bytes = len;
wbuf->current_bytes = len;
of_header_wire_object_id_get(obj, &object_id);
of_object_init_map[object_id](obj, version, len, 0);
return obj;
}
/**
* Bind an existing buffer to an LOCI object
*
* @param obj Pointer to the object to be updated
* @param buf Pointer to the buffer to bind to obj
* @param bytes Length of buf
* @param buf_free An optional free function to be applied to
* buf on deallocation
*
* This can be called on any LOCI object; it should not need to be
* overridden.
*/
int
of_object_buffer_bind(of_object_t *obj, uint8_t *buf, int bytes,
of_buffer_free_f buf_free)
{
of_wire_buffer_t *wbuf;
LOCI_ASSERT(buf != NULL);
LOCI_ASSERT(bytes > 0);
wbuf = of_wire_buffer_new_bind(buf, bytes, buf_free);
if (wbuf == NULL) {
return OF_ERROR_RESOURCE;
}
obj->wbuf = wbuf;
obj->obj_offset = 0;
obj->length = bytes;
return OF_ERROR_NONE;
}
/**
* Connect a child to a parent at the wire buffer level
*
* @param parent The top level object to bind to
* @param child The sub-object connecting to the parent
* @param offset The offset at which to attach the child RELATIVE
* TO THE PARENT in the buffer
* @param bytes The amount of the buffer dedicated to the child; see below
*
* This is used for 'get' accessors for composite types as well as
* iterator functions for lists, both read (first/next) and write
* (append_init, append_advance).
*
* Connect a child object to a parent by setting up the child's
* wire_object to point to the parent's underlying buffer. The value
* of the parameter bytes is important in determining how the child
* is initialized:
* @li If bytes <= 0, the length and type of the child are not modified;
* no additional space is added to the buffer.
* @li If bytes > 0, the current wire buffer is grown to
* accomodate this many bytes. This is to support append operations.
*
* If an error is returned, future references to the child object
* (until it is reinitialized) are undefined.
*/
static void
object_child_attach(of_object_t *parent, of_object_t *child,
int offset, int bytes)
{
of_object_attach(parent, child, offset, bytes);
/*
* bytes determines if this is a read or write setup.
* If > 0, grow the buffer to accomodate the space
* Otherwise do nothing
*/
if (bytes > 0) { /* Set internal length, request buffer space */
int tot_bytes; /* Total bytes to request for buffer if updated */
/* Set up space for the child in the parent's buffer */
tot_bytes = parent->obj_offset + offset + bytes;
of_wire_buffer_grow(parent->wbuf, tot_bytes);
}
/* if bytes == 0 don't do anything */
}
/**
* Check for room in an object's wire buffer.
* @param obj The object being checked
* @param new_len The desired length
* @return Boolean
*/
int
of_object_can_grow(of_object_t *obj, int new_len)
{
return OF_OBJECT_ABSOLUTE_OFFSET(obj, new_len) <=
WBUF_ALLOC_BYTES(obj->wbuf);
}
/**
* Set the xid of a message object
* @param obj The object being accessed
* @param xid The xid value to store in the wire buffer
* @return OF_ERROR_
* Since the XID is common across all versions, this is used
* for all XID accessors.
*/
int
of_object_xid_set(of_object_t *obj, uint32_t xid)
{
of_wire_buffer_t *wbuf;
if ((wbuf = OF_OBJECT_TO_WBUF(obj)) == NULL) {
return OF_ERROR_PARAM;
}
of_wire_buffer_u32_set(wbuf,
OF_OBJECT_ABSOLUTE_OFFSET(obj, OF_MESSAGE_XID_OFFSET), xid);
return OF_ERROR_NONE;
}
/**
* Get the xid of a message object
* @param obj The object being accessed
* @param xid Pointer to where to store the xid value
* @return OF_ERROR_
* Since the XID is common across all versions, this is used
* for all XID accessors.
*/
int
of_object_xid_get(of_object_t *obj, uint32_t *xid)
{
of_wire_buffer_t *wbuf;
if ((wbuf = OF_OBJECT_TO_WBUF(obj)) == NULL) {
return OF_ERROR_PARAM;
}
of_wire_buffer_u32_get(wbuf,
OF_OBJECT_ABSOLUTE_OFFSET(obj, OF_MESSAGE_XID_OFFSET), xid);
return OF_ERROR_NONE;
}
/****************************************************************
*
* Generic list operation implementations
*
****************************************************************/
/**
* Set up a child for appending to a parent list
* @param parent The parent; must be a list object
* @param child The child object; must be of type list element
* @return OF_ERROR_
*
* Attaches the wire buffer of the parent to the child by pointing
* the child to the end of the parent.
*
* Set the wire length and type from the child.
* Update the parent length adding the current child length
*
* After calling this function, the child object may be updated
* resulting in changes to the parent's wire buffer
*
*/
int
of_list_append_bind(of_object_t *parent, of_object_t *child)
{
if (parent == NULL || child == NULL ||
parent->wbuf == NULL) {
return OF_ERROR_PARAM;
}
if (!of_object_can_grow(parent, parent->length + child->length)) {
return OF_ERROR_RESOURCE;
}
object_child_attach(parent, child, parent->length,
child->length);
/* Update the wire length and type if needed */
of_object_wire_length_set(child, child->length);
of_object_wire_type_set(child);
/* Update the parent's length */
of_object_parent_length_update(parent, child->length);
OF_LENGTH_CHECK_ASSERT(parent);
return OF_ERROR_NONE;
}
/**
* Generic atomic list append operation
* @param list The list to which an item is being appended
* @param item THe item to append to the list
*
* The contents of the item are copied to the end of the list.
* Currently assumes the list is at the end of its parent.
*/
int
of_list_append(of_object_t *list, of_object_t *item)
{
int new_len;
new_len = list->length + item->length;
if (!of_object_can_grow(list, new_len)) {
return OF_ERROR_RESOURCE;
}
of_wire_buffer_grow(list->wbuf,
OF_OBJECT_ABSOLUTE_OFFSET(list, new_len));
MEMCPY(OF_OBJECT_BUFFER_INDEX(list, list->length),
OF_OBJECT_BUFFER_INDEX(item, 0), item->length);
/* Update the list's length */
of_object_parent_length_update(list, item->length);
OF_LENGTH_CHECK_ASSERT(list);
return OF_ERROR_NONE;
}
/**
* Generic list first function
* @param parent The parent; must be a list object
* @param child The child object; must be of type list element
* @return OF_ERROR_RANGE if list is empty
* @return OF_ERROR_
*
* Sets up the child to point to the first element in the list
*
* Child init must be called before this is called.
*
* @note TREAT AS PRIVATE
* Does not fully initialized object
*/
int
of_list_first(of_object_t *parent, of_object_t *child)
{
if (parent->length == 0) { /* Empty list */
return OF_ERROR_RANGE;
}
of_object_attach(parent, child, 0, child->length);
return OF_ERROR_NONE;
}
/**
* Return boolean indicating if child is pointing to last entry in parent
* @param parent The parent; must be a list object
* @param child The child object; must be of type list element
* @return OF_ERROR_RANGE if list is empty
* @return OF_ERROR_
*
*/
static int
of_list_is_last(of_object_t *parent, of_object_t *child)
{
if (child->obj_offset + child->length >=
parent->obj_offset + parent->length) {
return 1;
}
return 0;
}
/**
* Generic list next function
* @param parent The parent; must be a list object
* @param child The child object; must be of type list element
* @return OF_ERROR_RANGE if at end of list
* @return OF_ERROR_
*
* Advances the child to point to the subsequent element in the list.
* The wire buffer object must not have been modified since the
* previous call to _first or _next.
*
* @note TREAT AS PRIVATE
* Does not fully initialized object
*/
int
of_list_next(of_object_t *parent, of_object_t *child)
{
int offset;
LOCI_ASSERT(child->length > 0);
/* Get offset of parent */
if (of_list_is_last(parent, child)) {
return OF_ERROR_RANGE; /* We were on the last object */
}
/* Offset is relative to parent start */
offset = (child->obj_offset - parent->obj_offset) +
child->length;
of_object_attach(parent, child, offset, child->length);
return OF_ERROR_NONE;
}
void
of_object_wire_buffer_steal(of_object_t *obj, uint8_t **buffer)
{
LOCI_ASSERT(obj != NULL);
of_wire_buffer_steal(obj->wbuf, buffer);
obj->wbuf = NULL;
}
#define _MAX_PARENT_ITERATIONS 8
/**
* Iteratively update parent lengths thru hierarchy
* @param obj The object whose length is being updated
* @param delta The difference between the current and new lengths
*
* Note that this includes updating the object itself. It will
* iterate thru parents.
*
* Assumes delta > 0.
*/
void
of_object_parent_length_update(of_object_t *obj, int delta)
{
#ifndef NDEBUG
int count = 0;
of_wire_buffer_t *wbuf; /* For debug asserts only */
#endif
while (obj != NULL) {
LOCI_ASSERT(count++ < _MAX_PARENT_ITERATIONS);
obj->length += delta;
of_object_wire_length_set(obj, obj->length);
#ifndef NDEBUG
wbuf = obj->wbuf;
#endif
/* Asserts for wire length checking */
LOCI_ASSERT(obj->length + obj->obj_offset <=
WBUF_CURRENT_BYTES(wbuf));
if (obj->parent == NULL) {
LOCI_ASSERT(obj->length + obj->obj_offset ==
WBUF_CURRENT_BYTES(wbuf));
}
obj = obj->parent;
}
}
/**
* Use the type/length from the wire buffer and init the object
* @param obj The object being initialized
* @param base_object_id If > 0, this indicates the base object
* @param max_len If > 0, the max length to expect for the obj
* type for inheritance checking
* @return OF_ERROR_
*
* Used for inheritance type objects such as actions and OXMs
* The type is checked and if valid, the object is initialized.
* Then the length is taken from the buffer.
*
* Note that the object version must already be properly set.
*/
int
of_object_wire_init(of_object_t *obj, of_object_id_t base_object_id,
int max_len)
{
if (loci_class_metadata[obj->object_id].wire_type_get != NULL) {
of_object_id_t id;
loci_class_metadata[obj->object_id].wire_type_get(obj, &id);
obj->object_id = id;
/* Call the init function for this object type; do not push to wire */
of_object_init_map[id]((of_object_t *)(obj), obj->version, -1, 0);
}
if (loci_class_metadata[obj->object_id].wire_length_get != NULL) {
int length;
loci_class_metadata[obj->object_id].wire_length_get(obj, &length);
if (length < 0 || (max_len > 0 && length > max_len)) {
return OF_ERROR_PARSE;
}
obj->length = length;
} else {
/* @fixme Does this cover everything else? */
obj->length = of_object_fixed_len[obj->version][base_object_id];
}
return OF_ERROR_NONE;
}
/*
* Truncate an object to its initial length.
*
* This allows the caller to reuse a single allocated object even if
* it has been appended to.
*/
void
of_object_truncate(of_object_t *obj)
{
of_object_init_map[obj->object_id](obj, obj->version, -1, 0);
obj->wbuf->current_bytes = obj->length;
of_wire_length_set_f wire_length_set = loci_class_metadata[obj->object_id].wire_length_set;
if (wire_length_set != NULL) {
wire_length_set(obj, obj->length);
}
}