]> git.ipfire.org Git - thirdparty/util-linux.git/commitdiff
tests: (lsfd) add a case testing TCP sockets
authorMasatake YAMATO <yamato@redhat.com>
Sat, 1 Oct 2022 19:36:40 +0000 (04:36 +0900)
committerMasatake YAMATO <yamato@redhat.com>
Sat, 1 Oct 2022 20:41:37 +0000 (05:41 +0900)
This tests INET.LADDR, INET.RADDR, TCP.LADDR, TCP.LPORT,
TCP.RADDR, and TCP.RPORT columns.

Signed-off-by: Masatake YAMATO <yamato@redhat.com>
tests/expected/lsfd/mkfds-tcp [new file with mode: 0644]
tests/helpers/test_mkfds.c
tests/ts/lsfd/mkfds-tcp [new file with mode: 0755]

diff --git a/tests/expected/lsfd/mkfds-tcp b/tests/expected/lsfd/mkfds-tcp
new file mode 100644 (file)
index 0000000..0ea80d2
--- /dev/null
@@ -0,0 +1,4 @@
+    3  TCP   SOCK state=listen laddr=127.0.0.1:34567                                 listen   stream             1  127.0.0.1    0.0.0.0 127.0.0.1:34567     34567       0.0.0.0:0         0
+    4  TCP   SOCK state=established laddr=127.0.0.1:23456 raddr=127.0.0.1:34567 established   stream             0  127.0.0.1  127.0.0.1 127.0.0.1:23456     23456 127.0.0.1:34567     34567
+    5  TCP   SOCK state=established laddr=127.0.0.1:34567 raddr=127.0.0.1:23456 established   stream             0  127.0.0.1  127.0.0.1 127.0.0.1:34567     34567 127.0.0.1:23456     23456
+ASSOC,TYPE,STTYPE,NAME,SOCKSTATE,SOCKTYPE,SOCKLISTENING,INET.LADDR,INET.RADDR,TCP.LADDR,TCP.LPORT,TCP.RADDR,TCP.RPORT: 0
index 05b6db1e79f28fd32c358baf13051c7b908f0bfc..f2d61b9d51f6a1001435670b241c9bd7e5b0154e 100644 (file)
@@ -26,6 +26,8 @@
 #include <linux/if_ether.h>
 #include <linux/if_packet.h>
 #include <net/if.h>
+#include <netinet/in.h>
+#include <netinet/tcp.h>
 #include <sched.h>
 #include <signal.h>
 #include <stdbool.h>
@@ -1223,6 +1225,153 @@ static void *make_unix_in_new_netns(const struct factory *factory, struct fdesc
        return NULL;
 }
 
