net/devif, net/tcp, and net/udp: Extend the logic of 6c0ab0e077 so that all support Transport protocols support by IPv6 can handle the presence of IPv6 header extension options.

This commit is contained in:
Gregory Nutt 2018-11-02 17:50:01 -06:00
parent 6c0ab0e077
commit 42a018747e
7 changed files with 88 additions and 80 deletions

View file

@ -275,10 +275,10 @@ int ipv6_input(FAR struct net_driver_s *dev)
{
FAR struct ipv6_hdr_s *ipv6 = IPv6BUF;
FAR uint8_t *payload;
uint16_t hdrlen;
uint16_t llhdrlen;
uint16_t iphdrlen;
uint16_t pktlen;
uint8_t nxthdr;
bool exthdrs;
#ifdef CONFIG_NET_IPFORWARD
int ret;
#endif
@ -296,25 +296,62 @@ int ipv6_input(FAR struct net_driver_s *dev)
{
/* IP version and header length. */
nwarn("WARNING: Invalid IPv6 version: %d\n", ipv6->vtc >> 4);
#ifdef CONFIG_NET_STATISTICS
g_netstats.ipv6.drop++;
g_netstats.ipv6.vhlerr++;
#endif
nwarn("WARNING: Invalid IPv6 version: %d\n", ipv6->vtc >> 4);
goto drop;
}
/* Get the size of the packet minus the size of link layer header */
hdrlen = NET_LL_HDRLEN(dev);
if ((hdrlen + IPv6_HDRLEN) > dev->d_len)
llhdrlen = NET_LL_HDRLEN(dev);
if ((llhdrlen + IPv6_HDRLEN) > dev->d_len)
{
nwarn("WARNING: Packet shorter than IPv6 header\n");
goto drop;
}
dev->d_len -= hdrlen;
dev->d_len -= llhdrlen;
/* Make sure that all packet processing logic knows that there is an IPv6
* packet in the device buffer.
*/
IFF_SET_IPv6(dev->d_flags);
/* Parse IPv6 extension headers (parsed but ignored) */
payload = PAYLOAD; /* Assume payload starts right after IPv6 header */
iphdrlen = IPv6_HDRLEN; /* Total length of the IPv6 header */
nxthdr = ipv6->proto; /* Next header determined by IPv6 header prototype */
while (ipv6_exthdr(nxthdr))
{
FAR struct ipv6_extension_s *exthdr;
uint16_t extlen;
/* Just skip over the extension header */
exthdr = (FAR struct ipv6_extension_s *)payload;
extlen = EXTHDR_LEN((unsigned int)exthdr->len);
payload += extlen;
iphdrlen += extlen;
/* Check for a short packet */
if (iphdrlen > dev->d_len)
{
nwarn("WARNING: Packet shorter than IPv6 header\n");
goto drop;
}
/* Set up for the next time through the loop */
exthdr = (FAR struct ipv6_extension_s *)payload;
nxthdr = exthdr->nxthdr;
}
/* Check the size of the packet. If the size reported to us in d_len is
* smaller the size reported in the IP header, we assume that the packet
@ -326,10 +363,12 @@ int ipv6_input(FAR struct net_driver_s *dev)
* that follows the header. The device interface uses the d_len variable
* for holding the size of the entire packet, including the IP header but
* without the link layer header.
*
* REVISIT: Length will be set to zero if the extension header carries
* a Jumbo payload option.
*/
pktlen = ((uint16_t)ipv6->len[0] << 8) + (uint16_t)ipv6->len[1] +
IPv6_HDRLEN;
pktlen = ((uint16_t)ipv6->len[0] << 8) + (uint16_t)ipv6->len[1] + iphdrlen;
if (pktlen <= dev->d_len)
{
@ -367,7 +406,7 @@ int ipv6_input(FAR struct net_driver_s *dev)
ipv6_forward_broadcast(dev, ipv6);
}
#endif
return udp_ipv6_input(dev);
return udp_ipv6_input(dev, iphdrlen);
}
/* In other cases, the device must be assigned a non-zero IP address
@ -405,7 +444,7 @@ int ipv6_input(FAR struct net_driver_s *dev)
{
/* The packet was forwarded. Return success; d_len will
* be set appropriately by the forwarding logic: Cleared
* if the packet is forward via anoother device or non-
* if the packet is forward via another device or non-
* zero if it will be forwarded by the same device that
* it was received on.
*/
@ -418,56 +457,11 @@ int ipv6_input(FAR struct net_driver_s *dev)
/* Not destined for us and not forwardable... drop the packet. */
nwarn("WARNING: Not destined for us; not forwardable... Dropping!\n");
#ifdef CONFIG_NET_STATISTICS
g_netstats.ipv6.drop++;
#endif
goto drop;
}
}
}
/* Parse IPv6 extension headers (parsed but ignored) */
payload = PAYLOAD; /* Assume payload starts right after IPv6 header */
nxthdr = ipv6->proto; /* Next header determined by IPv6 header prototype */
exthdrs = false; /* No IPv6 extension headers found */
while (ipv6_exthdr(nxthdr))
{
FAR struct ipv6_extension_s *exthdr;
/* Just skip over the extension header */
exthdr = (FAR struct ipv6_extension_s *)payload;
payload += EXTHDR_LEN((unsigned int)exthdr->len);
exthdr = (FAR struct ipv6_extension_s *)payload;
nxthdr = exthdr->nxthdr;
exthdrs = true;
}
#ifdef CONFIG_NET_ICMPv6
/* Currently only icmpv6_input() can handle the presence of IPv6 extension
* headers. The issue is that with extension headers present, the
* transport layer header lies at a variable offset from the end of the
* IPv6 header and currently only ICMPv6 can deal with that offset.
*/
if (exthdrs && nxthdr != IP_PROTO_ICMP6)
#else
if (exthdrs)
#endif
{
nwarn("WARNING: Extension headers with proto=%u\n", nxthdr);
goto drop;
}
/* Make sure that all packet processing logic knows that there is an IPv6
* packet in the device buffer.
*/
IFF_SET_IPv6(dev->d_flags);
/* Now process the incoming packet according to the protocol specified in
* the next header IPv6 field.
*/
@ -478,16 +472,16 @@ int ipv6_input(FAR struct net_driver_s *dev)
case IP_PROTO_TCP: /* TCP input */
/* Forward the IPv6 TCP packet */
tcp_ipv6_input(dev);
tcp_ipv6_input(dev, iphdrlen);
#ifdef CONFIG_NET_6LOWPAN
/* TCP output comes through three different mechansims. Either from:
/* TCP output comes through three different mechanisms. Either from:
*
* 1. TCP socket output. For the case of TCP output to an
* IEEE802.15.4, the TCP output is caught in the socket
* send()/sendto() logic and and redirected to 6LoWPAN logic.
* 2. TCP output from the TCP state machine. That will occur
* during TCP packet processing by the TCP state meachine.
* during TCP packet processing by the TCP state machine.
* 3. TCP output resulting from TX or timer polling
*
* Case 3 is handled here. Logic here detects if (1) an attempt
@ -514,7 +508,7 @@ int ipv6_input(FAR struct net_driver_s *dev)
case IP_PROTO_UDP: /* UDP input */
/* Forward the IPv6 UDP packet */
udp_ipv6_input(dev);
udp_ipv6_input(dev, iphdrlen);
break;
#endif
@ -555,12 +549,11 @@ int ipv6_input(FAR struct net_driver_s *dev)
#endif /* CONFIG_NET_ICMPv6 */
default: /* Unrecognized/unsupported protocol */
nwarn("WARNING: Unrecognized IP protocol: %04x\n", ipv6->proto);
#ifdef CONFIG_NET_STATISTICS
g_netstats.ipv6.drop++;
g_netstats.ipv6.protoerr++;
#endif
nwarn("WARNING: Unrecognized IP protocol: %04x\n", ipv6->proto);
goto drop;
}
@ -573,6 +566,9 @@ int ipv6_input(FAR struct net_driver_s *dev)
*/
drop:
#ifdef CONFIG_NET_STATISTICS
g_netstats.ipv6.drop++;
#endif
dev->d_len = 0;
return OK;
}

