Doc: Migrate IPv6

Migrate https://cwiki.apache.org/confluence/display/NUTTX/IPv6 to
official wiki

Signed-off-by: Ludovic Vanasse <ludovicvanasse@gmail.com>
This commit is contained in:
Ludovic Vanasse 2024-10-20 12:39:28 -04:00 committed by Alan C. Assis
parent e9c107e0ba
commit 035faaccc7
2 changed files with 350 additions and 1 deletions

View file

@ -44,4 +44,5 @@ Guides
specialstuff_in_nuttxheaderfiles.rst
kernel_threads_with_custom_stacks.rst
versioning_and_task_names.rst
logging_rambuffer.rst
logging_rambuffer.rst
ipv6.rst

View file

@ -0,0 +1,348 @@
====
IPv6
====
.. warning::
Migrated from: https://cwiki.apache.org/confluence/display/NUTTX/IPv6
NuttX has supported the Internet Protocol Version 4 (IPv4) for many years.
There have been fragments of IPv6 in the code base for many years as well,
but these fragments were not more than place markers and not functional.
But recently, post NuttX-7.6, I have focused some effort into completing the
IPv6 implementation. This Wiki page contains notes from that integration
effort and will, hopefully, evolve to provide full documentation for NuttX
IPv6 support.
Current status: Basic functionality is complete and verified. That includes
ICMPPv6 Neighbor Discover Protocol, IPCMPv6 Echo Request/Response
(for ``ping6``), TCP/IPv6, and UDP/IPv6. It has also been proven that you
can support a platform with `both` IPv4 and IPv6 enabled.
Ethernet Driver Requirements
============================
Basic Driver Requirements
-------------------------
In order to support IPv6, Ethernet drivers must do the following:
* They must recognize IPv6 packets and call ``ipv6_input`` in order to pass the
packets into the network stack. This is equivalent to calling ``ipv4_input``
when an IPv4 pack is received.
* When sending an IPv6, the drivers must call ``neighbor_out()`` in order to add
the MAC address of the destination into the link layer header. IPv6's
`ICMPv6 Neighbor Discovery Protocol` is the moral equivalent of the
`Address Resolution Protocol` (ARP) used with IPv6. And the IPv6
``neighbor_out()`` performs a similar function to the IPv4 ``arp_out()``
function.
* Ethernet drivers must also support some additional address filtering.
For IPv4 support, most Ethernet drivers are configured to accept only
Ethernet packets with matching MAC addresses and broadcast packets (or
selected multicast packets if IGMP support is enabled). Additional
filtering support is needed to support IPv6.
All existing NuttX Ethernet drivers have already been modified to support
the requirements of the first two bullets. However, additional logic must
be added to most of the existing Ethernet drivers to support the final
requirement.
Multicast Address Filtering
---------------------------
Each Ethernet device connects to the Ethernet wire via a PHY and so
potentially has access to every packet that passes on the wire. In
`promiscuous` mode, that is the behavior that is desired but normally
it is not: The amount of traffic that appears on the wire would swamp
most modest MCUs in promiscuous mode. So instead, the Ethernet MAC
hardware will support address filtering. That is, the hardware will
look at the Ethernet header at the beginning of each packet and will
ignore packets that do not have the desired information in the Ethernet
header. The software will see only those filtered packets that are desired.
Typically, the Ethernet MAC is set-up for `unicast` address filtering: The
hardware is programmed so that that only packets whose destination Ethernet
MAC address matches the MAC address programmed into the hardware are accepted.
In addition, special `broadcast` Ethernet addresses will also be accepted.
In this way, the volume of Ethernet data received by the MCU is greatly
reduced.
`Multicast` addresses are a little different. Unlike broadcast addresses,
there are many possible multicast addresses and so the Ethernet MAC hardware
must support some special capability to match the destination Ethernet
address in an incoming packet with a variety of multicast addresses.
Usually this involves `hashing` the Ethernet address and performing a `hash
table lookup` to check for an address match.
Each Ethernet driver uses a common interface that is defined in
``nuttx/include/nuttx/net/netdev.h``. That interface defines, among other
things, a set of calls into the Ethernet driver to perform a variety of
functions. One of those functions is multicast address filtering:
.. code-block:: c
#ifdef CONFIG_NET_IGMP
int (*d_addmac)(FAR struct net_driver_s *dev, FAR const uint8_t *mac);
int (*d_rmmac)(FAR struct net_driver_s *dev, FAR const uint8_t *mac);
#endif
The ``d_addmac()`` interface adds a multicast address to the hash
table; ``d_rmmac()`` removes a multicast address from the hash table.
These interface is only required if IGMP is supported, but the underlying
ability to program multicast address filtered is required for full IPv6
support. This interface exists in all Ethernet drivers but most are
currently place holders and are `to-be-provided`. At present, only the
STMicro STM32, the TI Tiva TM4C, and the Atmel SAM3/4 and SAMA5D3/4
Ethernet drivers support multicast hash tables. This capability will
have to be added to any additional Ethernet drivers that are modified
to support IPv6.
ICMPv6 Neighbor Discovery Protocol
----------------------------------
The ICMPv6 Neighbor Discover protocol is the reason for this additional
address filtering. The ICMPv6 Neighbor Discovery Protocol is the
replacement for IPv4's ARP. It different from ARP in the it is
implemented not at the Ethernet link layer, but within the IPv6 layer.
In order to receive broadcast packets to ICMPv6, the IPv6 Multicast
address of 33.33.ff.xx.xx.xx is used, where the xx.xx.xx part derives
from the IPv6 address. The Ethernet driver filtering logic must be modified
so that it accepts packets directed to the that MAC address.
At present, this additional support is only implemented for the TI Tiva
TM4C129X Ethernet driver. Below is a snippet of code from that drier
showing how this is implemented:
.. code-block:: c
/* Set the MAC address */
tiva_macaddress(priv);
#ifdef CONFIG_NET_ICMPv6
/* Set up the IPv6 multicast address */
tiva_ipv6multicast(priv);
#endif
Where `tiva_macaddress()` sets up the normal MAC address filtering and
`tiva_ipv6multicast()` sets up the special filtering needed by IPv6:
.. code-block:: c
/****************************************************************************
* Function: tiva_ipv6multicast
*
* Description:
* Configure the IPv6 multicast MAC address.
*
* Parameters:
* priv - A reference to the private driver state structure
*
* Returned Value:
* OK on success; Negated errno on failure.
*
* Assumptions:
***************************************************************************/
#ifdef CONFIG_NET_ICMPv6
static void tiva_ipv6multicast(FAR struct tiva_ethmac_s *priv)
{
struct net_driver_s *dev;
uint16_t tmp16;
uint8_t mac[6];
/* For ICMPv6, we need to add the IPv6 multicast address
* For IPv6 multicast addresses, the Ethernet MAC is derived by
* the four low-order octets OR'ed with the MAC 33:33:00:00:00:00,
* so for example the IPv6 address FF02:DEAD:BEEF::1:3 would map
* to the Ethernet MAC address 33:33:00:01:00:03.
* NOTES: This appears correct for the ICMPv6 Router Solicitation
* Message, but the ICMPv6 Neighbor Solicitation message seems to
* use 33:33:ff:01:00:03.
*/
mac[0] = 0x33;
mac[1] = 0x33;
dev = &priv->dev;
tmp16 = dev->d_ipv6addr[6];
mac[2] = 0xff;
mac[3] = tmp16 >> 8;
tmp16 = dev->d_ipv6addr[7];
mac[4] = tmp16 & 0xff;
mac[5] = tmp16 >> 8;
nvdbg("IPv6 Multicast: %02x:%02x:%02x:%02x:%02x:%02x\n",
mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
(void)tiva_addmac(dev, mac);
#ifdef CONFIG_NET_ICMPv6_AUTOCONF
/* Add the IPv6 all link-local nodes Ethernet address. This is the
* address that we expect to receive ICMPv6 Router Advertisement
* packets.
*/
(void)tiva_addmac(dev, g_ipv6_ethallnodes.ether_addr_octet);
#endif /* CONFIG_NET_ICMPv6_AUTOCONF */
#ifdef CONFIG_NET_ICMPv6_ROUTER
/* Add the IPv6 all link-local routers Ethernet address. This is the
* address that we expect to receive ICMPv6 Router Solicitation
* packets.
*/
(void)tiva_addmac(dev, g_ipv6_ethallrouters.ether_addr_octet);
#endif /* CONFIG_NET_ICMPv6_ROUTER */
}
#endif /* CONFIG_NET_ICMPv6 */
The following Ethernet drivers are complete and IPv6 ready. All others
Ethernet drivers have all required IPv6 support `except` that they are
missing (1) the required ICMPv6 addressing filtering described above
and/or (2) support for multi-cast address filtering.
* STMicro STM32
* TI Tiva TM4C
* Atmel SAMA5D4
* NXP LPC17xx
Board Configurations
====================
At present, there are three board configuration that are pre-configured to
use IPv6: ``nuttx/boards/arm/tiva/dk-tm4c129x/configs/ipv6``,
``nuttx/boards/arm/stm32/stm32f4discovery/ipv6``, and
``nuttx/boards/arm/tiva/tm4c1294-launchpad/configs/ipv6``. These default
configurations have only IPv6 enabled. But the `README` files at in those
board directories describes how to enable `both` IPv4 and IPv6 simultaneously.
Ping
====
Ping from Host PC
-----------------
Ping from Windows cmd Terminal
``````````````````````````````
.. code-block:: bash
ping -6 fc00::2
Ping From Linux shell
`````````````````````
.. code-block:: bash
ping6 fc00::2
Ping from the NuttShell (NSH)
-----------------------------
.. code-block:: bash
nsh> ping6 fc00::2
NSH ifconfig
============
IPv4 Only
---------
``CONFIG_NET_IPv4=y`` and ``CONFIG_NET_IPv6=n``
.. code-block:: bash
nsh> ifconfig
eth0 Link encap: Ethernet HWaddr 00:1a:b6:02:81:14 at UP
inet addr:10.0.0.2 DRaddr:10.0.0.1 Mask:255.255.255.0
IPv4 TCP UDP ICMP
Received 003b 001c 0000 0004
Dropped 001b 0000 0000 0000
IPv4 VHL: 0000 Frg: 0000
Checksum 0000 0000 0000 ----
TCP ACK: 0000 SYN: 0000
RST: 0000 0000
Type 0000 ---- ---- 0000
Sent 0031 002d 0000 0004
Rexmit ---- ---- 0000 ----
NOTE: The detailed packet statistics only appear if
``CONFIG_NET_STATISTICS`` is enabled.
IPv6 Only
---------
``CONFIG_NET_IPv4=n`` and ``CONFIG_NET_IPv6=y``
.. code-block:: bash
nsh> ifconfig
eth0 Link encap: Ethernet HWaddr 00:1a:b6:02:81:14 at UP
inet6 addr:fc00::2
inet6 DRaddr:fc00::1
inet6 Mask:ffff:ffff:ffff::ffff:ffff:ffff:ff80
IPv6 TCP UDP ICMPv6
Received 0007 0000 0000 0007
Dropped 0000 0000 0000 0000
IPv6 VHL: 0000
Checksum ---- 0000 0000 ----
TCP ACK: 0000 SYN: 0000
RST: 0000 0000
Type 0000 ---- ---- 0000
Sent 0011 0000 0000 0011
Rexmit ---- ---- 0000 ----
Both IPv4 and IPv6
------------------
``CONFIG_NET_IPv4=y`` and ``CONFIG_NET_IPv6=y``
.. code-block:: bash
nsh> ifconfig
eth0 Link encap: Ethernet HWaddr 00:1a:b6:02:81:14 at UP
inet addr:10.0.0.2 DRaddr:10.0.0.1 Mask:255.255.255.0
inet6 addr:fc00::2
inet6 DRaddr:fc00::1
inet6 Mask:ffff:ffff:ffff::ffff:ffff:ffff:ff80
IPv4 IPv6 TCP UDP ICMP ICMPv6
Received 0047 000a 001c 0000 0004 000a
Dropped 0027 0000 0000 0000 0000 0000
IPv4 VHL: 0000 Frg: 0000
IPv6 VHL: 0000
Checksum 0000 ---- 0000 0000 ---- ----
TCP ACK: 0000 SYN: 0000
RST: 0000 0000
Type 0000 0000 ---- ---- 0000 0000
Sent 0033 000a 002f 0000 0004 000a
Rexmit ---- ---- ---- 0000 ---- ----
Tests, Applications, and Network Utilities
==========================================
In addition to the core RTOS support IPv6, changes are also required to
networking tests, to networking aware applications, and, of course, to all of
the network utils (``netutils``).
* NuttShell (NSH): IPv6 support is partially available. NSH is capable of
initializing the IPv6 domain and some of the NSH commands have been adapted
to support IPv6. A ping6 command has been added. But there are many commands
that still require updating.
* Tests: There are several networking tests in ``apps/examples``. The
``nettest`` test and the ``udp`` test have been adapted to work in the IPv6
domain, but none of the others have yet been adapted.
* Netutils: The network utilities in ``apps/netutils`` have been adapted to
work with IPv6: DHCP, FTP, TFTP, Telnet, etc. Support for managing IPv6
address have been included in the ``netlib``, but nothing else has yet been
updated.