xref: /qemu/tests/unit/test-util-sockets.c (revision 637de4dba268395f7f2e58e40349468c5e17c060)
158dc31f1SDaniel P. Berrange /*
258dc31f1SDaniel P. Berrange  * Tests for util/qemu-sockets.c
358dc31f1SDaniel P. Berrange  *
458dc31f1SDaniel P. Berrange  * Copyright 2018 Red Hat, Inc.
558dc31f1SDaniel P. Berrange  *
658dc31f1SDaniel P. Berrange  * This program is free software; you can redistribute it and/or modify
758dc31f1SDaniel P. Berrange  * it under the terms of the GNU General Public License as published by
858dc31f1SDaniel P. Berrange  * the Free Software Foundation; either version 2 of the License, or
958dc31f1SDaniel P. Berrange  * (at your option) any later version.
1058dc31f1SDaniel P. Berrange  *
1158dc31f1SDaniel P. Berrange  * This program is distributed in the hope that it will be useful,
1258dc31f1SDaniel P. Berrange  * but WITHOUT ANY WARRANTY; without even the implied warranty of
1358dc31f1SDaniel P. Berrange  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
1458dc31f1SDaniel P. Berrange  * GNU General Public License for more details.
1558dc31f1SDaniel P. Berrange  *
1658dc31f1SDaniel P. Berrange  * You should have received a copy of the GNU General Public License
1758dc31f1SDaniel P. Berrange  * along with this library; if not, see <http://www.gnu.org/licenses/>.
1858dc31f1SDaniel P. Berrange  *
1958dc31f1SDaniel P. Berrange  */
2058dc31f1SDaniel P. Berrange 
2158dc31f1SDaniel P. Berrange #include "qemu/osdep.h"
2258dc31f1SDaniel P. Berrange #include "qemu-common.h"
2358dc31f1SDaniel P. Berrange #include "qemu/sockets.h"
2458dc31f1SDaniel P. Berrange #include "qapi/error.h"
2558dc31f1SDaniel P. Berrange #include "socket-helpers.h"
2630bdb3c5SDaniel P. Berrange #include "monitor/monitor.h"
2758dc31f1SDaniel P. Berrange 
2858dc31f1SDaniel P. Berrange static void test_fd_is_socket_bad(void)
2958dc31f1SDaniel P. Berrange {
3058dc31f1SDaniel P. Berrange     char *tmp = g_strdup("qemu-test-util-sockets-XXXXXX");
3158dc31f1SDaniel P. Berrange     int fd = mkstemp(tmp);
3258dc31f1SDaniel P. Berrange     if (fd != 0) {
3358dc31f1SDaniel P. Berrange         unlink(tmp);
3458dc31f1SDaniel P. Berrange     }
3558dc31f1SDaniel P. Berrange     g_free(tmp);
3658dc31f1SDaniel P. Berrange 
3758dc31f1SDaniel P. Berrange     g_assert(fd >= 0);
3858dc31f1SDaniel P. Berrange 
3958dc31f1SDaniel P. Berrange     g_assert(!fd_is_socket(fd));
4058dc31f1SDaniel P. Berrange     close(fd);
4158dc31f1SDaniel P. Berrange }
4258dc31f1SDaniel P. Berrange 
4358dc31f1SDaniel P. Berrange static void test_fd_is_socket_good(void)
4458dc31f1SDaniel P. Berrange {
4558dc31f1SDaniel P. Berrange     int fd = qemu_socket(PF_INET, SOCK_STREAM, 0);
4658dc31f1SDaniel P. Berrange 
4758dc31f1SDaniel P. Berrange     g_assert(fd >= 0);
4858dc31f1SDaniel P. Berrange 
4958dc31f1SDaniel P. Berrange     g_assert(fd_is_socket(fd));
5058dc31f1SDaniel P. Berrange     close(fd);
5158dc31f1SDaniel P. Berrange }
5258dc31f1SDaniel P. Berrange 
5330bdb3c5SDaniel P. Berrange static int mon_fd = -1;
5430bdb3c5SDaniel P. Berrange static const char *mon_fdname;
5530bdb3c5SDaniel P. Berrange 
5630bdb3c5SDaniel P. Berrange int monitor_get_fd(Monitor *mon, const char *fdname, Error **errp)
5730bdb3c5SDaniel P. Berrange {
5830bdb3c5SDaniel P. Berrange     g_assert(cur_mon);
5930bdb3c5SDaniel P. Berrange     g_assert(mon == cur_mon);
6030bdb3c5SDaniel P. Berrange     if (mon_fd == -1 || !g_str_equal(mon_fdname, fdname)) {
6130bdb3c5SDaniel P. Berrange         error_setg(errp, "No fd named %s", fdname);
6230bdb3c5SDaniel P. Berrange         return -1;
6330bdb3c5SDaniel P. Berrange     }
6430bdb3c5SDaniel P. Berrange     return dup(mon_fd);
6530bdb3c5SDaniel P. Berrange }
6630bdb3c5SDaniel P. Berrange 
6730bdb3c5SDaniel P. Berrange /* Syms in libqemustub.a are discarded at .o file granularity.
6830bdb3c5SDaniel P. Berrange  * To replace monitor_get_fd() we must ensure everything in
6930bdb3c5SDaniel P. Berrange  * stubs/monitor.c is defined, to make sure monitor.o is discarded
7030bdb3c5SDaniel P. Berrange  * otherwise we get duplicate syms at link time.
7130bdb3c5SDaniel P. Berrange  */
7262aa1d88SPeter Xu __thread Monitor *cur_mon;
73*637de4dbSMarkus Armbruster int monitor_vprintf(Monitor *mon, const char *fmt, va_list ap) { abort(); }
7430bdb3c5SDaniel P. Berrange void monitor_init(Chardev *chr, int flags) {}
7530bdb3c5SDaniel P. Berrange 
7630bdb3c5SDaniel P. Berrange 
771723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_good(void)
7830bdb3c5SDaniel P. Berrange {
7930bdb3c5SDaniel P. Berrange     SocketAddress addr;
8030bdb3c5SDaniel P. Berrange     int fd;
8130bdb3c5SDaniel P. Berrange 
8230bdb3c5SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
8330bdb3c5SDaniel P. Berrange     mon_fdname = "myfd";
8430bdb3c5SDaniel P. Berrange     mon_fd = qemu_socket(AF_INET, SOCK_STREAM, 0);
8530bdb3c5SDaniel P. Berrange     g_assert_cmpint(mon_fd, >, STDERR_FILENO);
8630bdb3c5SDaniel P. Berrange 
8730bdb3c5SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
8830bdb3c5SDaniel P. Berrange     addr.u.fd.str = g_strdup(mon_fdname);
8930bdb3c5SDaniel P. Berrange 
9030bdb3c5SDaniel P. Berrange     fd = socket_connect(&addr, &error_abort);
9130bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, -1);
9230bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, mon_fd);
9330bdb3c5SDaniel P. Berrange     close(fd);
9430bdb3c5SDaniel P. Berrange 
9530bdb3c5SDaniel P. Berrange     fd = socket_listen(&addr, &error_abort);
9630bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, -1);
9730bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, mon_fd);
9830bdb3c5SDaniel P. Berrange     close(fd);
9930bdb3c5SDaniel P. Berrange 
10030bdb3c5SDaniel P. Berrange     g_free(addr.u.fd.str);
10130bdb3c5SDaniel P. Berrange     mon_fdname = NULL;
10230bdb3c5SDaniel P. Berrange     close(mon_fd);
10330bdb3c5SDaniel P. Berrange     mon_fd = -1;
10430bdb3c5SDaniel P. Berrange     g_free(cur_mon);
10530bdb3c5SDaniel P. Berrange     cur_mon = NULL;
10630bdb3c5SDaniel P. Berrange }
10730bdb3c5SDaniel P. Berrange 
1081723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_bad(void)
10930bdb3c5SDaniel P. Berrange {
11030bdb3c5SDaniel P. Berrange     SocketAddress addr;
11130bdb3c5SDaniel P. Berrange     Error *err = NULL;
11230bdb3c5SDaniel P. Berrange     int fd;
11330bdb3c5SDaniel P. Berrange 
11430bdb3c5SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
11530bdb3c5SDaniel P. Berrange     mon_fdname = "myfd";
11630bdb3c5SDaniel P. Berrange     mon_fd = dup(STDOUT_FILENO);
11730bdb3c5SDaniel P. Berrange     g_assert_cmpint(mon_fd, >, STDERR_FILENO);
11830bdb3c5SDaniel P. Berrange 
11930bdb3c5SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
12030bdb3c5SDaniel P. Berrange     addr.u.fd.str = g_strdup(mon_fdname);
12130bdb3c5SDaniel P. Berrange 
12230bdb3c5SDaniel P. Berrange     fd = socket_connect(&addr, &err);
12330bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
12430bdb3c5SDaniel P. Berrange     error_free_or_abort(&err);
12530bdb3c5SDaniel P. Berrange 
12630bdb3c5SDaniel P. Berrange     fd = socket_listen(&addr, &err);
12730bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
12830bdb3c5SDaniel P. Berrange     error_free_or_abort(&err);
12930bdb3c5SDaniel P. Berrange 
13030bdb3c5SDaniel P. Berrange     g_free(addr.u.fd.str);
13130bdb3c5SDaniel P. Berrange     mon_fdname = NULL;
13230bdb3c5SDaniel P. Berrange     close(mon_fd);
13330bdb3c5SDaniel P. Berrange     mon_fd = -1;
13430bdb3c5SDaniel P. Berrange     g_free(cur_mon);
13530bdb3c5SDaniel P. Berrange     cur_mon = NULL;
13630bdb3c5SDaniel P. Berrange }
13730bdb3c5SDaniel P. Berrange 
1381723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_nomon(void)
1391723d6b1SDaniel P. Berrange {
1401723d6b1SDaniel P. Berrange     SocketAddress addr;
1411723d6b1SDaniel P. Berrange     Error *err = NULL;
1421723d6b1SDaniel P. Berrange     int fd;
1431723d6b1SDaniel P. Berrange 
1441723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1451723d6b1SDaniel P. Berrange 
1461723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1471723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup("myfd");
1481723d6b1SDaniel P. Berrange 
1491723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
1501723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1511723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1521723d6b1SDaniel P. Berrange 
1531723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
1541723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1551723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1561723d6b1SDaniel P. Berrange 
1571723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
1581723d6b1SDaniel P. Berrange }
1591723d6b1SDaniel P. Berrange 
1601723d6b1SDaniel P. Berrange 
1611723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_good(void)
1621723d6b1SDaniel P. Berrange {
1631723d6b1SDaniel P. Berrange     SocketAddress addr;
1641723d6b1SDaniel P. Berrange     int fd, sfd;
1651723d6b1SDaniel P. Berrange 
1661723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1671723d6b1SDaniel P. Berrange     sfd = qemu_socket(AF_INET, SOCK_STREAM, 0);
1681723d6b1SDaniel P. Berrange     g_assert_cmpint(sfd, >, STDERR_FILENO);
1691723d6b1SDaniel P. Berrange 
1701723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1711723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", sfd);
1721723d6b1SDaniel P. Berrange 
1731723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &error_abort);
1741723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, sfd);
1751723d6b1SDaniel P. Berrange 
1761723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &error_abort);
1771723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, sfd);
1781723d6b1SDaniel P. Berrange 
1791723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
1801723d6b1SDaniel P. Berrange     close(sfd);
1811723d6b1SDaniel P. Berrange }
1821723d6b1SDaniel P. Berrange 
1831723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_bad(void)
1841723d6b1SDaniel P. Berrange {
1851723d6b1SDaniel P. Berrange     SocketAddress addr;
1861723d6b1SDaniel P. Berrange     Error *err = NULL;
1871723d6b1SDaniel P. Berrange     int fd, sfd;
1881723d6b1SDaniel P. Berrange 
1891723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1901723d6b1SDaniel P. Berrange     sfd = dup(STDOUT_FILENO);
1911723d6b1SDaniel P. Berrange 
1921723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1931723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", sfd);
1941723d6b1SDaniel P. Berrange 
1951723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
1961723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1971723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1981723d6b1SDaniel P. Berrange 
1991723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
2001723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2011723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2021723d6b1SDaniel P. Berrange 
2031723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
2041723d6b1SDaniel P. Berrange     close(sfd);
2051723d6b1SDaniel P. Berrange }
2061723d6b1SDaniel P. Berrange 
2071723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_nocli(void)
2081723d6b1SDaniel P. Berrange {
2091723d6b1SDaniel P. Berrange     SocketAddress addr;
2101723d6b1SDaniel P. Berrange     Error *err = NULL;
2111723d6b1SDaniel P. Berrange     int fd;
2121723d6b1SDaniel P. Berrange 
2131723d6b1SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
2141723d6b1SDaniel P. Berrange 
2151723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
2161723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", STDOUT_FILENO);
2171723d6b1SDaniel P. Berrange 
2181723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
2191723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2201723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2211723d6b1SDaniel P. Berrange 
2221723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
2231723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2241723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2251723d6b1SDaniel P. Berrange 
2261723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
2271723d6b1SDaniel P. Berrange }
2281723d6b1SDaniel P. Berrange 
2291723d6b1SDaniel P. Berrange 
23058dc31f1SDaniel P. Berrange int main(int argc, char **argv)
23158dc31f1SDaniel P. Berrange {
23258dc31f1SDaniel P. Berrange     bool has_ipv4, has_ipv6;
23358dc31f1SDaniel P. Berrange 
23458dc31f1SDaniel P. Berrange     socket_init();
23558dc31f1SDaniel P. Berrange 
23658dc31f1SDaniel P. Berrange     g_test_init(&argc, &argv, NULL);
23758dc31f1SDaniel P. Berrange 
23858dc31f1SDaniel P. Berrange     /* We're creating actual IPv4/6 sockets, so we should
23958dc31f1SDaniel P. Berrange      * check if the host running tests actually supports
24058dc31f1SDaniel P. Berrange      * each protocol to avoid breaking tests on machines
24158dc31f1SDaniel P. Berrange      * with either IPv4 or IPv6 disabled.
24258dc31f1SDaniel P. Berrange      */
24358dc31f1SDaniel P. Berrange     if (socket_check_protocol_support(&has_ipv4, &has_ipv6) < 0) {
24458dc31f1SDaniel P. Berrange         return 1;
24558dc31f1SDaniel P. Berrange     }
24658dc31f1SDaniel P. Berrange 
24758dc31f1SDaniel P. Berrange     if (has_ipv4) {
24858dc31f1SDaniel P. Berrange         g_test_add_func("/util/socket/is-socket/bad",
24958dc31f1SDaniel P. Berrange                         test_fd_is_socket_bad);
25058dc31f1SDaniel P. Berrange         g_test_add_func("/util/socket/is-socket/good",
25158dc31f1SDaniel P. Berrange                         test_fd_is_socket_good);
2521723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/good",
2531723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_good);
2541723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/bad",
2551723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_bad);
2561723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/nomon",
2571723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_nomon);
2581723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/good",
2591723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_good);
2601723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/bad",
2611723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_bad);
2621723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/nocli",
2631723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_nocli);
26458dc31f1SDaniel P. Berrange     }
26558dc31f1SDaniel P. Berrange 
26658dc31f1SDaniel P. Berrange     return g_test_run();
26758dc31f1SDaniel P. Berrange }
268