1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 18:52:45 +01:00
RIOT/sys/include/net/gnrc.h

311 lines
11 KiB
C
Raw Normal View History

/*
* Copyright (C) 2015 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.
*/
/**
2015-12-07 16:17:05 +01:00
* @defgroup net_gnrc Generic (GNRC) network stack
* @ingroup net
2015-08-10 02:41:08 +02:00
* @brief RIOT's modular default IP network stack.
* @see [Martine Lenders' master thesis](http://doc.riot-os.org/mlenders_msc.pdf)
* about GNRC's design and evaluation and the slide set of
* [its defense](http://doc.riot-os.org/mlenders_msc_def.pdf).
2015-12-04 19:19:17 +01:00
*
* About
* =====
*
* This module is currently the default network stack for RIOT and includes
* many components ranging from a @ref net_gnrc_netif through a fully-featured
2017-02-14 16:11:17 +01:00
* @ref net_gnrc_ipv6 implementation with @ref net_gnrc_sixlowpan "6LowPAN"
* extensions to an @ref net_gnrc_udp "UDP" implementation and
* @ref net_gnrc_rpl.
2015-12-04 19:19:17 +01:00
*
* A list of all features contained in the @ref net_gnrc is available in the
* `Modules` section above.
*
* Integration into RIOT
* =====================
*
* From the application layer the @ref net_gnrc can be accessed through the
* @ref net_sock, while the interface to the @ref drivers_netdev_api is
* defined by the @ref net_gnrc_netif.
2015-12-04 19:19:17 +01:00
*
* Architecture
* ============
*
* ![GNRC Network Stack](riot-gnrc.svg)
*
* Each layer of the network stack runs in its own thread and each lower layer
* thread has a higher priority than any upper layer thread. In this regard,
* the thread of the MAC layer implementation has the highest priority and
* threads on the application layer have the lowest priority. The communication
* between threads is handled by the kernel's @ref core_msg functionality and
* by the @ref net_gnrc_netapi. Most of the times IPC will take place between
* threads of neighboring layers for packets that traverse the network stack up
* or down.
*
* Due to the design of @ref net_gnrc "GNRC" and the nature of inter-process
* communication, it is crucial for a new module that introduces a new thread
* to follow a certain programming construct if it desires to interact with
* other threads without blocking the system: Utilizing an `event loop`.
*
* Hence, a thread for @ref net_gnrc "GNRC" will usually consist of four basic
* steps.
*
* 1. Initialize a message queue (note that its size **must** be a power of two,
* see @ref msg_init_queue())
2015-12-04 19:19:17 +01:00
* 2. register for a @ref net_gnrc_nettype
* 3. wait for a message
* 4. react appropriately to a message and return to 3.
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
* void *_event_loop(void *arg)
* {
* static msg_t _msg_q[Q_SZ];
* (void)arg;
* msg_init_queue(_msg_q, Q_SZ);
* gnrc_netreg_entry me_reg = GNRC_NETREG_ENTRY_INIT_PID(
* GNRC_NETREG_DEMUX_CTX_ALL,
* sched_active_pid);
2015-12-04 19:19:17 +01:00
* gnrc_netreg_register(GNRC_NETTYPE_IPV6, &me_reg);
* while (1) {
* msg_receive(&msg);
* switch (msg.type) {
* case TYPE1:
* callback1();
* break;
* ...
* }
* }
*
* return NULL;
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* @note As an example have a look at the event loops of @ref net_gnrc_ipv6 and
* @ref net_gnrc_rpl
*
* Receiving / Transmitting Packets
* ================================
*
* Packets can be received or transmitted by interacting with the @ref
* net_gnrc_netapi.
*
* Receiving Packets
* -----------------
*
* The reception of a @ref net_gnrc_pkt from another thread is handled by the
* @ref net_gnrc_netapi module. In order to receive a @ref net_gnrc_pkt of a
* specific type, it is necessary to register for the appropriate @ref
* net_gnrc_nettype first. Your thread will then be able to receive certain
* commands defined in the @ref net_gnrc_netapi module (e.g. @ref
* net_gnrc_netapi::GNRC_NETAPI_MSG_TYPE_RCV) for all @ref net_gnrc_pkt
* "Packets" that your thread registered for.
*
* The following example will sketch how to receive incoming and outgoing UDP
* traffic on port 80.
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
* void *_event_loop(void *arg)
* {
* static msg_t _msg_q[Q_SZ];
* msg_t msg, reply;
* reply.type = GNRC_NETAPI_MSG_TYPE_ACK;
* reply.content.value = -ENOTSUP;
* msg_init_queue(_msg_q, Q_SZ);
* gnrc_pktsnip_t *pkt = NULL;
* gnrc_netreg_entry me_reg = { .demux_ctx = 80, .pid = thread_getpid() };
* gnrc_netreg_register(GNRC_NETTYPE_UDP, &me_reg);
*
* while (1) {
* msg_receive(&msg);
* switch (msg.type) {
* case GNRC_NETAPI_MSG_TYPE_RCV:
* pkt = msg.content.ptr;
2015-12-04 19:19:17 +01:00
* _handle_incoming_pkt(pkt);
* break;
* case GNRC_NETAPI_MSG_TYPE_SND:
* pkt = msg.content.ptr;
2015-12-04 19:19:17 +01:00
* _handle_outgoing_pkt(pkt);
* break;
* case GNRC_NETAPI_MSG_TYPE_SET:
* case GNRC_NETAPI_MSG_TYPE_GET:
* msg_reply(&msg, &reply);
* break;
* default:
* break;
* }
* }
*
* return NULL;
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
2016-10-09 09:25:47 +02:00
* @note When receiving a message of type @ref GNRC_NETAPI_MSG_TYPE_SET or
* @ref GNRC_NETAPI_MSG_TYPE_GET, it is necessary to acknowledge it by
* calling @ref msg_reply() with a message of type
* @ref GNRC_NETAPI_MSG_TYPE_ACK which contains the actual size of the
* GET message's content on success or an error code otherwise.
2015-12-04 19:19:17 +01:00
*
* @note Do not forget to unregister with @ref gnrc_netreg_unregister() if you
* leave the function
* context
*
* Transmitting Packets
* --------------------
*
2016-10-09 09:25:47 +02:00
* A packet is transmitted by relaying it to threads interested in handling (and
* dispatching) packets of its type. To do this, the @ref net_gnrc_netapi
* offers dispatch helper functions called @ref gnrc_netapi_dispatch_send()
* and gnrc_netapi_dispatch_receive().
*
* The following example sketches the usage and assumes a valid @ref
* net_gnrc_pkt named `pkt`.
2015-12-04 19:19:17 +01:00
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
2016-10-09 09:25:47 +02:00
* gnrc_pktsnip_t *pkt;
*
* pkt = gnrc_pktbuf_add(NULL, data, size, GNRC_NETTYPE_UNDEF);
* if (pkt == NULL) {
* puts("Error: unable to copy data to packet buffer\n");
* return;
* }
2015-12-04 19:19:17 +01:00
* if (!gnrc_netapi_dispatch_send(GNRC_NETTYPE_UDP, 80, pkt)) {
* puts("Error: no thread is interested");
* gnrc_pktbuf_release(pkt);
* return;
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
2016-10-09 09:25:47 +02:00
* First, the data to be sent is added to the @ref net_gnrc_pktbuf "packet buffer".
* This ensures its intactness during the sending process. After the data to be
* sent has been added to the packet buffer, its parent data structure can safely
* be freed or re-used.
*
* Then, the @ref net_gnrc_pkt "pkt" will be sent to all threads that registered
* for @ref GNRC_NETTYPE_UDP and the demux context `80`. Every registered thread
2015-12-04 19:19:17 +01:00
* will receive a @ref GNRC_NETAPI_MSG_TYPE_SND command and can access the @ref
2016-10-09 09:25:47 +02:00
* net_gnrc_pkt. Note that at this point, the threads receiving pkt act as its
* owners, so please don't modify pkt after calling any dispatch function.
*
* If @ref gnrc_netapi_dispatch_send() is replaced by @ref
2015-12-04 19:19:17 +01:00
* gnrc_netapi_dispatch_receive() then threads will receive the @ref
* GNRC_NETAPI_MSG_TYPE_RCV command instead, again with access to the @ref
* net_gnrc_pkt.
*
2016-10-09 09:25:47 +02:00
* @note If the data to be sent requires extra headers to be added for successful
* transmission (in the example, this would be IP and UDP headers), these
* have to be built manually before calling @ref gnrc_netapi_dispatch_send().
* In the interest of conciseness, this is omitted in this tutorial;
* please refer to @ref gnrc_udp_hdr_build(), @ref gnrc_ipv6_hdr_build()
* etc. for more information.
*
* @note GNRC is implemented according to the respective standards. So please
* note, that sending to a IPv6 link-local address always requires you
* by definition to also provide the interface you want to send to,
* otherwise address resolution might fail.
*
2015-12-04 19:19:17 +01:00
* How To Use
* ==========
* @ref net_gnrc is highly modular and can be adjusted to include only the
* desired features. In the following several of the available modules will be
* stated that you can include in your application's Makefile.
*
* - To include the default network device(s) on your board:
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_netdev_default
2015-12-04 19:19:17 +01:00
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* - To auto-initialize these network devices as GNRC network interfaces
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += auto_init_gnrc_netif
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* - You may choose to build either as an IPv6 Node
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_ipv6_default
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* or as an IPv6 Router
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_ipv6_router_default
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* An IPv6 Router can forward packets, while an IPv6 Node will simply drop
* packets not targeted to it. If an IEEE 802.15.4 network device is present
* @ref net_gnrc_sixlowpan (with @ref net_gnrc_sixlowpan_frag and @ref
* net_gnrc_sixlowpan_iphc) will be included automatically.
* - For basic IPv6 (and 6LoWPAN) functionalities choose instead
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_ipv6
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* or
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_ipv6_router
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* respectively. Those modules provide the bare minimum of IPv6
* functionalities (no @ref net_gnrc_icmpv6). Because of that, the
* @ref net_gnrc_ipv6_nib needs to be configured manually. If an IEEE 802.15.4
* device is present @ref net_gnrc_sixlowpan will be included automatically,
* but no fragmentation or header compression support will be provided.
2015-12-04 19:19:17 +01:00
*
* - For @ref net_gnrc_icmpv6_echo "ICMPv6 echo request/reply (ping)"
* functionality:
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_icmpv6_echo
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* - For @ref net_gnrc_udp support include
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_udp
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* - To use @ref net_sock_udp with GNRC include
2015-12-04 19:19:17 +01:00
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_sock_udp
2015-12-04 19:19:17 +01:00
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* - To include the @ref net_gnrc_rpl module
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.mk}
* USEMODULE += gnrc_rpl
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
* This will include the @ref net_gnrc_rpl module. To provide forwarding
2015-12-09 13:18:23 +01:00
* capabilities it is necessary to build the application with
2015-12-04 19:19:17 +01:00
* `gnrc_ipv6_router_default` (or `gnrc_ipv6_router`), not
* `gnrc_ipv6_default` (or `gnrc_ipv6`).
*
* @{
*
* @file
* @brief Includes all essential GNRC network stack base modules
*
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*/
#ifndef NET_GNRC_H
#define NET_GNRC_H
2015-08-06 15:37:11 +02:00
#include "net/netopt.h"
#include "net/gnrc/netapi.h"
#include "net/gnrc/netreg.h"
#include "net/gnrc/nettype.h"
2017-11-16 18:06:46 +01:00
#include "net/gnrc/netif.h"
#include "net/gnrc/netif/hdr.h"
#include "net/gnrc/pktbuf.h"
#include "net/gnrc/pkt.h"
#ifdef __cplusplus
extern "C" {
#endif
/* this file does not provide anything on it's own */
#ifdef __cplusplus
}
#endif
#endif /* NET_GNRC_H */
/** @} */