2014-12-11 10:58:56 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
*
|
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
* directory for more details.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @defgroup net_ng_pktbuf Packet buffer
|
|
|
|
* @ingroup net
|
|
|
|
* @brief A global network packet buffer.
|
|
|
|
*
|
|
|
|
* @note **WARNING!!** Do not store data structures that are not packed
|
|
|
|
* (defined with `__attribute__((packed))`) or enforce alignment in
|
|
|
|
* in any way in here if @ref NG_PKTBUF_SIZE > 0. On some RISC architectures
|
|
|
|
* this *will* lead to alignment problems and can potentially result
|
|
|
|
* in segmentation/hard faults and other unexpected behaviour.
|
|
|
|
*
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Interface definition for the global network buffer. Network devices
|
|
|
|
* and layers can allocate space for packets here.
|
|
|
|
*
|
|
|
|
* @author Martine Lenders <mlenders@inf.fu-berlin.de>
|
|
|
|
*/
|
|
|
|
#ifndef NG_PKTBUF_H_
|
|
|
|
#define NG_PKTBUF_H_
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "atomic.h"
|
2015-05-22 14:31:23 +02:00
|
|
|
#include "cpu_conf.h"
|
2014-12-11 10:58:56 +01:00
|
|
|
#include "net/ng_pkt.h"
|
|
|
|
#include "net/ng_nettype.h"
|
2015-04-14 02:06:03 +02:00
|
|
|
#include "utlist.h"
|
2014-12-11 10:58:56 +01:00
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @def NG_PKTBUF_SIZE
|
|
|
|
* @brief Maximum size of the static packet buffer.
|
|
|
|
*
|
|
|
|
* @details The rational here is to have at least space for 4 full-MTU IPv6
|
|
|
|
* packages (2 incoming, 2 outgoing; 2 * 2 * 1280 B = 5 KiB) +
|
|
|
|
* Meta-Data (roughly estimated to 1 KiB; might be smaller). If
|
|
|
|
* @ref NG_PKTBUF_SIZE is 0 the packet buffer will use dynamic memory
|
|
|
|
* management to allocate packets.
|
|
|
|
*/
|
|
|
|
#ifndef NG_PKTBUF_SIZE
|
|
|
|
#define NG_PKTBUF_SIZE (6144)
|
|
|
|
#endif /* NG_PKTBUF_SIZE */
|
|
|
|
|
|
|
|
/**
|
2015-03-09 13:37:35 +01:00
|
|
|
* @brief Adds a new ng_pktsnip_t and its packet to the packet buffer.
|
|
|
|
*
|
|
|
|
* @details This function is very powerful and reflects the unique characterics
|
|
|
|
* of ng_pktsnip_t of being reversed for either the sending or
|
|
|
|
* receiving context. Because of this the assumtion of the transmission
|
|
|
|
* direction, the state of the packet buffer and the values for the
|
|
|
|
* members of the resulting ng_pktsnip_t can be very different after
|
|
|
|
* execution of this function depending on what parameters you use:
|
|
|
|
*
|
2015-04-22 18:44:31 +02:00
|
|
|
* * The return value of this function is a @ref ng_pktsnip_t struct referred
|
|
|
|
* to as `result`
|
|
|
|
* * for most cases the build-up of `result` will be pretty straight forward: Its
|
|
|
|
* members will be exactly as the given parameters (ng_pktsnip_t::next of
|
|
|
|
* result will be set to @p pkt). If @p pkt is not NULL it and in turn `result`
|
|
|
|
* are assumed to be in sending direction. For packet creation (@p pkt == NULL)
|
|
|
|
* no assumtions about direction of `result` will be made (since its
|
|
|
|
* ng_pktsnip::next will be set to NULL).
|
2015-03-09 13:37:35 +01:00
|
|
|
* * if @p pkt != NULL, @p data = `pkt->data`, @p size < `pkt->size` receiving
|
|
|
|
* direction is assumed and the following values will be set:
|
2015-04-22 18:44:31 +02:00
|
|
|
* * ng_pktsnip_t::next of `result` = `pkt->next`
|
|
|
|
* * ng_pktsnip_t::data of `result` = @p data
|
|
|
|
* * ng_pktsnip_t::size of `result` = @p size
|
|
|
|
* * ng_pktsnip_t::next of @p pkt = `result`
|
2015-03-09 13:37:35 +01:00
|
|
|
* * ng_pktsnip_t::data of @p pkt = @p data + @p size
|
|
|
|
* * ng_pktsnip_t::size of @p pkt = old size value - @p size
|
|
|
|
*
|
2015-04-22 18:44:31 +02:00
|
|
|
* graphically this can be represented as follows:
|
|
|
|
* @code
|
2015-03-09 13:37:35 +01:00
|
|
|
* Before After
|
|
|
|
* ====== =====
|
|
|
|
* (next)
|
|
|
|
* pkt->data result->data <== pkt->data
|
|
|
|
* v v v
|
|
|
|
* +--------------------------------+ +----------------+---------------+
|
|
|
|
* +--------------------------------+ +----------------+---------------+
|
|
|
|
* \__________pkt->size___________/ \_result->size_/ \__pkt->size__/
|
2015-04-22 18:44:31 +02:00
|
|
|
* @endcode
|
2015-03-09 13:37:35 +01:00
|
|
|
*
|
|
|
|
* @note **Do not** change the ng_pktsnip_t::data and ng_pktsnip_t::size
|
|
|
|
* of a ng_pktsnip_t created by this function externally, except if
|
|
|
|
* they both are null or data is not from inside the packet buffer.
|
|
|
|
* This will most likely create memory leaks.
|
|
|
|
*
|
|
|
|
* @param[in,out] pkt The packet you want to add a ng_pktsnip_t to. Leave
|
|
|
|
* NULL if you want to create a new packet. Members may
|
|
|
|
* change values; see above.
|
|
|
|
* @param[in] data Data of the new ng_pktsnip_t. If @p data is NULL no data
|
2015-04-22 18:44:31 +02:00
|
|
|
* will be inserted into `result`. If @p data is already
|
2015-03-09 13:37:35 +01:00
|
|
|
* in the packet buffer (e.g. a payload of an already
|
2015-04-22 18:44:31 +02:00
|
|
|
* allocated packet) it will not be duplicated.
|
2015-03-09 13:37:35 +01:00
|
|
|
* @param[in] size Length of @p data. If @p size is 0 no data will be inserted
|
|
|
|
* into the the packet buffer and ng_pktsnip_t::data will be
|
|
|
|
* set to @p data.
|
|
|
|
* @param[in] type Protocol type of the ng_pktsnip_t.
|
2014-12-11 10:58:56 +01:00
|
|
|
*
|
|
|
|
* @return Pointer to the packet part that represents the new ng_pktsnip_t.
|
|
|
|
* @return NULL, if no space is left in the packet buffer.
|
2015-05-26 16:45:16 +02:00
|
|
|
* @return NULL, if @p pkt != NULL, @p data = `pkt->data`,
|
|
|
|
* and @p size > `pkt->data`.
|
2014-12-11 10:58:56 +01:00
|
|
|
*/
|
2015-03-09 13:37:35 +01:00
|
|
|
ng_pktsnip_t *ng_pktbuf_add(ng_pktsnip_t *pkt, void *data, size_t size,
|
2014-12-11 10:58:56 +01:00
|
|
|
ng_nettype_t type);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Reallocates ng_pktsnip_t::data of @p pkt in the packet buffer, without
|
|
|
|
* changing the content.
|
|
|
|
*
|
|
|
|
* @pre `pkt->users == 1 && pkt->next == NULL` and @p pkt must be in packet buffer
|
|
|
|
*
|
|
|
|
* @details If enough memory is available behind it or @p size is smaller than
|
|
|
|
* the original size the packet then ng_pktsnip_t::data of @p pkt will
|
|
|
|
* not be moved. Otherwise, it will be moved. If no space is available
|
|
|
|
* nothing happens.
|
|
|
|
*
|
2015-03-09 13:37:35 +01:00
|
|
|
* @param[in] pkt A packet part.
|
|
|
|
* @param[in] size The size for @p pkt.
|
2014-12-11 10:58:56 +01:00
|
|
|
*
|
|
|
|
* @return 0, on success
|
|
|
|
* @return EINVAL, if precondition is not met
|
|
|
|
* @return ENOENT, if ng_pktsnip_t::data of @p pkt was not from the packet buffer.
|
|
|
|
* @return ENOMEM, if no space is left in the packet buffer or size was 0.
|
|
|
|
*/
|
|
|
|
int ng_pktbuf_realloc_data(ng_pktsnip_t *pkt, size_t size);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Increases ng_pktsnip_t::users of @p pkt atomically.
|
|
|
|
*
|
|
|
|
* @param[in] pkt A packet.
|
|
|
|
* @param[in] num Number you want to increment ng_pktsnip_t::users of @p pkt by.
|
|
|
|
*/
|
2015-03-09 13:37:35 +01:00
|
|
|
void ng_pktbuf_hold(ng_pktsnip_t *pkt, unsigned int num);
|
2014-12-11 10:58:56 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Decreases ng_pktsnip_t::users of @p pkt atomically and removes it if it
|
|
|
|
* reaches 0.
|
|
|
|
*
|
|
|
|
* @param[in] pkt A packet.
|
|
|
|
*/
|
|
|
|
void ng_pktbuf_release(ng_pktsnip_t *pkt);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Must be called once before there is a write operation in a thread.
|
|
|
|
*
|
|
|
|
* @details This function duplicates a packet in the packet buffer if
|
|
|
|
* ng_pktsnip_t::users of @p pkt > 1.
|
|
|
|
*
|
|
|
|
* @note Do *not* call this function in a thread twice on the same packet.
|
|
|
|
*
|
|
|
|
* @param[in] pkt The packet you want to write into.
|
|
|
|
*
|
|
|
|
* @return The (new) pointer to the pkt.
|
|
|
|
* @return NULL, if ng_pktsnip_t::users of @p pkt > 1 and if there is not anough
|
|
|
|
* space in the packet buffer.
|
|
|
|
*/
|
|
|
|
ng_pktsnip_t *ng_pktbuf_start_write(ng_pktsnip_t *pkt);
|
|
|
|
|
2015-04-14 02:06:03 +02:00
|
|
|
/**
|
|
|
|
* @brief Deletes a snip from a packet and the packet buffer.
|
|
|
|
*
|
|
|
|
* @param[in] pkt A packet.
|
|
|
|
* @param[in] snip A snip in the packet.
|
2015-04-14 13:46:32 +02:00
|
|
|
*
|
|
|
|
* @return The new reference to @p pkt.
|
2015-04-14 02:06:03 +02:00
|
|
|
*/
|
2015-04-14 13:46:32 +02:00
|
|
|
static inline ng_pktsnip_t *ng_pktbuf_remove_snip(ng_pktsnip_t *pkt,
|
|
|
|
ng_pktsnip_t *snip)
|
2015-04-14 02:06:03 +02:00
|
|
|
{
|
|
|
|
LL_DELETE(pkt, snip);
|
|
|
|
snip->next = NULL;
|
|
|
|
ng_pktbuf_release(snip);
|
2015-04-14 13:46:32 +02:00
|
|
|
|
|
|
|
return pkt;
|
2015-04-14 02:06:03 +02:00
|
|
|
}
|
|
|
|
|
2014-12-11 10:58:56 +01:00
|
|
|
#ifdef DEVELHELP
|
|
|
|
/**
|
|
|
|
* @brief Prints some statistics about the packet buffer to stdout.
|
|
|
|
*
|
|
|
|
* @note Only available with DEVELHELP defined.
|
|
|
|
*
|
|
|
|
* @details Statistics include maximum number of reserved bytes.
|
|
|
|
*/
|
|
|
|
void ng_pktbuf_stats(void);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* for testing */
|
|
|
|
#ifdef TEST_SUITES
|
|
|
|
/**
|
|
|
|
* @brief Checks if packet buffer is empty
|
|
|
|
*
|
|
|
|
* @return true, if packet buffer is empty
|
|
|
|
* @return false, if packet buffer is not empty
|
|
|
|
*/
|
|
|
|
bool ng_pktbuf_is_empty(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Resets the whole packet buffer
|
|
|
|
*/
|
|
|
|
void ng_pktbuf_reset(void);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* NG_PKTBUF_H_ */
|
|
|
|
/** @} */
|