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