View file

@ -157,8 +157,8 @@ struct pollfd; /* Forward reference */
* Input Parameters:
* dev - The device driver structure containing the received ICMPv6
* packet
* icmpv6 - Start of the ICMPv6 packet which may lie at an offset from
* the IPv6 if IPv6 extension headers are present.
* icmpv6 - Start of the ICMPv6 payload which may lie at an offset from
* the IPv6 header if IPv6 extension headers are present.
*
* Returned Value:
* None

View file

@ -216,8 +216,8 @@ drop:
* Input Parameters:
* dev - The device driver structure containing the received ICMPv6
* packet
* icmpv6 - Start of the ICMPv6 packet which may lie at an offset from
* the IPv6 if IPv6 extension headers are present.
* icmpv6 - Start of the ICMPv6 payload which may lie at an offset from
* the IPv6 header if IPv6 extension headers are present.
*
* Returned Value:
* None

View file

@ -1030,7 +1030,10 @@ void tcp_ipv4_input(FAR struct net_driver_s *dev);
* Handle incoming TCP input with IPv4 header
*
* Input Parameters:
* dev - The device driver structure containing the received TCP packet.
* dev - The device driver structure containing the received TCP packet.
* iplen - The size of the IPv6 header. This may be larger than
* IPv6_HDRLEN the IPv6 header if IPv6 extension headers are
* present.
*
* Returned Value:
* None
@ -1041,7 +1044,7 @@ void tcp_ipv4_input(FAR struct net_driver_s *dev);
****************************************************************************/
#ifdef CONFIG_NET_IPv6
void tcp_ipv6_input(FAR struct net_driver_s *dev);
void tcp_ipv6_input(FAR struct net_driver_s *dev, unsigned int iplen);
#endif
/****************************************************************************

View file

@ -73,9 +73,9 @@
* Handle incoming TCP input
*
* Input Parameters:
* dev - The device driver structure containing the received TCP packet.
* dev - The device driver structure containing the received TCP packet.
* domain - IP domain (PF_INET or PF_INET6)
* iplen - Lngth of the IP header (IPv4_HDRLEN or IPv6_HDRLEN).
* iplen - Length of the IP header (IPv4_HDRLEN or IPv6_HDRLEN).
*
* Returned Value:
* None
@ -1048,7 +1048,10 @@ void tcp_ipv4_input(FAR struct net_driver_s *dev)
* Handle incoming TCP input with IPv4 header
*
* Input Parameters:
* dev - The device driver structure containing the received TCP packet.
* dev - The device driver structure containing the received TCP packet.
* iplen - The size of the IPv6 header. This may be larger than
* IPv6_HDRLEN the IPv6 header if IPv6 extension headers are
* present.
*
* Returned Value:
* None
@ -1059,7 +1062,7 @@ void tcp_ipv4_input(FAR struct net_driver_s *dev)
****************************************************************************/
#ifdef CONFIG_NET_IPv6
void tcp_ipv6_input(FAR struct net_driver_s *dev)
void tcp_ipv6_input(FAR struct net_driver_s *dev, unsigned int iplen)
{
/* Configure to receive an TCP IPv6 packet */
@ -1067,7 +1070,7 @@ void tcp_ipv6_input(FAR struct net_driver_s *dev)
/* Then process in the TCP IPv6 input */
tcp_input(dev, PF_INET6, IPv6_HDRLEN);
tcp_input(dev, PF_INET6, iplen);
}
#endif

