This is an automated email from the ASF dual-hosted git repository.
xiaoxiang pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/nuttx-apps.git
The following commit(s) were added to refs/heads/master by this push:
new 098f5836f apps/system: Add tcpdump command
098f5836f is described below
commit 098f5836f9a02a044dfd32542e5891e50493d95c
Author: Zhe Weng <[email protected]>
AuthorDate: Mon Feb 6 19:35:54 2023 +0800
apps/system: Add tcpdump command
Signed-off-by: Zhe Weng <[email protected]>
---
system/tcpdump/Kconfig | 31 +++++
system/tcpdump/Make.defs | 23 ++++
system/tcpdump/Makefile | 30 +++++
system/tcpdump/tcpdump.c | 314 +++++++++++++++++++++++++++++++++++++++++++++++
4 files changed, 398 insertions(+)
diff --git a/system/tcpdump/Kconfig b/system/tcpdump/Kconfig
new file mode 100644
index 000000000..5683d5a20
--- /dev/null
+++ b/system/tcpdump/Kconfig
@@ -0,0 +1,31 @@
+#
+# For a description of the syntax of this configuration file,
+# see the file kconfig-language.txt in the NuttX tools repository.
+#
+
+config SYSTEM_TCPDUMP
+ tristate "tcpdump command"
+ default n
+ depends on NET_PKT
+ select SYSTEM_ARGTABLE3
+ ---help---
+ Enable support for the 'tcpdump' command.
+
+if SYSTEM_TCPDUMP
+
+config SYSTEM_TCPDUMP_PROGNAME
+ string "tcpdump program name"
+ default "tcpdump"
+ ---help---
+ This is the name of the program that will be used when the NSH
ELF
+ program is installed.
+
+config SYSTEM_TCPDUMP_PRIORITY
+ int "tcpdump task priority"
+ default 100
+
+config SYSTEM_TCPDUMP_STACKSIZE
+ int "tcpdump stack size"
+ default DEFAULT_TASK_STACKSIZE
+
+endif
diff --git a/system/tcpdump/Make.defs b/system/tcpdump/Make.defs
new file mode 100644
index 000000000..37e90067e
--- /dev/null
+++ b/system/tcpdump/Make.defs
@@ -0,0 +1,23 @@
+############################################################################
+# apps/system/tcpdump/Make.defs
+#
+# 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.
+#
+############################################################################
+
+ifneq ($(CONFIG_SYSTEM_TCPDUMP),)
+CONFIGURED_APPS += $(APPDIR)/system/tcpdump
+endif
diff --git a/system/tcpdump/Makefile b/system/tcpdump/Makefile
new file mode 100644
index 000000000..4968b15ec
--- /dev/null
+++ b/system/tcpdump/Makefile
@@ -0,0 +1,30 @@
+############################################################################
+# apps/system/tcpdump/Makefile
+#
+# 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.
+#
+############################################################################
+
+include $(APPDIR)/Make.defs
+
+PROGNAME = $(CONFIG_SYSTEM_TCPDUMP_PROGNAME)
+PRIORITY = $(CONFIG_SYSTEM_TCPDUMP_PRIORITY)
+STACKSIZE = $(CONFIG_SYSTEM_TCPDUMP_STACKSIZE)
+MODULE = $(CONFIG_SYSTEM_TCPDUMP)
+
+MAINSRC = tcpdump.c
+
+include $(APPDIR)/Application.mk
diff --git a/system/tcpdump/tcpdump.c b/system/tcpdump/tcpdump.c
new file mode 100644
index 000000000..54b078818
--- /dev/null
+++ b/system/tcpdump/tcpdump.c
@@ -0,0 +1,314 @@
+/****************************************************************************
+ * apps/system/tcpdump/tcpdump.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 <errno.h>
+#include <fcntl.h>
+#include <net/if.h>
+#include <netpacket/packet.h>
+#include <stdint.h>
+#include <stdio.h>
+#include <sys/types.h>
+#include <unistd.h>
+
+#include <nuttx/net/netconfig.h>
+
+#include "argtable3.h"
+
+/****************************************************************************
+ * Pre-processor Definitions
+ ****************************************************************************/
+
+#define TCPDUMP_MAGIC 0xa1b23c4d /* nanosecond-resolution */
+
+#define TCPDUMP_VERSION_MAJOR 2
+#define TCPDUMP_VERSION_MINOR 4
+
+#define DEFAULT_SNAPLEN 262144
+
+#define LINKTYPE_ETHERNET 1
+
+/****************************************************************************
+ * Private Types
+ ****************************************************************************/
+
+struct pcap_filehdr_s
+{
+ uint32_t magic; /* magic number */
+ uint16_t version_major; /* major version number */
+ uint16_t version_minor; /* minor version number */
+ int32_t thiszone; /* GMT to local correction; this is always 0 */
+ uint32_t sigfigs; /* accuracy of timestamps; this is always 0 */
+ uint32_t snaplen; /* max length saved portion of each pkt */
+ uint32_t linktype; /* data link type (LINKTYPE_*) */
+};
+
+struct pcap_pkthdr_s
+{
+ uint32_t ts_sec; /* timestamp seconds */
+ uint32_t ts_nsec; /* timestamp nanoseconds */
+ uint32_t caplen; /* length of portion present */
+ uint32_t len; /* length of this packet (off wire) */
+};
+
+struct tcpdump_args_s
+{
+ FAR struct arg_str *interface;
+ FAR struct arg_str *file;
+ FAR struct arg_int *snaplen;
+ FAR struct arg_end *end;
+};
+
+struct tcpdump_cfgs_s
+{
+ int fd;
+ int sd;
+ uint32_t snaplen;
+};
+
+/****************************************************************************
+ * Private Data
+ ****************************************************************************/
+
+static volatile bool g_exiting;
+
+/****************************************************************************
+ * Private Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: sigexit
+ ****************************************************************************/
+
+static void sigexit(int signo)
+{
+ g_exiting = true;
+}
+
+/****************************************************************************
+ * Name: write_filehdr
+ ****************************************************************************/
+
+static int write_filehdr(int fd, uint32_t snaplen)
+{
+ /* No need to change byte order of any field, reader will swap all fields
+ * if magic number is in swapped order.
+ */
+
+ struct pcap_filehdr_s hdr =
+ {
+ TCPDUMP_MAGIC, /* magic */
+ TCPDUMP_VERSION_MAJOR, /* version_major */
+ TCPDUMP_VERSION_MINOR, /* version_minor */
+ 0, /* thiszone */
+ 0, /* sigfigs */
+ snaplen, /* snaplen */
+ LINKTYPE_ETHERNET /* linktype */
+ };
+
+ /* Write hdr into file. */
+
+ if (write(fd, &hdr, sizeof(hdr)) < 0)
+ {
+ perror("ERROR: write() failed");
+ return -errno;
+ }
+
+ return OK;
+}
+
+/****************************************************************************
+ * Name: write_packet
+ ****************************************************************************/
+
+static int write_packet(int fd, uint32_t snaplen, uint32_t pkt_len,
+ FAR const void *buf, FAR const struct timespec *ts)
+{
+ struct pcap_pkthdr_s hdr =
+ {
+ ts->tv_sec, /* ts_sec */
+ ts->tv_nsec, /* ts_nsec */
+ MIN(snaplen, pkt_len), /* caplen */
+ pkt_len /* len */
+ };
+
+ /* Write hdr into file. */
+
+ if (write(fd, &hdr, sizeof(hdr)) < 0)
+ {
+ perror("ERROR: write() failed");
+ return -errno;
+ }
+
+ /* Write pkt into file. */
+
+ if (write(fd, buf, hdr.caplen) < 0)
+ {
+ perror("ERROR: write() failed");
+ return -errno;
+ }
+
+ return OK;
+}
+
+/****************************************************************************
+ * Name: socket_open
+ ****************************************************************************/
+
+static int socket_open(int ifindex)
+{
+ int sd;
+ struct sockaddr_ll addr;
+
+ sd = socket(PF_PACKET, SOCK_RAW, 0);
+ if (sd < 0)
+ {
+ perror("ERROR: failed to create packet socket");
+ return -errno;
+ }
+
+ /* Prepare sockaddr struct */
+
+ addr.sll_family = AF_PACKET;
+ addr.sll_ifindex = ifindex;
+ if (bind(sd, (FAR const struct sockaddr *)&addr, sizeof(addr)) < 0)
+ {
+ perror("ERROR: binding socket failed");
+ close(sd);
+ return -errno;
+ }
+
+ return sd;
+}
+
+/****************************************************************************
+ * Name: do_capture
+ ****************************************************************************/
+
+static void do_capture(FAR const struct tcpdump_cfgs_s *cfgs)
+{
+ ssize_t len;
+ uint8_t buf[MAX_NETDEV_PKTSIZE];
+ struct timespec ts;
+
+ /* Write file header */
+
+ if (write_filehdr(cfgs->fd, cfgs->snaplen) < 0)
+ {
+ return;
+ }
+
+ /* Dump packets */
+
+ while ((len = read(cfgs->sd, buf, sizeof(buf))) >= 0 && !g_exiting)
+ {
+ if (len == 0)
+ {
+ continue;
+ }
+
+ if (clock_gettime(CLOCK_REALTIME, &ts) < 0)
+ {
+ perror("ERROR: clock_gettime() failed");
+ return;
+ }
+
+ if (write_packet(cfgs->fd, cfgs->snaplen, len, buf, &ts) < 0)
+ {
+ return;
+ }
+ }
+
+ if (!g_exiting)
+ {
+ perror("ERROR: read() failed");
+ }
+}
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+int main(int argc, FAR char *argv[])
+{
+ int ifindex;
+ int nerrors;
+ struct tcpdump_cfgs_s cfgs;
+ struct tcpdump_args_s args;
+
+ g_exiting = false;
+ signal(SIGINT, sigexit);
+
+ args.interface = arg_str1("i", "interface", "interface", "Capture device");
+ args.file = arg_str1("w", NULL, "file", "Path to dump file");
+ args.snaplen = arg_int0("s", "snapshot-length", "snaplen",
+ "Max dump length of each packet");
+ args.end = arg_end(3);
+
+ nerrors = arg_parse(argc, argv, (FAR void**)&args);
+ if (nerrors != 0)
+ {
+ arg_print_errors(stdout, args.end, argv[0]);
+ printf("Usage:\n");
+ arg_print_glossary(stdout, (FAR void**)&args, " %-30s %s\n");
+ return 0;
+ }
+
+ ifindex = if_nametoindex(args.interface->sval[0]);
+ if (ifindex == 0)
+ {
+ printf("Failed to get index of device %s\n", args.interface->sval[0]);
+ return 0;
+ }
+
+ cfgs.fd = open(args.file->sval[0], O_WRONLY | O_CREAT | O_TRUNC, 0644);
+ if (cfgs.fd < 0)
+ {
+ perror("ERROR: open() failed");
+ return 0;
+ }
+
+ cfgs.sd = socket_open(ifindex);
+ if (cfgs.sd < 0)
+ {
+ close(cfgs.fd);
+ return 0;
+ }
+
+ if (args.snaplen->count > 0)
+ {
+ cfgs.snaplen = *args.snaplen->ival;
+ }
+ else
+ {
+ cfgs.snaplen = DEFAULT_SNAPLEN;
+ }
+
+ do_capture(&cfgs);
+
+ close(cfgs.sd);
+ close(cfgs.fd);
+ return 0;
+}