2 * Helper functions for tests using sockets
4 * Copyright 2015-2018 Red Hat, Inc.
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 2 or
9 * (at your option) version 3 of the License.
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.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, see <http://www.gnu.org/licenses/>.
21 #include "qemu/osdep.h"
22 #include "qemu-common.h"
23 #include "qemu/sockets.h"
24 #include "socket-helpers.h"
27 # define AI_ADDRCONFIG 0
29 #ifndef EAI_ADDRFAMILY
30 # define EAI_ADDRFAMILY 0
34 * @hostname: a DNS name or numeric IP address
36 * Check whether it is possible to bind & connect to ports
37 * on the DNS name or IP address @hostname. If an IP address
38 * is used, it must not be a wildcard address.
40 * Returns 0 on success, -1 on error with errno set
42 static int socket_can_bind_connect(const char *hostname
, int family
)
44 int lfd
= -1, cfd
= -1, afd
= -1;
45 struct addrinfo ai
, *res
= NULL
;
46 struct sockaddr_storage ss
;
47 socklen_t sslen
= sizeof(ss
);
49 socklen_t soerrlen
= sizeof(soerr
);
50 bool check_soerr
= false;
54 memset(&ai
, 0, sizeof(ai
));
55 ai
.ai_flags
= AI_CANONNAME
| AI_ADDRCONFIG
;
56 ai
.ai_family
= family
;
57 ai
.ai_socktype
= SOCK_STREAM
;
60 rc
= getaddrinfo(hostname
, NULL
, &ai
, &res
);
62 if (rc
== EAI_ADDRFAMILY
||
64 errno
= EADDRNOTAVAIL
;
71 lfd
= qemu_socket(res
->ai_family
, res
->ai_socktype
, res
->ai_protocol
);
76 cfd
= qemu_socket(res
->ai_family
, res
->ai_socktype
, res
->ai_protocol
);
81 if (bind(lfd
, res
->ai_addr
, res
->ai_addrlen
) < 0) {
85 if (listen(lfd
, 1) < 0) {
89 if (getsockname(lfd
, (struct sockaddr
*)&ss
, &sslen
) < 0) {
93 qemu_set_nonblock(cfd
);
94 if (connect(cfd
, (struct sockaddr
*)&ss
, sslen
) < 0) {
95 if (errno
== EINPROGRESS
) {
103 afd
= accept(lfd
, (struct sockaddr
*)&ss
, &sslen
);
109 if (qemu_getsockopt(cfd
, SOL_SOCKET
, SO_ERROR
, &soerr
, &soerrlen
) < 0) {
137 int socket_check_protocol_support(bool *has_ipv4
, bool *has_ipv6
)
139 *has_ipv4
= *has_ipv6
= false;
141 if (socket_can_bind_connect("127.0.0.1", PF_INET
) < 0) {
142 if (errno
!= EADDRNOTAVAIL
) {
149 if (socket_can_bind_connect("::1", PF_INET6
) < 0) {
150 if (errno
!= EADDRNOTAVAIL
) {