From d10cd8d585bceb3e1a53d898560a468d3583ca99 Mon Sep 17 00:00:00 2001 From: chao an Date: Fri, 9 Sep 2022 11:13:15 +0800 Subject: [PATCH] net/procfs: add udp profs support add proc node /proc/net/udp to show the active udp connections from procfs nuttx style: server> cat /proc/net/udp UDP sl local_address remote_address flg txsz rxsz (IPv4) 1: 0.0.0.0: 5285 192.168.31.22: 7779 9 0 1041 2: 0.0.0.0: 5286 192.168.31.22: 7779 9 0 1041 3: 0.0.0.0: 5287 192.168.31.22: 7779 9 0 1041 linux kernel: https://www.kernel.org/doc/Documentation/networking/proc_net_tcp.txt Signed-off-by: chao an --- net/procfs/Make.defs | 3 + net/procfs/net_procfs.c | 8 ++ net/procfs/net_udp.c | 209 ++++++++++++++++++++++++++++++++++++++++ net/procfs/procfs.h | 23 +++++ 4 files changed, 243 insertions(+) create mode 100644 net/procfs/net_udp.c diff --git a/net/procfs/Make.defs b/net/procfs/Make.defs index e00db6f09d..cf7fb43bcf 100644 --- a/net/procfs/Make.defs +++ b/net/procfs/Make.defs @@ -36,6 +36,9 @@ endif ifeq ($(CONFIG_NET_TCP),y) NET_CSRCS += net_tcp.c endif +ifeq ($(CONFIG_NET_UDP),y) + NET_CSRCS += net_udp.c +endif endif # Routing table diff --git a/net/procfs/net_procfs.c b/net/procfs/net_procfs.c index 353ccad1e6..bd862fef63 100644 --- a/net/procfs/net_procfs.c +++ b/net/procfs/net_procfs.c @@ -132,6 +132,14 @@ static const struct netprocfs_entry_s g_net_entries[] = } }, # endif +# ifdef CONFIG_NET_UDP + { + DTYPE_FILE, "udp", + { + netprocfs_read_udpstats + } + }, +# endif #endif #ifdef CONFIG_NET_ROUTE { diff --git a/net/procfs/net_udp.c b/net/procfs/net_udp.c new file mode 100644 index 0000000000..d78faf276e --- /dev/null +++ b/net/procfs/net_udp.c @@ -0,0 +1,209 @@ +/**************************************************************************** + * 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 + +#include +#include +#include +#include + +#include +#include + +#include + +#include "procfs/procfs.h" +#include "udp/udp.h" + +/**************************************************************************** + * Pre-processor Definitions + ****************************************************************************/ + +#ifdef CONFIG_NET_UDP + +#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[addrlen + 1]; + char local[addrlen + 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 + iob_get_queue_size(&conn->readahead)); + + 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 */ diff --git a/net/procfs/procfs.h b/net/procfs/procfs.h index 79701f438b..fb4a2cf1bc 100644 --- a/net/procfs/procfs.h +++ b/net/procfs/procfs.h @@ -187,6 +187,29 @@ ssize_t netprocfs_read_tcpstats(FAR struct netprocfs_file_s *priv, FAR char *buffer, size_t buflen); #endif +/**************************************************************************** + * 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. + * + ****************************************************************************/ + +#ifdef CONFIG_NET_UDP +ssize_t netprocfs_read_udpstats(FAR struct netprocfs_file_s *priv, + FAR char *buffer, size_t buflen); +#endif + /**************************************************************************** * Name: netprocfs_read_routes *