View file

@ -520,7 +520,10 @@ int udp_ipv4_input(FAR struct net_driver_s *dev);
* Handle incoming UDP input in an IPv6 packet
*
* Input Parameters:
* dev - The device driver structure containing the received UDP packet
* dev - The device driver structure containing the received UDP packet
* iplen - The size of the IPv6 header. This may be larger than
* IPv6_HDRLEN the IPv6 header if IPv6 extension headers are
* present.
*
* Returned Value:
* OK The packet has been processed and can be deleted
@ -533,7 +536,7 @@ int udp_ipv4_input(FAR struct net_driver_s *dev);
****************************************************************************/
#ifdef CONFIG_NET_IPv6
int udp_ipv6_input(FAR struct net_driver_s *dev);
int udp_ipv6_input(FAR struct net_driver_s *dev, unsigned int iplen);
#endif
/****************************************************************************

View file

@ -271,7 +271,10 @@ int udp_ipv4_input(FAR struct net_driver_s *dev)
* Handle incoming UDP input in an IPv6 packet
*
* Input Parameters:
* dev - The device driver structure containing the received UDP packet
* dev - The device driver structure containing the received UDP packet
* iplen - The size of the IPv6 header. This may be larger than
* IPv6_HDRLEN the IPv6 header if IPv6 extension headers are
* present.
*
* Returned Value:
* OK The packet has been processed and can be deleted
@ -284,7 +287,7 @@ int udp_ipv4_input(FAR struct net_driver_s *dev)
****************************************************************************/
#ifdef CONFIG_NET_IPv6
int udp_ipv6_input(FAR struct net_driver_s *dev)
int udp_ipv6_input(FAR struct net_driver_s *dev, unsigned int iplen)
{
/* Configure to receive an UDP IPv6 packet */
@ -292,7 +295,7 @@ int udp_ipv6_input(FAR struct net_driver_s *dev)
/* Then process in the UDP IPv6 input */
return udp_input(dev, IPv6_HDRLEN);
return udp_input(dev, iplen);
}
#endif