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