]> git.proxmox.com Git - mirror_qemu.git/blame - tests/test-util-sockets.c
test-util-sockets: Factor out test_socket_unix_abstract_one()
[mirror_qemu.git] / tests / test-util-sockets.c
CommitLineData
58dc31f1
DB
1/*
2 * Tests for util/qemu-sockets.c
3 *
4 * Copyright 2018 Red Hat, Inc.
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this library; if not, see <http://www.gnu.org/licenses/>.
18 *
19 */
20
21#include "qemu/osdep.h"
22#include "qemu-common.h"
23#include "qemu/sockets.h"
24#include "qapi/error.h"
25#include "socket-helpers.h"
30bdb3c5 26#include "monitor/monitor.h"
58dc31f1
DB
27
28static void test_fd_is_socket_bad(void)
29{
30 char *tmp = g_strdup("qemu-test-util-sockets-XXXXXX");
31 int fd = mkstemp(tmp);
32 if (fd != 0) {
33 unlink(tmp);
34 }
35 g_free(tmp);
36
37 g_assert(fd >= 0);
38
39 g_assert(!fd_is_socket(fd));
40 close(fd);
41}
42
43static void test_fd_is_socket_good(void)
44{
45 int fd = qemu_socket(PF_INET, SOCK_STREAM, 0);
46
47 g_assert(fd >= 0);
48
49 g_assert(fd_is_socket(fd));
50 close(fd);
51}
52
30bdb3c5
DB
53static int mon_fd = -1;
54static const char *mon_fdname;
947e4744 55__thread Monitor *cur_mon;
30bdb3c5
DB
56
57int monitor_get_fd(Monitor *mon, const char *fdname, Error **errp)
58{
59 g_assert(cur_mon);
60 g_assert(mon == cur_mon);
61 if (mon_fd == -1 || !g_str_equal(mon_fdname, fdname)) {
62 error_setg(errp, "No fd named %s", fdname);
63 return -1;
64 }
65 return dup(mon_fd);
66}
67
d2a71d74 68/*
947e4744
KW
69 * Syms of stubs in libqemuutil.a are discarded at .o file
70 * granularity. To replace monitor_get_fd() and monitor_cur(), we
71 * must ensure that we also replace any other symbol that is used in
72 * the binary and would be taken from the same stub object file,
30bdb3c5
DB
73 * otherwise we get duplicate syms at link time.
74 */
947e4744 75Monitor *monitor_cur(void) { return cur_mon; }
637de4db 76int monitor_vprintf(Monitor *mon, const char *fmt, va_list ap) { abort(); }
30bdb3c5 77
8330bd53 78#ifndef _WIN32
1723d6b1 79static void test_socket_fd_pass_name_good(void)
30bdb3c5
DB
80{
81 SocketAddress addr;
82 int fd;
83
84 cur_mon = g_malloc(1); /* Fake a monitor */
85 mon_fdname = "myfd";
86 mon_fd = qemu_socket(AF_INET, SOCK_STREAM, 0);
87 g_assert_cmpint(mon_fd, >, STDERR_FILENO);
88
89 addr.type = SOCKET_ADDRESS_TYPE_FD;
90 addr.u.fd.str = g_strdup(mon_fdname);
91
92 fd = socket_connect(&addr, &error_abort);
93 g_assert_cmpint(fd, !=, -1);
94 g_assert_cmpint(fd, !=, mon_fd);
95 close(fd);
96
e5b6353c 97 fd = socket_listen(&addr, 1, &error_abort);
30bdb3c5
DB
98 g_assert_cmpint(fd, !=, -1);
99 g_assert_cmpint(fd, !=, mon_fd);
100 close(fd);
101
102 g_free(addr.u.fd.str);
103 mon_fdname = NULL;
104 close(mon_fd);
105 mon_fd = -1;
106 g_free(cur_mon);
107 cur_mon = NULL;
108}
109
1723d6b1 110static void test_socket_fd_pass_name_bad(void)
30bdb3c5
DB
111{
112 SocketAddress addr;
113 Error *err = NULL;
114 int fd;
115
116 cur_mon = g_malloc(1); /* Fake a monitor */
117 mon_fdname = "myfd";
118 mon_fd = dup(STDOUT_FILENO);
119 g_assert_cmpint(mon_fd, >, STDERR_FILENO);
120
121 addr.type = SOCKET_ADDRESS_TYPE_FD;
122 addr.u.fd.str = g_strdup(mon_fdname);
123
124 fd = socket_connect(&addr, &err);
125 g_assert_cmpint(fd, ==, -1);
126 error_free_or_abort(&err);
127
e5b6353c 128 fd = socket_listen(&addr, 1, &err);
30bdb3c5
DB
129 g_assert_cmpint(fd, ==, -1);
130 error_free_or_abort(&err);
131
132 g_free(addr.u.fd.str);
133 mon_fdname = NULL;
134 close(mon_fd);
135 mon_fd = -1;
136 g_free(cur_mon);
137 cur_mon = NULL;
138}
139
1723d6b1
DB
140static void test_socket_fd_pass_name_nomon(void)
141{
142 SocketAddress addr;
143 Error *err = NULL;
144 int fd;
145
146 g_assert(cur_mon == NULL);
147
148 addr.type = SOCKET_ADDRESS_TYPE_FD;
149 addr.u.fd.str = g_strdup("myfd");
150
151 fd = socket_connect(&addr, &err);
152 g_assert_cmpint(fd, ==, -1);
153 error_free_or_abort(&err);
154
e5b6353c 155 fd = socket_listen(&addr, 1, &err);
1723d6b1
DB
156 g_assert_cmpint(fd, ==, -1);
157 error_free_or_abort(&err);
158
159 g_free(addr.u.fd.str);
160}
161
162
163static void test_socket_fd_pass_num_good(void)
164{
165 SocketAddress addr;
166 int fd, sfd;
167
168 g_assert(cur_mon == NULL);
169 sfd = qemu_socket(AF_INET, SOCK_STREAM, 0);
170 g_assert_cmpint(sfd, >, STDERR_FILENO);
171
172 addr.type = SOCKET_ADDRESS_TYPE_FD;
173 addr.u.fd.str = g_strdup_printf("%d", sfd);
174
175 fd = socket_connect(&addr, &error_abort);
176 g_assert_cmpint(fd, ==, sfd);
177
e5b6353c 178 fd = socket_listen(&addr, 1, &error_abort);
1723d6b1
DB
179 g_assert_cmpint(fd, ==, sfd);
180
181 g_free(addr.u.fd.str);
182 close(sfd);
183}
184
185static void test_socket_fd_pass_num_bad(void)
186{
187 SocketAddress addr;
188 Error *err = NULL;
189 int fd, sfd;
190
191 g_assert(cur_mon == NULL);
192 sfd = dup(STDOUT_FILENO);
193
194 addr.type = SOCKET_ADDRESS_TYPE_FD;
195 addr.u.fd.str = g_strdup_printf("%d", sfd);
196
197 fd = socket_connect(&addr, &err);
198 g_assert_cmpint(fd, ==, -1);
199 error_free_or_abort(&err);
200
e5b6353c 201 fd = socket_listen(&addr, 1, &err);
1723d6b1
DB
202 g_assert_cmpint(fd, ==, -1);
203 error_free_or_abort(&err);
204
205 g_free(addr.u.fd.str);
206 close(sfd);
207}
208
209static void test_socket_fd_pass_num_nocli(void)
210{
211 SocketAddress addr;
212 Error *err = NULL;
213 int fd;
214
215 cur_mon = g_malloc(1); /* Fake a monitor */
216
217 addr.type = SOCKET_ADDRESS_TYPE_FD;
218 addr.u.fd.str = g_strdup_printf("%d", STDOUT_FILENO);
219
220 fd = socket_connect(&addr, &err);
221 g_assert_cmpint(fd, ==, -1);
222 error_free_or_abort(&err);
223
e5b6353c 224 fd = socket_listen(&addr, 1, &err);
1723d6b1
DB
225 g_assert_cmpint(fd, ==, -1);
226 error_free_or_abort(&err);
227
228 g_free(addr.u.fd.str);
229}
8330bd53 230#endif
1723d6b1 231
4d3a329a 232#ifdef __linux__
4d3a329a
XZ
233static gpointer unix_server_thread_func(gpointer user_data)
234{
718a9be0
MA
235 SocketAddress *addr = user_data;
236 int fd;
237 int connfd;
4d3a329a
XZ
238 struct sockaddr_un un;
239 socklen_t len = sizeof(un);
240
718a9be0 241 fd = socket_listen(addr, 1, &error_abort);
4d3a329a
XZ
242 g_assert_cmpint(fd, >=, 0);
243 g_assert(fd_is_socket(fd));
244
245 connfd = accept(fd, (struct sockaddr *)&un, &len);
246 g_assert_cmpint(connfd, !=, -1);
9ce22da0 247 close(connfd);
4d3a329a
XZ
248
249 close(fd);
4d3a329a
XZ
250 return NULL;
251}
252
253static gpointer unix_client_thread_func(gpointer user_data)
254{
718a9be0
MA
255 SocketAddress *addr = user_data;
256 int fd;
4d3a329a 257
718a9be0 258 fd = socket_connect(addr, &error_abort);
4d3a329a 259 g_assert_cmpint(fd, >=, 0);
4d3a329a 260 close(fd);
4d3a329a
XZ
261 return NULL;
262}
263
89cb0bb5
MA
264static void test_socket_unix_abstract_one(SocketAddress *addr)
265{
266 GThread *serv, *cli;
267
268 serv = g_thread_new("abstract_unix_server",
269 unix_server_thread_func,
270 addr);
271
272 sleep(1);
273
274 cli = g_thread_new("abstract_unix_client",
275 unix_client_thread_func,
276 addr);
277
278 g_thread_join(cli);
279 g_thread_join(serv);
280}
281
4d3a329a
XZ
282static void test_socket_unix_abstract_good(void)
283{
718a9be0 284 SocketAddress addr;
4d3a329a 285
718a9be0
MA
286 addr.type = SOCKET_ADDRESS_TYPE_UNIX;
287 addr.u.q_unix.path = g_strdup_printf("unix-%d-%u",
288 getpid(), g_random_int());
289 addr.u.q_unix.has_abstract = true;
290 addr.u.q_unix.abstract = true;
4d3a329a
XZ
291
292 /* non tight socklen serv and cli */
718a9be0
MA
293 addr.u.q_unix.has_tight = false;
294 addr.u.q_unix.tight = false;
89cb0bb5 295 test_socket_unix_abstract_one(&addr);
4d3a329a
XZ
296
297 /* tight socklen serv and cli */
718a9be0
MA
298 addr.u.q_unix.has_tight = true;
299 addr.u.q_unix.tight = true;
89cb0bb5 300 test_socket_unix_abstract_one(&addr);
4d3a329a 301
718a9be0 302 g_free(addr.u.q_unix.path);
4d3a329a
XZ
303}
304#endif
1723d6b1 305
58dc31f1
DB
306int main(int argc, char **argv)
307{
308 bool has_ipv4, has_ipv6;
309
8330bd53 310 qemu_init_main_loop(&error_abort);
58dc31f1
DB
311 socket_init();
312
313 g_test_init(&argc, &argv, NULL);
314
315 /* We're creating actual IPv4/6 sockets, so we should
316 * check if the host running tests actually supports
317 * each protocol to avoid breaking tests on machines
318 * with either IPv4 or IPv6 disabled.
319 */
320 if (socket_check_protocol_support(&has_ipv4, &has_ipv6) < 0) {
a4eb74a6
MAL
321 g_printerr("socket_check_protocol_support() failed\n");
322 goto end;
58dc31f1
DB
323 }
324
325 if (has_ipv4) {
326 g_test_add_func("/util/socket/is-socket/bad",
327 test_fd_is_socket_bad);
328 g_test_add_func("/util/socket/is-socket/good",
329 test_fd_is_socket_good);
8330bd53 330#ifndef _WIN32
1723d6b1
DB
331 g_test_add_func("/socket/fd-pass/name/good",
332 test_socket_fd_pass_name_good);
333 g_test_add_func("/socket/fd-pass/name/bad",
334 test_socket_fd_pass_name_bad);
335 g_test_add_func("/socket/fd-pass/name/nomon",
336 test_socket_fd_pass_name_nomon);
337 g_test_add_func("/socket/fd-pass/num/good",
338 test_socket_fd_pass_num_good);
339 g_test_add_func("/socket/fd-pass/num/bad",
340 test_socket_fd_pass_num_bad);
341 g_test_add_func("/socket/fd-pass/num/nocli",
342 test_socket_fd_pass_num_nocli);
8330bd53 343#endif
58dc31f1
DB
344 }
345
4d3a329a
XZ
346#ifdef __linux__
347 g_test_add_func("/util/socket/unix-abstract/good",
348 test_socket_unix_abstract_good);
349#endif
350
a4eb74a6 351end:
58dc31f1
DB
352 return g_test_run();
353}