2007-11-07 00:38:14 +01:00
|
|
|
/****************************************************************************
|
2014-06-18 18:50:09 +02:00
|
|
|
* net/udp/udp_send.c
|
2007-11-07 00:38:14 +01:00
|
|
|
*
|
2011-01-23 18:40:10 +01:00
|
|
|
* Copyright (C) 2007-2009, 2011 Gregory Nutt. All rights reserved.
|
2012-09-13 20:32:24 +02:00
|
|
|
* Author: Gregory Nutt <gnutt@nuttx.org>
|
2007-11-07 00:38:14 +01:00
|
|
|
*
|
|
|
|
* Adapted for NuttX from logic in uIP which also has a BSD-like license:
|
|
|
|
*
|
|
|
|
* Original author Adam Dunkels <adam@dunkels.com>
|
|
|
|
* Copyright () 2001-2003, Adam Dunkels.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote
|
|
|
|
* products derived from this software without specific prior
|
|
|
|
* written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS
|
|
|
|
* OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
|
|
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
|
|
|
|
* GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#if defined(CONFIG_NET) && defined(CONFIG_NET_UDP)
|
|
|
|
|
|
|
|
#include <debug.h>
|
|
|
|
|
2014-07-05 03:13:08 +02:00
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
2014-06-24 16:53:28 +02:00
|
|
|
#include <nuttx/net/netconfig.h>
|
2014-06-24 17:28:44 +02:00
|
|
|
#include <nuttx/net/netdev.h>
|
2014-06-26 17:32:39 +02:00
|
|
|
#include <nuttx/net/netstats.h>
|
2014-07-05 03:13:08 +02:00
|
|
|
#include <nuttx/net/ip.h>
|
|
|
|
#include <nuttx/net/udp.h>
|
2007-11-07 00:38:14 +01:00
|
|
|
|
2014-06-29 02:07:02 +02:00
|
|
|
#include "devif/devif.h"
|
2014-06-28 01:51:32 +02:00
|
|
|
#include "utils/utils.h"
|
2014-06-25 02:55:01 +02:00
|
|
|
#include "udp/udp.h"
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
2009-12-15 15:53:45 +01:00
|
|
|
* Pre-processor Definitions
|
2007-11-07 00:38:14 +01:00
|
|
|
****************************************************************************/
|
|
|
|
|
2014-11-15 20:13:23 +01:00
|
|
|
#define UDPBUF ((struct udp_iphdr_s *)&dev->d_buf[NET_LL_HDRLEN(dev)])
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Variables
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Variables
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
2014-06-25 02:55:01 +02:00
|
|
|
* Name: udp_send
|
2007-11-07 00:38:14 +01:00
|
|
|
*
|
|
|
|
* Description:
|
2014-04-12 20:13:01 +02:00
|
|
|
* Set-up to send a UDP packet
|
2007-11-07 00:38:14 +01:00
|
|
|
*
|
|
|
|
* Parameters:
|
2014-04-12 20:13:01 +02:00
|
|
|
* dev - The device driver structure to use in the send operation
|
2007-11-07 00:38:14 +01:00
|
|
|
* conn - The UDP "connection" structure holding port information
|
|
|
|
*
|
|
|
|
* Return:
|
|
|
|
* None
|
|
|
|
*
|
|
|
|
* Assumptions:
|
|
|
|
* Called from the interrupt level or with interrupts disabled.
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2014-06-28 00:48:12 +02:00
|
|
|
void udp_send(struct net_driver_s *dev, struct udp_conn_s *conn)
|
2007-11-07 00:38:14 +01:00
|
|
|
{
|
2014-06-25 02:55:01 +02:00
|
|
|
FAR struct udp_iphdr_s *pudpbuf = UDPBUF;
|
2009-03-08 15:42:36 +01:00
|
|
|
|
2007-11-23 14:31:28 +01:00
|
|
|
if (dev->d_sndlen > 0)
|
2007-11-07 00:38:14 +01:00
|
|
|
{
|
2014-04-12 20:13:01 +02:00
|
|
|
/* The total length to send is the size of the application data plus
|
|
|
|
* the IP and UDP headers (and, eventually, the Ethernet header)
|
2007-11-07 00:38:14 +01:00
|
|
|
*/
|
|
|
|
|
2014-07-05 21:04:48 +02:00
|
|
|
dev->d_len = dev->d_sndlen + IPUDP_HDRLEN;
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
/* Initialize the IP header. Note that for IPv6, the IP length field
|
|
|
|
* does not include the IPv6 IP header length.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef CONFIG_NET_IPv6
|
|
|
|
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->vtc = 0x60;
|
|
|
|
pudpbuf->tcf = 0x00;
|
|
|
|
pudpbuf->flow = 0x00;
|
|
|
|
pudpbuf->len[0] = (dev->d_sndlen >> 8);
|
|
|
|
pudpbuf->len[1] = (dev->d_sndlen & 0xff);
|
2014-07-05 03:13:08 +02:00
|
|
|
pudpbuf->nexthdr = IP_PROTO_UDP;
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->hoplimit = conn->ttl;
|
2007-11-07 00:38:14 +01:00
|
|
|
|
2014-06-29 21:11:08 +02:00
|
|
|
net_ipaddr_copy(pudpbuf->srcipaddr, &dev->d_ipaddr);
|
|
|
|
net_ipaddr_copy(pudpbuf->destipaddr, &conn->ripaddr);
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
#else /* CONFIG_NET_IPv6 */
|
|
|
|
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->vhl = 0x45;
|
|
|
|
pudpbuf->tos = 0;
|
|
|
|
pudpbuf->len[0] = (dev->d_len >> 8);
|
|
|
|
pudpbuf->len[1] = (dev->d_len & 0xff);
|
2007-11-07 00:38:14 +01:00
|
|
|
++g_ipid;
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->ipid[0] = g_ipid >> 8;
|
|
|
|
pudpbuf->ipid[1] = g_ipid & 0xff;
|
|
|
|
pudpbuf->ipoffset[0] = 0;
|
|
|
|
pudpbuf->ipoffset[1] = 0;
|
|
|
|
pudpbuf->ttl = conn->ttl;
|
2014-07-05 03:13:08 +02:00
|
|
|
pudpbuf->proto = IP_PROTO_UDP;
|
2007-11-07 00:38:14 +01:00
|
|
|
|
2014-06-29 21:20:00 +02:00
|
|
|
net_ipaddr_hdrcopy(pudpbuf->srcipaddr, &dev->d_ipaddr);
|
|
|
|
net_ipaddr_hdrcopy(pudpbuf->destipaddr, &conn->ripaddr);
|
2008-09-03 16:57:36 +02:00
|
|
|
|
2007-11-07 00:38:14 +01:00
|
|
|
/* Calculate IP checksum. */
|
|
|
|
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->ipchksum = 0;
|
2014-06-28 01:51:32 +02:00
|
|
|
pudpbuf->ipchksum = ~(ip_chksum(dev));
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
#endif /* CONFIG_NET_IPv6 */
|
|
|
|
|
|
|
|
/* Initialize the UDP header */
|
|
|
|
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->srcport = conn->lport;
|
|
|
|
pudpbuf->destport = conn->rport;
|
2014-07-05 21:04:48 +02:00
|
|
|
pudpbuf->udplen = HTONS(dev->d_sndlen + UDP_HDRLEN);
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_NET_UDP_CHECKSUMS
|
|
|
|
/* Calculate UDP checksum. */
|
|
|
|
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->udpchksum = 0;
|
2014-06-25 02:55:01 +02:00
|
|
|
pudpbuf->udpchksum = ~(udp_chksum(dev));
|
2009-03-08 15:42:36 +01:00
|
|
|
if (pudpbuf->udpchksum == 0)
|
2007-11-07 00:38:14 +01:00
|
|
|
{
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->udpchksum = 0xffff;
|
2007-11-07 00:38:14 +01:00
|
|
|
}
|
|
|
|
#else
|
2009-03-08 15:42:36 +01:00
|
|
|
pudpbuf->udpchksum = 0;
|
2007-11-20 00:09:39 +01:00
|
|
|
#endif
|
2007-11-07 00:38:14 +01:00
|
|
|
|
2009-09-15 17:44:14 +02:00
|
|
|
nllvdbg("Outgoing UDP packet length: %d (%d)\n",
|
|
|
|
dev->d_len, (pudpbuf->len[0] << 8) | pudpbuf->len[1]);
|
2007-11-07 00:38:14 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_NET_STATISTICS
|
2014-06-26 17:32:39 +02:00
|
|
|
g_netstats.udp.sent++;
|
|
|
|
g_netstats.ip.sent++;
|
2007-11-07 00:38:14 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* CONFIG_NET && CONFIG_NET_UDP */
|