1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
RIOT/sys/include/net/gnrc/ipv6/ext/frag.h

196 lines
6.2 KiB
C

/*
* Copyright (C) 2019 Freie Universität Berlin
*
* 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_gnrc_ipv6_ext_frag Support for IPv6 fragmentation extension
* @ingroup net_gnrc_ipv6_ext
* @brief GNRC implementation of IPv6 fragmentation extension
* @{
*
* @file
* @brief GNRC fragmentation extension definitions
*
* @author Martine Lenders <m.lenders@fu-berlin.de>
*/
#ifndef NET_GNRC_IPV6_EXT_FRAG_H
#define NET_GNRC_IPV6_EXT_FRAG_H
#include <stdbool.h>
#include <stdint.h>
#include "clist.h"
#include "net/gnrc/pkt.h"
#include "net/gnrc/pktbuf.h"
#include "net/ipv6/hdr.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Message type to time reassembly buffer garbage collection
*/
#define GNRC_IPV6_EXT_FRAG_RBUF_GC (0xfe00U)
/**
* @brief Message type to continue fragmenting a datagram from a given
* fragmentation send buffer
*
* Expected type: @ref gnrc_ipv6_ext_frag_send_t
*/
#define GNRC_IPV6_EXT_FRAG_CONTINUE (0xfe01U)
/**
* @brief Message type to send a fragment of an IPv6 datagram.
*
* Expected type: @ref gnrc_pktsnip_t
*/
#define GNRC_IPV6_EXT_FRAG_SEND (0xfe02U)
/**
* @brief Data type to describe limits of a single fragment in the reassembly
* buffer
*/
typedef struct gnrc_ipv6_ext_frag_limits {
struct gnrc_ipv6_ext_frag_limits *next; /**< limits of next fragment */
uint16_t start; /**< the start (= offset) of the fragment */
uint16_t end; /**< the exclusive end (= offset + length) of the
* fragment */
} gnrc_ipv6_ext_frag_limits_t;
/**
* @brief Fragmentation send buffer type
*/
typedef struct {
gnrc_pktsnip_t *pkt; /**< the IPv6 packet to fragment */
gnrc_pktsnip_t *per_frag; /**< per fragment headers */
uint32_t id; /**< the identification for the fragment header */
uint16_t path_mtu; /**< path MTU to destination of
* gnrc_ipv6_ext_frag_send_t::pkt */
uint16_t offset; /**< current fragmentation offset */
} gnrc_ipv6_ext_frag_send_t;
/**
* @brief A reassembly buffer entry
*/
typedef struct {
gnrc_pktsnip_t *pkt; /**< the (partly) reassembled packet */
ipv6_hdr_t *ipv6; /**< the IPv6 header of gnrc_ipv6_ext_frag_rbuf_t::pkt */
/**
* @brief The limits of the fragments in the reassembled packet
*
* @note Members of this list can be cast to gnrc_ipv6_ext_frag_limits_t.
*/
clist_node_t limits;
uint32_t id; /**< the identification from the fragment headers */
uint32_t arrival; /**< arrival time of last received fragment */
uint16_t pkt_len; /**< length of gnrc_ipv6_ext_frag_rbuf_t::pkt */
uint8_t last; /**< received last fragment */
} gnrc_ipv6_ext_frag_rbuf_t;
/**
* @brief Initializes IPv6 fragmentation and reassembly
* @internal
*/
void gnrc_ipv6_ext_frag_init(void);
/**
* @brief Send an IPv6 packet fragmented
*
* @param[in] pkt The IPv6 packet. The packet must have an already
* prepared @ref GNRC_NETTYPE_NETIF snip as its first
* snip. The packet must contain at least an IPv6 header
* and any number of IPv6 extension headers after that.
* @param[in] path_mtu Path MTU to destination of IPv6 packet.
*/
void gnrc_ipv6_ext_frag_send_pkt(gnrc_pktsnip_t *pkt, unsigned path_mtu);
/**
* @brief (Continue to) fragment packet already in fragmentation send buffer
*
* @pre `snd_buf != NULL`
*
* @param[in,out] snd_buf A fragmentation send buffer entry. May not be NULL.
*/
void gnrc_ipv6_ext_frag_send(gnrc_ipv6_ext_frag_send_t *snd_buf);
/**
* @brief Reassemble fragmented IPv6 packet
*
* @param[in] pkt A fragment of the IPv6 packet to be reassembled containing
* the fragment header in the first snip.
*
* @return The reassembled packet when @p pkt completed the reassembly
* @return NULL, when there are still fragments missing or an error occurred
* during reassembly
*/
gnrc_pktsnip_t *gnrc_ipv6_ext_frag_reass(gnrc_pktsnip_t *pkt);
/**
* @name Reassembly buffer operations
* @{
*/
/**
* @brief Get a reassembly buffer by the identifying parameters
*
* @internal
* @see [RFC 8200, section 4.5](https://tools.ietf.org/html/rfc8200#section-4.5)
*
* @param[in] hdr IPv6 header to get source and destination address from.
* @param[in] id The identification from the fragment header.
*
* @return A reassembly buffer matching @p id ipv6_hdr_t::src and ipv6_hdr::dst
* of @p hdr or first free reassembly buffer. Will never be NULL, as
* in the case of the reassembly buffer being full, the entry with the
* lowest gnrc_ipv6_ext_frag_rbuf_t::arrival (serial-number-like) is
* removed.
*/
gnrc_ipv6_ext_frag_rbuf_t *gnrc_ipv6_ext_frag_rbuf_get(ipv6_hdr_t *ipv6,
uint32_t id);
/**
* @brief Frees a reassembly buffer entry (but does not release its
* gnrc_ipv6_ext_frag_rbuf_t::pkt)
*
* @param[in] rbuf A reassembly buffer entry.
*/
void gnrc_ipv6_ext_frag_rbuf_free(gnrc_ipv6_ext_frag_rbuf_t *rbuf);
/**
* @brief Delete a reassembly buffer entry (and release its
* gnrc_ipv6_ext_frag_rbuf_t::pkt)
*
* @note May be used by the IPv6 thread to remove a timed out reassembly
* buffer entry.
*
* @param[in] rbuf A reassembly buffer entry.
*/
static inline void gnrc_ipv6_ext_frag_rbuf_del(gnrc_ipv6_ext_frag_rbuf_t *rbuf)
{
gnrc_pktbuf_release(rbuf->pkt);
rbuf->pkt = NULL;
gnrc_ipv6_ext_frag_rbuf_free(rbuf);
}
/**
* @brief Garbage-collect reassembly buffer
*
* This calls @ref gnrc_ipv6_ext_frag_rbuf_del() for all reassembly buffer
* entries for which * gnrc_ipv6_ext_frag_rbuf_t::arrival is
* @ref GNRC_IPV6_EXT_FRAG_RBUF_TIMEOUT_US in the past.
*/
void gnrc_ipv6_ext_frag_rbuf_gc(void);
/** @} */
#ifdef __cplusplus
}
#endif
#endif /* NET_GNRC_IPV6_EXT_FRAG_H */
/** @} */