075eb6a6d2
When using IOB queue to store readahead data, we use one IOB for each UDP packet. Then if the packets are very small, like 10Bytes per packet, we'll use ~1600 IOBs just for 16KB recv buffer size, which is wasteful and dangerous. So change conn->readahead to a single IOB chain like TCP. Benefits: - Using memory and IOBs more efficiently (small packets are common in UDP) Side effects: - UDP recv buffer size may count the overhead - A little bit drop in performance (<1%, more seek & copy) Signed-off-by: Zhe Weng <wengzhe@xiaomi.com>
210 lines
6.3 KiB
C
210 lines
6.3 KiB
C
/****************************************************************************
|
|
* net/procfs/net_udp.c
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright ownership. The
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance with the
|
|
* License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
* License for the specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Included Files
|
|
****************************************************************************/
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <debug.h>
|
|
|
|
#include <arpa/inet.h>
|
|
#include <netinet/in.h>
|
|
|
|
#include <nuttx/net/netstats.h>
|
|
|
|
#include "procfs/procfs.h"
|
|
#include "udp/udp.h"
|
|
|
|
/****************************************************************************
|
|
* Pre-processor Definitions
|
|
****************************************************************************/
|
|
|
|
#ifdef NET_UDP_HAVE_STACK
|
|
|
|
#ifdef CONFIG_NET_IPv6
|
|
# define UDP_LINELEN 180
|
|
#else
|
|
# define UDP_LINELEN 120
|
|
#endif
|
|
|
|
/****************************************************************************
|
|
* Private Functions
|
|
****************************************************************************/
|
|
|
|
static ssize_t netprocfs_udpstats(FAR struct netprocfs_file_s *priv,
|
|
FAR char *buffer, size_t buflen,
|
|
uint8_t domain, int *skip)
|
|
{
|
|
int addrlen = (domain == PF_INET) ?
|
|
INET_ADDRSTRLEN : INET6_ADDRSTRLEN;
|
|
FAR struct udp_conn_s *conn = NULL;
|
|
char remote[INET6_ADDRSTRLEN + 1];
|
|
char local[INET6_ADDRSTRLEN + 1];
|
|
int len = 0;
|
|
void *laddr;
|
|
void *raddr;
|
|
|
|
net_lock();
|
|
|
|
local[addrlen] = '\0';
|
|
remote[addrlen] = '\0';
|
|
|
|
while ((conn = udp_nextconn(conn)) != NULL)
|
|
{
|
|
#if defined(CONFIG_NET_IPv4) && defined(CONFIG_NET_IPv6)
|
|
if (conn->domain != domain)
|
|
{
|
|
continue;
|
|
}
|
|
#endif /* CONFIG_NET_IPv4 && CONFIG_NET_IPv6 */
|
|
|
|
if (++(*skip) <= priv->offset)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (buflen - len < UDP_LINELEN)
|
|
{
|
|
break;
|
|
}
|
|
|
|
#ifdef CONFIG_NET_IPv4
|
|
# ifdef CONFIG_NET_IPv6
|
|
if (domain == PF_INET)
|
|
# endif /* CONFIG_NET_IPv6 */
|
|
{
|
|
laddr = &conn->u.ipv4.laddr;
|
|
raddr = &conn->u.ipv4.raddr;
|
|
}
|
|
#endif /* CONFIG_NET_IPv4 */
|
|
|
|
#ifdef CONFIG_NET_IPv6
|
|
# ifdef CONFIG_NET_IPv4
|
|
else
|
|
# endif /* CONFIG_NET_IPv4 */
|
|
{
|
|
laddr = &conn->u.ipv6.laddr;
|
|
raddr = &conn->u.ipv6.raddr;
|
|
}
|
|
#endif /* CONFIG_NET_IPv6 */
|
|
|
|
len += snprintf(buffer + len, buflen - len,
|
|
" %2" PRIu8
|
|
": %3" PRIx8
|
|
#if CONFIG_NET_SEND_BUFSIZE > 0
|
|
" %6" PRIu32
|
|
#endif
|
|
" %6u",
|
|
priv->offset++,
|
|
conn->sconn.s_flags,
|
|
#if CONFIG_NET_SEND_BUFSIZE > 0
|
|
udp_wrbuffer_inqueue_size(conn),
|
|
#endif
|
|
(conn->readahead) ? conn->readahead->io_pktlen : 0);
|
|
|
|
len += snprintf(buffer + len, buflen - len,
|
|
" %*s:%-6" PRIu16 " %*s:%-6" PRIu16 "\n",
|
|
(domain == PF_INET6) ? addrlen / 2 : addrlen,
|
|
inet_ntop(domain, laddr, local, addrlen),
|
|
ntohs(conn->lport),
|
|
(domain == PF_INET6) ? addrlen / 2 : addrlen,
|
|
inet_ntop(domain, raddr, remote, addrlen),
|
|
ntohs(conn->rport));
|
|
}
|
|
|
|
net_unlock();
|
|
|
|
return len;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* Public Functions
|
|
****************************************************************************/
|
|
|
|
/****************************************************************************
|
|
* Name: netprocfs_read_udpstats
|
|
*
|
|
* Description:
|
|
* Read and format UDP statistics.
|
|
*
|
|
* Input Parameters:
|
|
* priv - A reference to the network procfs file structure
|
|
* buffer - The user-provided buffer into which network status will be
|
|
* returned.
|
|
* bulen - The size in bytes of the user provided buffer.
|
|
*
|
|
* Returned Value:
|
|
* Zero (OK) is returned on success; a negated errno value is returned
|
|
* on failure.
|
|
*
|
|
****************************************************************************/
|
|
|
|
ssize_t netprocfs_read_udpstats(FAR struct netprocfs_file_s *priv,
|
|
FAR char *buffer, size_t buflen)
|
|
{
|
|
int skip = 1;
|
|
int len = 0;
|
|
|
|
net_lock();
|
|
|
|
if (udp_nextconn(NULL) != NULL)
|
|
{
|
|
if (priv->offset == 0)
|
|
{
|
|
len = snprintf(buffer, buflen, "UDP sl "
|
|
" flg "
|
|
#if CONFIG_NET_SEND_BUFSIZE > 0
|
|
"txsz "
|
|
#endif
|
|
"rxsz "
|
|
"%-*s "
|
|
"%-*s\n"
|
|
,
|
|
INET6_ADDRSTRLEN / 2,
|
|
"local_address",
|
|
INET6_ADDRSTRLEN / 2,
|
|
"remote_address"
|
|
);
|
|
priv->offset = 1;
|
|
}
|
|
|
|
#ifdef CONFIG_NET_IPv4
|
|
len += netprocfs_udpstats(priv, buffer + len,
|
|
buflen - len, PF_INET, &skip);
|
|
#endif /* CONFIG_NET_IPv4 */
|
|
|
|
#ifdef CONFIG_NET_IPv6
|
|
len += netprocfs_udpstats(priv, buffer + len,
|
|
buflen - len, PF_INET6, &skip);
|
|
#endif /* CONFIG_NET_IPv6 */
|
|
}
|
|
|
|
net_unlock();
|
|
|
|
return len;
|
|
}
|
|
|
|
#endif /* CONFIG_NET_UDP && !CONFIG_NET_UDP_NO_STACK */
|