+static void *make_tcp(const struct factory *factory, struct fdesc fdescs[],
+                                   int argc, char ** argv)
+{
+       struct arg server_port = decode_arg("server-port", factory->params, argc, argv);
+       unsigned short iserver_port = (unsigned short)ARG_INTEGER(server_port);
+       struct arg client_port = decode_arg("client-port", factory->params, argc, argv);
+       unsigned short iclient_port = (unsigned short)ARG_INTEGER(client_port);
+
+       struct sockaddr_in sin, cin;
+       int ssd, csd, asd;
+
+       const int y = 1;
+
+       free_arg(&server_port);
+       free_arg(&client_port);
+
+       ssd = socket(AF_INET, SOCK_STREAM, 0);
+       if (ssd < 0)
+               err(EXIT_FAILURE,
+                   _("failed to make a tcp socket for listening"));
+
+       if (setsockopt(ssd, SOL_SOCKET,
+                      SO_REUSEADDR, (const char *)&y, sizeof(y)) < 0) {
+               int e = errno;
+               close(ssd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to setsockopt(SO_REUSEADDR)");
+       }
+
+       if (ssd != fdescs[0].fd) {
+               if (dup2(ssd, fdescs[0].fd) < 0) {
+                       int e = errno;
+                       close(ssd);
+                       errno = e;
+                       err(EXIT_FAILURE, "failed to dup %d -> %d", ssd, fdescs[0].fd);
+               }
+               close(ssd);
+               ssd = fdescs[0].fd;
+       }
+
+       memset(&sin, 0, sizeof(sin));
+       sin.sin_family = AF_INET;
+       sin.sin_port = htons(iserver_port);
+       sin.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
+       if (bind(ssd, &sin, sizeof(sin)) < 0) {
+               int e = errno;
+               close(ssd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to bind a listening socket");
+       }
+
+       if (listen(ssd, 1) < 0) {
+               int e = errno;
+               close(ssd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to listen a socket");
+       }
+
+       csd = socket(AF_INET, SOCK_STREAM, 0);
+       if (csd < 0) {
+               int e = errno;
+               close(ssd);
+               errno = e;
+               err(EXIT_FAILURE,
+                   _("failed to make a tcp client socket"));
+       }
+
+       if (setsockopt(csd, SOL_SOCKET,
+                      SO_REUSEADDR, (const char *)&y, sizeof(y)) < 0) {
+               int e = errno;
+               close(ssd);
+               close(csd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to setsockopt(SO_REUSEADDR)");
+       }
+
+       if (csd != fdescs[1].fd) {
+               if (dup2(csd, fdescs[1].fd) < 0) {
+                       int e = errno;
+                       close(ssd);
+                       close(csd);
+                       errno = e;
+                       err(EXIT_FAILURE, "failed to dup %d -> %d", csd, fdescs[1].fd);
+               }
+               close(csd);
+               csd = fdescs[1].fd;
+       }
+
+       memset(&cin, 0, sizeof(cin));
+       cin.sin_family = AF_INET;
+       cin.sin_port = htons(iclient_port);
+       cin.sin_addr.s_addr = htonl(INADDR_LOOPBACK);
+       if (bind(csd, &cin, sizeof(cin)) < 0) {
+               int e = errno;
+               close(ssd);
+               close(csd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to bind a client socket");
+       }
+
+       if (connect(csd, &sin, sizeof(sin)) < 0) {
+               int e = errno;
+               close(ssd);
+               close(csd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to connect a client socket to the serer socket");
+       }
+
+       asd = accept(ssd, NULL, NULL);
+       if (asd < 0) {
+               int e = errno;
+               close(ssd);
+               close(csd);
+               errno = e;
+               err(EXIT_FAILURE, "failed to accept a socket from the listening socket");
+       }
+       if (asd != fdescs[2].fd) {
+               if (dup2(asd, fdescs[2].fd) < 0) {
+                       int e = errno;
+                       close(ssd);
+                       close(csd);
+                       errno = e;
+                       err(EXIT_FAILURE, "failed to dup %d -> %d", asd, fdescs[2].fd);
+               }
+               close(asd);
+               asd = fdescs[2].fd;
+       }
+
+       fdescs[0] = (struct fdesc) {
+               .fd    = fdescs[0].fd,
+               .close = close_fdesc,
+               .data  = NULL,
+       };
+       fdescs[1] = (struct fdesc) {
+               .fd    = fdescs[1].fd,
+               .close = close_fdesc,
+               .data  = NULL,
+       };
+       fdescs[2] = (struct fdesc) {
+               .fd    = fdescs[2].fd,
+               .close = close_fdesc,
+               .data  = NULL,
+       };
+
+       return NULL;
+}
+
 #define PARAM_END { .name = NULL, }
 static const struct factory factories[] = {
        {
@@ -1518,6 +1667,29 @@ static const struct factory factories[] = {
                        PARAM_END
                },
        },
+       {
+               .name = "tcp",
+               .desc = "AF_INET+SOCK_STREAM sockets",
+               .priv = false,
+               .N    = 3,
+               .EX_N = 0,
+               .make = make_tcp,
+               .params = (struct parameter []) {
+                       {
+                               .name = "server-port",
+                               .type = PTYPE_INTEGER,
+                               .desc = "TCP port the server may listen",
+                               .defv.integer = 12345,
+                       },
+                       {
+                               .name = "client-port",
+                               .type = PTYPE_INTEGER,
+                               .desc = "TCP port the client may bind",
+                               .defv.integer = 23456,
+                       },
+                       PARAM_END
+               }
+       },
 };
 
 static int count_parameters(const struct factory *factory)
diff --git a/tests/ts/lsfd/mkfds-tcp b/tests/ts/lsfd/mkfds-tcp
new file mode 100755 (executable)
index 0000000..374def7
--- /dev/null
@@ -0,0 +1,49 @@
+#!/bin/bash
+#
+# Copyright (C) 2022 Masatake YAMATO <yamato@redhat.com>
+#
+# This file is part of util-linux.
+#
+# This file is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 2 of the License, or
+# (at your option) any later version.
+#
+# This file is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+TS_TOPDIR="${0%/*}/../.."
+TS_DESC="TCP sockets"
+
+. "$TS_TOPDIR"/functions.sh
+ts_init "$*"
+
+ts_check_test_command "$TS_CMD_LSFD"
+ts_check_test_command "$TS_HELPER_MKFDS"
+
+ts_cd "$TS_OUTDIR"
+
+PID=
+FDS=3
+FDC=4
+FDA=5
+EXPR='(TYPE == "TCP") and (FD >= 3) and (FD <= 5)'
+
+{
+    coproc MKFDS { "$TS_HELPER_MKFDS" tcp $FDS $FDC $FDA \
+                                     server-port=34567 \
+                                     client-port=23456 ; }
+    if read -r -u "${MKFDS[0]}" PID; then
+       ${TS_CMD_LSFD} -n \
+                      -o ASSOC,TYPE,STTYPE,NAME,SOCKSTATE,SOCKTYPE,SOCKLISTENING,INET.LADDR,INET.RADDR,TCP.LADDR,TCP.LPORT,TCP.RADDR,TCP.RPORT \
+                      -p "${PID}" -Q "${EXPR}"
+       echo 'ASSOC,TYPE,STTYPE,NAME,SOCKSTATE,SOCKTYPE,SOCKLISTENING,INET.LADDR,INET.RADDR,TCP.LADDR,TCP.LPORT,TCP.RADDR,TCP.RPORT': $?
+
+       kill -CONT "${PID}"
+       wait "${MKFDS_PID}"
+    fi
+} > "$TS_OUTPUT" 2>&1
+
+ts_finalize