xref: /qemu/tests/unit/test-util-sockets.c (revision a4eb74a66a0f34d53127e240c1eeae42073bc558)
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;
73637de4dbSMarkus Armbruster int monitor_vprintf(Monitor *mon, const char *fmt, va_list ap) { abort(); }
74fbfc29e3SKevin Wolf void monitor_init_qmp(Chardev *chr, bool pretty) {}
75fbfc29e3SKevin Wolf void monitor_init_hmp(Chardev *chr, bool use_readline) {}
7630bdb3c5SDaniel P. Berrange 
7730bdb3c5SDaniel P. Berrange 
781723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_good(void)
7930bdb3c5SDaniel P. Berrange {
8030bdb3c5SDaniel P. Berrange     SocketAddress addr;
8130bdb3c5SDaniel P. Berrange     int fd;
8230bdb3c5SDaniel P. Berrange 
8330bdb3c5SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
8430bdb3c5SDaniel P. Berrange     mon_fdname = "myfd";
8530bdb3c5SDaniel P. Berrange     mon_fd = qemu_socket(AF_INET, SOCK_STREAM, 0);
8630bdb3c5SDaniel P. Berrange     g_assert_cmpint(mon_fd, >, STDERR_FILENO);
8730bdb3c5SDaniel P. Berrange 
8830bdb3c5SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
8930bdb3c5SDaniel P. Berrange     addr.u.fd.str = g_strdup(mon_fdname);
9030bdb3c5SDaniel P. Berrange 
9130bdb3c5SDaniel P. Berrange     fd = socket_connect(&addr, &error_abort);
9230bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, -1);
9330bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, mon_fd);
9430bdb3c5SDaniel P. Berrange     close(fd);
9530bdb3c5SDaniel P. Berrange 
9630bdb3c5SDaniel P. Berrange     fd = socket_listen(&addr, &error_abort);
9730bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, -1);
9830bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, !=, mon_fd);
9930bdb3c5SDaniel P. Berrange     close(fd);
10030bdb3c5SDaniel P. Berrange 
10130bdb3c5SDaniel P. Berrange     g_free(addr.u.fd.str);
10230bdb3c5SDaniel P. Berrange     mon_fdname = NULL;
10330bdb3c5SDaniel P. Berrange     close(mon_fd);
10430bdb3c5SDaniel P. Berrange     mon_fd = -1;
10530bdb3c5SDaniel P. Berrange     g_free(cur_mon);
10630bdb3c5SDaniel P. Berrange     cur_mon = NULL;
10730bdb3c5SDaniel P. Berrange }
10830bdb3c5SDaniel P. Berrange 
1091723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_bad(void)
11030bdb3c5SDaniel P. Berrange {
11130bdb3c5SDaniel P. Berrange     SocketAddress addr;
11230bdb3c5SDaniel P. Berrange     Error *err = NULL;
11330bdb3c5SDaniel P. Berrange     int fd;
11430bdb3c5SDaniel P. Berrange 
11530bdb3c5SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
11630bdb3c5SDaniel P. Berrange     mon_fdname = "myfd";
11730bdb3c5SDaniel P. Berrange     mon_fd = dup(STDOUT_FILENO);
11830bdb3c5SDaniel P. Berrange     g_assert_cmpint(mon_fd, >, STDERR_FILENO);
11930bdb3c5SDaniel P. Berrange 
12030bdb3c5SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
12130bdb3c5SDaniel P. Berrange     addr.u.fd.str = g_strdup(mon_fdname);
12230bdb3c5SDaniel P. Berrange 
12330bdb3c5SDaniel P. Berrange     fd = socket_connect(&addr, &err);
12430bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
12530bdb3c5SDaniel P. Berrange     error_free_or_abort(&err);
12630bdb3c5SDaniel P. Berrange 
12730bdb3c5SDaniel P. Berrange     fd = socket_listen(&addr, &err);
12830bdb3c5SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
12930bdb3c5SDaniel P. Berrange     error_free_or_abort(&err);
13030bdb3c5SDaniel P. Berrange 
13130bdb3c5SDaniel P. Berrange     g_free(addr.u.fd.str);
13230bdb3c5SDaniel P. Berrange     mon_fdname = NULL;
13330bdb3c5SDaniel P. Berrange     close(mon_fd);
13430bdb3c5SDaniel P. Berrange     mon_fd = -1;
13530bdb3c5SDaniel P. Berrange     g_free(cur_mon);
13630bdb3c5SDaniel P. Berrange     cur_mon = NULL;
13730bdb3c5SDaniel P. Berrange }
13830bdb3c5SDaniel P. Berrange 
1391723d6b1SDaniel P. Berrange static void test_socket_fd_pass_name_nomon(void)
1401723d6b1SDaniel P. Berrange {
1411723d6b1SDaniel P. Berrange     SocketAddress addr;
1421723d6b1SDaniel P. Berrange     Error *err = NULL;
1431723d6b1SDaniel P. Berrange     int fd;
1441723d6b1SDaniel P. Berrange 
1451723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1461723d6b1SDaniel P. Berrange 
1471723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1481723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup("myfd");
1491723d6b1SDaniel P. Berrange 
1501723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
1511723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1521723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1531723d6b1SDaniel P. Berrange 
1541723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
1551723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1561723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1571723d6b1SDaniel P. Berrange 
1581723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
1591723d6b1SDaniel P. Berrange }
1601723d6b1SDaniel P. Berrange 
1611723d6b1SDaniel P. Berrange 
1621723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_good(void)
1631723d6b1SDaniel P. Berrange {
1641723d6b1SDaniel P. Berrange     SocketAddress addr;
1651723d6b1SDaniel P. Berrange     int fd, sfd;
1661723d6b1SDaniel P. Berrange 
1671723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1681723d6b1SDaniel P. Berrange     sfd = qemu_socket(AF_INET, SOCK_STREAM, 0);
1691723d6b1SDaniel P. Berrange     g_assert_cmpint(sfd, >, STDERR_FILENO);
1701723d6b1SDaniel P. Berrange 
1711723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1721723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", sfd);
1731723d6b1SDaniel P. Berrange 
1741723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &error_abort);
1751723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, sfd);
1761723d6b1SDaniel P. Berrange 
1771723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &error_abort);
1781723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, sfd);
1791723d6b1SDaniel P. Berrange 
1801723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
1811723d6b1SDaniel P. Berrange     close(sfd);
1821723d6b1SDaniel P. Berrange }
1831723d6b1SDaniel P. Berrange 
1841723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_bad(void)
1851723d6b1SDaniel P. Berrange {
1861723d6b1SDaniel P. Berrange     SocketAddress addr;
1871723d6b1SDaniel P. Berrange     Error *err = NULL;
1881723d6b1SDaniel P. Berrange     int fd, sfd;
1891723d6b1SDaniel P. Berrange 
1901723d6b1SDaniel P. Berrange     g_assert(cur_mon == NULL);
1911723d6b1SDaniel P. Berrange     sfd = dup(STDOUT_FILENO);
1921723d6b1SDaniel P. Berrange 
1931723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
1941723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", sfd);
1951723d6b1SDaniel P. Berrange 
1961723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
1971723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
1981723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
1991723d6b1SDaniel P. Berrange 
2001723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
2011723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2021723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2031723d6b1SDaniel P. Berrange 
2041723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
2051723d6b1SDaniel P. Berrange     close(sfd);
2061723d6b1SDaniel P. Berrange }
2071723d6b1SDaniel P. Berrange 
2081723d6b1SDaniel P. Berrange static void test_socket_fd_pass_num_nocli(void)
2091723d6b1SDaniel P. Berrange {
2101723d6b1SDaniel P. Berrange     SocketAddress addr;
2111723d6b1SDaniel P. Berrange     Error *err = NULL;
2121723d6b1SDaniel P. Berrange     int fd;
2131723d6b1SDaniel P. Berrange 
2141723d6b1SDaniel P. Berrange     cur_mon = g_malloc(1); /* Fake a monitor */
2151723d6b1SDaniel P. Berrange 
2161723d6b1SDaniel P. Berrange     addr.type = SOCKET_ADDRESS_TYPE_FD;
2171723d6b1SDaniel P. Berrange     addr.u.fd.str = g_strdup_printf("%d", STDOUT_FILENO);
2181723d6b1SDaniel P. Berrange 
2191723d6b1SDaniel P. Berrange     fd = socket_connect(&addr, &err);
2201723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2211723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2221723d6b1SDaniel P. Berrange 
2231723d6b1SDaniel P. Berrange     fd = socket_listen(&addr, &err);
2241723d6b1SDaniel P. Berrange     g_assert_cmpint(fd, ==, -1);
2251723d6b1SDaniel P. Berrange     error_free_or_abort(&err);
2261723d6b1SDaniel P. Berrange 
2271723d6b1SDaniel P. Berrange     g_free(addr.u.fd.str);
2281723d6b1SDaniel P. Berrange }
2291723d6b1SDaniel P. Berrange 
2301723d6b1SDaniel P. Berrange 
23158dc31f1SDaniel P. Berrange int main(int argc, char **argv)
23258dc31f1SDaniel P. Berrange {
23358dc31f1SDaniel P. Berrange     bool has_ipv4, has_ipv6;
23458dc31f1SDaniel P. Berrange 
23558dc31f1SDaniel P. Berrange     socket_init();
23658dc31f1SDaniel P. Berrange 
23758dc31f1SDaniel P. Berrange     g_test_init(&argc, &argv, NULL);
23858dc31f1SDaniel P. Berrange 
23958dc31f1SDaniel P. Berrange     /* We're creating actual IPv4/6 sockets, so we should
24058dc31f1SDaniel P. Berrange      * check if the host running tests actually supports
24158dc31f1SDaniel P. Berrange      * each protocol to avoid breaking tests on machines
24258dc31f1SDaniel P. Berrange      * with either IPv4 or IPv6 disabled.
24358dc31f1SDaniel P. Berrange      */
24458dc31f1SDaniel P. Berrange     if (socket_check_protocol_support(&has_ipv4, &has_ipv6) < 0) {
245*a4eb74a6SMarc-André Lureau         g_printerr("socket_check_protocol_support() failed\n");
246*a4eb74a6SMarc-André Lureau         goto end;
24758dc31f1SDaniel P. Berrange     }
24858dc31f1SDaniel P. Berrange 
24958dc31f1SDaniel P. Berrange     if (has_ipv4) {
25058dc31f1SDaniel P. Berrange         g_test_add_func("/util/socket/is-socket/bad",
25158dc31f1SDaniel P. Berrange                         test_fd_is_socket_bad);
25258dc31f1SDaniel P. Berrange         g_test_add_func("/util/socket/is-socket/good",
25358dc31f1SDaniel P. Berrange                         test_fd_is_socket_good);
2541723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/good",
2551723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_good);
2561723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/bad",
2571723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_bad);
2581723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/name/nomon",
2591723d6b1SDaniel P. Berrange                         test_socket_fd_pass_name_nomon);
2601723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/good",
2611723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_good);
2621723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/bad",
2631723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_bad);
2641723d6b1SDaniel P. Berrange         g_test_add_func("/socket/fd-pass/num/nocli",
2651723d6b1SDaniel P. Berrange                         test_socket_fd_pass_num_nocli);
26658dc31f1SDaniel P. Berrange     }
26758dc31f1SDaniel P. Berrange 
268*a4eb74a6SMarc-André Lureau end:
26958dc31f1SDaniel P. Berrange     return g_test_run();
27058dc31f1SDaniel P. Berrange }
271