qemu-sockets.c 26.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 *  inet and unix socket functions for qemu
 *
 *  (c) 2008 Gerd Hoffmann <kraxel@redhat.com>
 *
 *  This program is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; under version 2 of the License.
 *
 *  This program is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
14 15 16
 *
 * Contributions after 2012-01-13 are licensed under the terms of the
 * GNU GPL, version 2 or (at your option) any later version.
17
 */
18 19 20 21 22 23 24
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <errno.h>
#include <unistd.h>

25
#include "monitor/monitor.h"
26 27
#include "qemu/sockets.h"
#include "qemu/main-loop.h"
28 29 30 31 32 33 34

#ifndef AI_ADDRCONFIG
# define AI_ADDRCONFIG 0
#endif

static const int on=1, off=0;

35 36 37 38
/* used temporarily until all users are converted to QemuOpts */
QemuOptsList socket_optslist = {
    .name = "socket",
    .head = QTAILQ_HEAD_INITIALIZER(socket_optslist.head),
G
Gerd Hoffmann 已提交
39 40 41 42
    .desc = {
        {
            .name = "path",
            .type = QEMU_OPT_STRING,
G
Gerd Hoffmann 已提交
43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
        },{
            .name = "host",
            .type = QEMU_OPT_STRING,
        },{
            .name = "port",
            .type = QEMU_OPT_STRING,
        },{
            .name = "to",
            .type = QEMU_OPT_NUMBER,
        },{
            .name = "ipv4",
            .type = QEMU_OPT_BOOL,
        },{
            .name = "ipv6",
            .type = QEMU_OPT_BOOL,
G
Gerd Hoffmann 已提交
58 59 60 61 62
        },
        { /* end if list */ }
    },
};

63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
static int inet_getport(struct addrinfo *e)
{
    struct sockaddr_in *i4;
    struct sockaddr_in6 *i6;

    switch (e->ai_family) {
    case PF_INET6:
        i6 = (void*)e->ai_addr;
        return ntohs(i6->sin6_port);
    case PF_INET:
        i4 = (void*)e->ai_addr;
        return ntohs(i4->sin_port);
    default:
        return 0;
    }
}

static void inet_setport(struct addrinfo *e, int port)
{
    struct sockaddr_in *i4;
    struct sockaddr_in6 *i6;

    switch (e->ai_family) {
    case PF_INET6:
        i6 = (void*)e->ai_addr;
        i6->sin6_port = htons(port);
        break;
    case PF_INET:
        i4 = (void*)e->ai_addr;
        i4->sin_port = htons(port);
        break;
    }
}

97
const char *inet_strfamily(int family)
98 99 100 101 102 103
{
    switch (family) {
    case PF_INET6: return "ipv6";
    case PF_INET:  return "ipv4";
    case PF_UNIX:  return "unix";
    }
104
    return "unknown";
105 106
}

107
int inet_listen_opts(QemuOpts *opts, int port_offset, Error **errp)
108 109
{
    struct addrinfo ai,*res,*e;
G
Gerd Hoffmann 已提交
110
    const char *addr;
111 112 113
    char port[33];
    char uaddr[INET6_ADDRSTRLEN+1];
    char uport[33];
114
    int slisten, rc, to, port_min, port_max, p;
115 116 117 118

    memset(&ai,0, sizeof(ai));
    ai.ai_flags = AI_PASSIVE | AI_ADDRCONFIG;
    ai.ai_family = PF_UNSPEC;
G
Gerd Hoffmann 已提交
119
    ai.ai_socktype = SOCK_STREAM;
120

121 122
    if ((qemu_opt_get(opts, "host") == NULL) ||
        (qemu_opt_get(opts, "port") == NULL)) {
123
        error_setg(errp, "host and/or port not specified");
G
Gerd Hoffmann 已提交
124
        return -1;
125
    }
G
Gerd Hoffmann 已提交
126 127
    pstrcpy(port, sizeof(port), qemu_opt_get(opts, "port"));
    addr = qemu_opt_get(opts, "host");
128

G
Gerd Hoffmann 已提交
129 130
    to = qemu_opt_get_number(opts, "to", 0);
    if (qemu_opt_get_bool(opts, "ipv4", 0))
131
        ai.ai_family = PF_INET;
G
Gerd Hoffmann 已提交
132
    if (qemu_opt_get_bool(opts, "ipv6", 0))
133 134 135 136 137 138 139
        ai.ai_family = PF_INET6;

    /* lookup */
    if (port_offset)
        snprintf(port, sizeof(port), "%d", atoi(port) + port_offset);
    rc = getaddrinfo(strlen(addr) ? addr : NULL, port, &ai, &res);
    if (rc != 0) {
140 141
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
142 143 144 145 146
        return -1;
    }

    /* create socket + bind */
    for (e = res; e != NULL; e = e->ai_next) {
V
vibi 已提交
147 148 149
        getnameinfo((struct sockaddr*)e->ai_addr,e->ai_addrlen,
		        uaddr,INET6_ADDRSTRLEN,uport,32,
		        NI_NUMERICHOST | NI_NUMERICSERV);
K
Kevin Wolf 已提交
150
        slisten = qemu_socket(e->ai_family, e->ai_socktype, e->ai_protocol);
V
vibi 已提交
151
        if (slisten < 0) {
152
            if (!e->ai_next) {
153
                error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
154
            }
V
vibi 已提交
155 156
            continue;
        }
157

158
        qemu_setsockopt(slisten, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
159 160 161
#ifdef IPV6_V6ONLY
        if (e->ai_family == PF_INET6) {
            /* listen on both ipv4 and ipv6 */
162 163
            qemu_setsockopt(slisten, IPPROTO_IPV6, IPV6_V6ONLY, &off,
                            sizeof(off));
164 165 166
        }
#endif

167 168 169 170
        port_min = inet_getport(e);
        port_max = to ? to + port_offset : port_min;
        for (p = port_min; p <= port_max; p++) {
            inet_setport(e, p);
171 172 173
            if (bind(slisten, e->ai_addr, e->ai_addrlen) == 0) {
                goto listen;
            }
174
            if (p == port_max) {
175
                if (!e->ai_next) {
176
                    error_set_errno(errp, errno, QERR_SOCKET_BIND_FAILED);
177
                }
178 179 180 181 182 183 184 185 186
            }
        }
        closesocket(slisten);
    }
    freeaddrinfo(res);
    return -1;

listen:
    if (listen(slisten,1) != 0) {
187
        error_set_errno(errp, errno, QERR_SOCKET_LISTEN_FAILED);
188
        closesocket(slisten);
V
vibi 已提交
189
        freeaddrinfo(res);
190 191
        return -1;
    }
G
Gerd Hoffmann 已提交
192 193 194 195 196
    snprintf(uport, sizeof(uport), "%d", inet_getport(e) - port_offset);
    qemu_opt_set(opts, "host", uaddr);
    qemu_opt_set(opts, "port", uport);
    qemu_opt_set(opts, "ipv6", (e->ai_family == PF_INET6) ? "on" : "off");
    qemu_opt_set(opts, "ipv4", (e->ai_family != PF_INET6) ? "on" : "off");
197 198 199 200
    freeaddrinfo(res);
    return slisten;
}

201 202 203 204 205 206 207 208
#ifdef _WIN32
#define QEMU_SOCKET_RC_INPROGRESS(rc) \
    ((rc) == -EINPROGRESS || (rc) == -EWOULDBLOCK || (rc) == -WSAEALREADY)
#else
#define QEMU_SOCKET_RC_INPROGRESS(rc) \
    ((rc) == -EINPROGRESS)
#endif

209 210 211 212 213 214 215 216 217 218
/* Struct to store connect state for non blocking connect */
typedef struct ConnectState {
    int fd;
    struct addrinfo *addr_list;
    struct addrinfo *current_addr;
    NonBlockingConnectHandler *callback;
    void *opaque;
} ConnectState;

static int inet_connect_addr(struct addrinfo *addr, bool *in_progress,
219
                             ConnectState *connect_state, Error **errp);
220 221

static void wait_for_connect(void *opaque)
222
{
223 224 225 226 227 228 229 230
    ConnectState *s = opaque;
    int val = 0, rc = 0;
    socklen_t valsize = sizeof(val);
    bool in_progress;

    qemu_set_fd_handler2(s->fd, NULL, NULL, NULL, NULL);

    do {
231
        rc = qemu_getsockopt(s->fd, SOL_SOCKET, SO_ERROR, &val, &valsize);
232 233 234 235 236 237 238 239 240 241 242 243 244 245
    } while (rc == -1 && socket_error() == EINTR);

    /* update rc to contain error */
    if (!rc && val) {
        rc = -1;
    }

    /* connect error */
    if (rc < 0) {
        closesocket(s->fd);
        s->fd = rc;
    }

    /* try to connect to the next address on the list */
246 247 248
    if (s->current_addr) {
        while (s->current_addr->ai_next != NULL && s->fd < 0) {
            s->current_addr = s->current_addr->ai_next;
249
            s->fd = inet_connect_addr(s->current_addr, &in_progress, s, NULL);
250 251 252 253
            /* connect in progress */
            if (in_progress) {
                return;
            }
254
        }
255 256

        freeaddrinfo(s->addr_list);
257
    }
258

259 260
    if (s->callback) {
        s->callback(s->fd, s->opaque);
261
    }
262 263 264 265
    g_free(s);
}

static int inet_connect_addr(struct addrinfo *addr, bool *in_progress,
266
                             ConnectState *connect_state, Error **errp)
267 268 269 270
{
    int sock, rc;

    *in_progress = false;
271 272 273

    sock = qemu_socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol);
    if (sock < 0) {
274
        error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
275 276
        return -1;
    }
277
    qemu_setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
278
    if (connect_state != NULL) {
279
        qemu_set_nonblock(sock);
280 281 282 283 284 285 286 287 288
    }
    /* connect to peer */
    do {
        rc = 0;
        if (connect(sock, addr->ai_addr, addr->ai_addrlen) < 0) {
            rc = -socket_error();
        }
    } while (rc == -EINTR);

289 290 291 292 293
    if (connect_state != NULL && QEMU_SOCKET_RC_INPROGRESS(rc)) {
        connect_state->fd = sock;
        qemu_set_fd_handler2(sock, NULL, NULL, wait_for_connect,
                             connect_state);
        *in_progress = true;
294
    } else if (rc < 0) {
295
        error_set_errno(errp, errno, QERR_SOCKET_CONNECT_FAILED);
296 297 298 299 300 301 302 303 304 305
        closesocket(sock);
        return -1;
    }
    return sock;
}

static struct addrinfo *inet_parse_connect_opts(QemuOpts *opts, Error **errp)
{
    struct addrinfo ai, *res;
    int rc;
G
Gerd Hoffmann 已提交
306 307
    const char *addr;
    const char *port;
308

309
    memset(&ai, 0, sizeof(ai));
310

311 312
    ai.ai_flags = AI_CANONNAME | AI_ADDRCONFIG;
    ai.ai_family = PF_UNSPEC;
G
Gerd Hoffmann 已提交
313
    ai.ai_socktype = SOCK_STREAM;
314

G
Gerd Hoffmann 已提交
315 316 317
    addr = qemu_opt_get(opts, "host");
    port = qemu_opt_get(opts, "port");
    if (addr == NULL || port == NULL) {
318
        error_setg(errp, "host and/or port not specified");
319
        return NULL;
320 321
    }

322
    if (qemu_opt_get_bool(opts, "ipv4", 0)) {
323
        ai.ai_family = PF_INET;
324 325
    }
    if (qemu_opt_get_bool(opts, "ipv6", 0)) {
326
        ai.ai_family = PF_INET6;
327
    }
328 329

    /* lookup */
330 331
    rc = getaddrinfo(addr, port, &ai, &res);
    if (rc != 0) {
332 333
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
334 335 336 337 338
        return NULL;
    }
    return res;
}

339 340 341 342 343 344
/**
 * Create a socket and connect it to an address.
 *
 * @opts: QEMU options, recognized parameters strings "host" and "port",
 *        bools "ipv4" and "ipv6".
 * @errp: set on error
345 346
 * @callback: callback function for non-blocking connect
 * @opaque: opaque for callback function
347 348
 *
 * Returns: -1 on error, file descriptor on success.
349 350 351 352
 *
 * If @callback is non-null, the connect is non-blocking.  If this
 * function succeeds, callback will be called when the connection
 * completes, with the file descriptor on success, or -1 on error.
353
 */
354 355
int inet_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
356 357 358
{
    struct addrinfo *res, *e;
    int sock = -1;
359 360
    bool in_progress;
    ConnectState *connect_state = NULL;
361 362 363 364 365 366

    res = inet_parse_connect_opts(opts, errp);
    if (!res) {
        return -1;
    }

367 368 369 370 371
    if (callback != NULL) {
        connect_state = g_malloc0(sizeof(*connect_state));
        connect_state->addr_list = res;
        connect_state->callback = callback;
        connect_state->opaque = opaque;
372 373 374
    }

    for (e = res; e != NULL; e = e->ai_next) {
375 376 377 378
        if (error_is_set(errp)) {
            error_free(*errp);
            *errp = NULL;
        }
379 380 381
        if (connect_state != NULL) {
            connect_state->current_addr = e;
        }
382
        sock = inet_connect_addr(e, &in_progress, connect_state, errp);
383 384 385 386 387 388 389
        if (in_progress) {
            return sock;
        } else if (sock >= 0) {
            /* non blocking socket immediate success, call callback */
            if (callback != NULL) {
                callback(sock, opaque);
            }
390
            break;
391
        }
392
    }
393
    g_free(connect_state);
394
    freeaddrinfo(res);
395
    return sock;
396 397
}

398
int inet_dgram_opts(QemuOpts *opts, Error **errp)
G
Gerd Hoffmann 已提交
399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
{
    struct addrinfo ai, *peer = NULL, *local = NULL;
    const char *addr;
    const char *port;
    int sock = -1, rc;

    /* lookup peer addr */
    memset(&ai,0, sizeof(ai));
    ai.ai_flags = AI_CANONNAME | AI_ADDRCONFIG;
    ai.ai_family = PF_UNSPEC;
    ai.ai_socktype = SOCK_DGRAM;

    addr = qemu_opt_get(opts, "host");
    port = qemu_opt_get(opts, "port");
    if (addr == NULL || strlen(addr) == 0) {
        addr = "localhost";
    }
    if (port == NULL || strlen(port) == 0) {
417
        error_setg(errp, "remote port not specified");
G
Gerd Hoffmann 已提交
418 419 420 421 422 423 424 425 426
        return -1;
    }

    if (qemu_opt_get_bool(opts, "ipv4", 0))
        ai.ai_family = PF_INET;
    if (qemu_opt_get_bool(opts, "ipv6", 0))
        ai.ai_family = PF_INET6;

    if (0 != (rc = getaddrinfo(addr, port, &ai, &peer))) {
427 428
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
G
Gerd Hoffmann 已提交
429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446
	return -1;
    }

    /* lookup local addr */
    memset(&ai,0, sizeof(ai));
    ai.ai_flags = AI_PASSIVE;
    ai.ai_family = peer->ai_family;
    ai.ai_socktype = SOCK_DGRAM;

    addr = qemu_opt_get(opts, "localaddr");
    port = qemu_opt_get(opts, "localport");
    if (addr == NULL || strlen(addr) == 0) {
        addr = NULL;
    }
    if (!port || strlen(port) == 0)
        port = "0";

    if (0 != (rc = getaddrinfo(addr, port, &ai, &local))) {
447 448
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
449
        goto err;
G
Gerd Hoffmann 已提交
450 451 452
    }

    /* create socket */
K
Kevin Wolf 已提交
453
    sock = qemu_socket(peer->ai_family, peer->ai_socktype, peer->ai_protocol);
G
Gerd Hoffmann 已提交
454
    if (sock < 0) {
455
        error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
G
Gerd Hoffmann 已提交
456 457
        goto err;
    }
458
    qemu_setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &on, sizeof(on));
G
Gerd Hoffmann 已提交
459 460 461

    /* bind socket */
    if (bind(sock, local->ai_addr, local->ai_addrlen) < 0) {
462
        error_set_errno(errp, errno, QERR_SOCKET_BIND_FAILED);
G
Gerd Hoffmann 已提交
463 464 465 466 467
        goto err;
    }

    /* connect to peer */
    if (connect(sock,peer->ai_addr,peer->ai_addrlen) < 0) {
468
        error_set_errno(errp, errno, QERR_SOCKET_CONNECT_FAILED);
G
Gerd Hoffmann 已提交
469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485
        goto err;
    }

    freeaddrinfo(local);
    freeaddrinfo(peer);
    return sock;

err:
    if (-1 != sock)
        closesocket(sock);
    if (local)
        freeaddrinfo(local);
    if (peer)
        freeaddrinfo(peer);
    return -1;
}

G
Gerd Hoffmann 已提交
486
/* compatibility wrapper */
487
InetSocketAddress *inet_parse(const char *str, Error **errp)
G
Gerd Hoffmann 已提交
488
{
489
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
490
    const char *optstr, *h;
491
    char host[64];
G
Gerd Hoffmann 已提交
492
    char port[33];
493
    int to;
G
Gerd Hoffmann 已提交
494 495
    int pos;

496 497
    addr = g_new0(InetSocketAddress, 1);

G
Gerd Hoffmann 已提交
498 499 500
    /* parse address */
    if (str[0] == ':') {
        /* no host given */
501 502
        host[0] = '\0';
        if (1 != sscanf(str, ":%32[^,]%n", port, &pos)) {
503
            error_setg(errp, "error parsing port in address '%s'", str);
504
            goto fail;
G
Gerd Hoffmann 已提交
505 506 507
        }
    } else if (str[0] == '[') {
        /* IPv6 addr */
508
        if (2 != sscanf(str, "[%64[^]]]:%32[^,]%n", host, port, &pos)) {
509
            error_setg(errp, "error parsing IPv6 address '%s'", str);
510
            goto fail;
G
Gerd Hoffmann 已提交
511
        }
512
        addr->ipv6 = addr->has_ipv6 = true;
G
Gerd Hoffmann 已提交
513
    } else {
514
        /* hostname or IPv4 addr */
515
        if (2 != sscanf(str, "%64[^:]:%32[^,]%n", host, port, &pos)) {
516
            error_setg(errp, "error parsing address '%s'", str);
517
            goto fail;
G
Gerd Hoffmann 已提交
518
        }
519 520 521
        if (host[strspn(host, "0123456789.")] == '\0') {
            addr->ipv4 = addr->has_ipv4 = true;
        }
G
Gerd Hoffmann 已提交
522
    }
523 524 525

    addr->host = g_strdup(host);
    addr->port = g_strdup(port);
G
Gerd Hoffmann 已提交
526 527 528 529

    /* parse options */
    optstr = str + pos;
    h = strstr(optstr, ",to=");
530
    if (h) {
531 532 533
        h += 4;
        if (sscanf(h, "%d%n", &to, &pos) != 1 ||
            (h[pos] != '\0' && h[pos] != ',')) {
534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552
            error_setg(errp, "error parsing to= argument");
            goto fail;
        }
        addr->has_to = true;
        addr->to = to;
    }
    if (strstr(optstr, ",ipv4")) {
        addr->ipv4 = addr->has_ipv4 = true;
    }
    if (strstr(optstr, ",ipv6")) {
        addr->ipv6 = addr->has_ipv6 = true;
    }
    return addr;

fail:
    qapi_free_InetSocketAddress(addr);
    return NULL;
}

553
static void inet_addr_to_opts(QemuOpts *opts, const InetSocketAddress *addr)
554 555 556 557 558 559 560 561 562 563 564 565 566 567 568
{
    bool ipv4 = addr->ipv4 || !addr->has_ipv4;
    bool ipv6 = addr->ipv6 || !addr->has_ipv6;

    if (!ipv4 || !ipv6) {
        qemu_opt_set_bool(opts, "ipv4", ipv4);
        qemu_opt_set_bool(opts, "ipv6", ipv6);
    }
    if (addr->has_to) {
        char to[20];
        snprintf(to, sizeof(to), "%d", addr->to);
        qemu_opt_set(opts, "to", to);
    }
    qemu_opt_set(opts, "host", addr->host);
    qemu_opt_set(opts, "port", addr->port);
G
Gerd Hoffmann 已提交
569 570
}

G
Gerd Hoffmann 已提交
571
int inet_listen(const char *str, char *ostr, int olen,
572
                int socktype, int port_offset, Error **errp)
G
Gerd Hoffmann 已提交
573 574 575 576
{
    QemuOpts *opts;
    char *optstr;
    int sock = -1;
577
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
578

579 580
    addr = inet_parse(str, errp);
    if (addr != NULL) {
581
        opts = qemu_opts_create_nofail(&socket_optslist);
582 583
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
584
        sock = inet_listen_opts(opts, port_offset, errp);
G
Gerd Hoffmann 已提交
585 586 587 588 589 590 591 592 593 594 595 596 597 598
        if (sock != -1 && ostr) {
            optstr = strchr(str, ',');
            if (qemu_opt_get_bool(opts, "ipv6", 0)) {
                snprintf(ostr, olen, "[%s]:%s%s",
                         qemu_opt_get(opts, "host"),
                         qemu_opt_get(opts, "port"),
                         optstr ? optstr : "");
            } else {
                snprintf(ostr, olen, "%s:%s%s",
                         qemu_opt_get(opts, "host"),
                         qemu_opt_get(opts, "port"),
                         optstr ? optstr : "");
            }
        }
599
        qemu_opts_del(opts);
G
Gerd Hoffmann 已提交
600 601 602 603
    }
    return sock;
}

604 605 606 607 608 609 610 611 612
/**
 * Create a blocking socket and connect it to an address.
 *
 * @str: address string
 * @errp: set in case of an error
 *
 * Returns -1 in case of error, file descriptor on success
 **/
int inet_connect(const char *str, Error **errp)
G
Gerd Hoffmann 已提交
613 614 615
{
    QemuOpts *opts;
    int sock = -1;
616
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
617

618 619
    addr = inet_parse(str, errp);
    if (addr != NULL) {
620
        opts = qemu_opts_create_nofail(&socket_optslist);
621 622
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
623
        sock = inet_connect_opts(opts, errp, NULL, NULL);
624
        qemu_opts_del(opts);
625 626 627 628 629 630
    }
    return sock;
}

/**
 * Create a non-blocking socket and connect it to an address.
631 632
 * Calls the callback function with fd in case of success or -1 in case of
 * error.
633 634
 *
 * @str: address string
635 636 637
 * @callback: callback function that is called when connect completes,
 *            cannot be NULL.
 * @opaque: opaque for callback function
638 639
 * @errp: set in case of an error
 *
640
 * Returns: -1 on immediate error, file descriptor on success.
641
 **/
642 643 644
int inet_nonblocking_connect(const char *str,
                             NonBlockingConnectHandler *callback,
                             void *opaque, Error **errp)
645 646 647
{
    QemuOpts *opts;
    int sock = -1;
648
    InetSocketAddress *addr;
649

650 651
    g_assert(callback != NULL);

652 653
    addr = inet_parse(str, errp);
    if (addr != NULL) {
654
        opts = qemu_opts_create_nofail(&socket_optslist);
655 656
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
657
        sock = inet_connect_opts(opts, errp, callback, opaque);
658
        qemu_opts_del(opts);
659
    }
G
Gerd Hoffmann 已提交
660 661 662
    return sock;
}

663 664
#ifndef _WIN32

665
int unix_listen_opts(QemuOpts *opts, Error **errp)
666 667
{
    struct sockaddr_un un;
G
Gerd Hoffmann 已提交
668 669
    const char *path = qemu_opt_get(opts, "path");
    int sock, fd;
670

K
Kevin Wolf 已提交
671
    sock = qemu_socket(PF_UNIX, SOCK_STREAM, 0);
672
    if (sock < 0) {
673
        error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
V
vibi 已提交
674
        return -1;
675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692
    }

    memset(&un, 0, sizeof(un));
    un.sun_family = AF_UNIX;
    if (path && strlen(path)) {
        snprintf(un.sun_path, sizeof(un.sun_path), "%s", path);
    } else {
        char *tmpdir = getenv("TMPDIR");
        snprintf(un.sun_path, sizeof(un.sun_path), "%s/qemu-socket-XXXXXX",
                 tmpdir ? tmpdir : "/tmp");
        /*
         * This dummy fd usage silences the mktemp() unsecure warning.
         * Using mkstemp() doesn't make things more secure here
         * though.  bind() complains about existing files, so we have
         * to unlink first and thus re-open the race window.  The
         * worst case possible is bind() failing, i.e. a DoS attack.
         */
        fd = mkstemp(un.sun_path); close(fd);
G
Gerd Hoffmann 已提交
693
        qemu_opt_set(opts, "path", un.sun_path);
694 695 696 697
    }

    unlink(un.sun_path);
    if (bind(sock, (struct sockaddr*) &un, sizeof(un)) < 0) {
698
        error_set_errno(errp, errno, QERR_SOCKET_BIND_FAILED);
699 700 701
        goto err;
    }
    if (listen(sock, 1) < 0) {
702
        error_set_errno(errp, errno, QERR_SOCKET_LISTEN_FAILED);
703 704 705 706 707 708 709 710 711 712
        goto err;
    }

    return sock;

err:
    closesocket(sock);
    return -1;
}

713 714
int unix_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
715 716
{
    struct sockaddr_un un;
G
Gerd Hoffmann 已提交
717
    const char *path = qemu_opt_get(opts, "path");
718 719
    ConnectState *connect_state = NULL;
    int sock, rc;
720

G
Gerd Hoffmann 已提交
721
    if (NULL == path) {
722
        error_setg(errp, "unix connect: no path specified");
G
Gerd Hoffmann 已提交
723 724 725
        return -1;
    }

K
Kevin Wolf 已提交
726
    sock = qemu_socket(PF_UNIX, SOCK_STREAM, 0);
727
    if (sock < 0) {
728
        error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
V
vibi 已提交
729
        return -1;
730
    }
731 732 733 734
    if (callback != NULL) {
        connect_state = g_malloc0(sizeof(*connect_state));
        connect_state->callback = callback;
        connect_state->opaque = opaque;
735
        qemu_set_nonblock(sock);
736
    }
737 738 739 740

    memset(&un, 0, sizeof(un));
    un.sun_family = AF_UNIX;
    snprintf(un.sun_path, sizeof(un.sun_path), "%s", path);
741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762

    /* connect to peer */
    do {
        rc = 0;
        if (connect(sock, (struct sockaddr *) &un, sizeof(un)) < 0) {
            rc = -socket_error();
        }
    } while (rc == -EINTR);

    if (connect_state != NULL && QEMU_SOCKET_RC_INPROGRESS(rc)) {
        connect_state->fd = sock;
        qemu_set_fd_handler2(sock, NULL, NULL, wait_for_connect,
                             connect_state);
        return sock;
    } else if (rc >= 0) {
        /* non blocking socket immediate success, call callback */
        if (callback != NULL) {
            callback(sock, opaque);
        }
    }

    if (rc < 0) {
763
        error_set_errno(errp, -rc, QERR_SOCKET_CONNECT_FAILED);
764
        close(sock);
765
        sock = -1;
766 767
    }

768
    g_free(connect_state);
769 770 771
    return sock;
}

772 773 774 775
#else

int unix_listen_opts(QemuOpts *opts, Error **errp)
{
776
    error_setg(errp, "unix sockets are not available on windows");
777 778 779 780
    errno = ENOTSUP;
    return -1;
}

781 782
int unix_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
783
{
784
    error_setg(errp, "unix sockets are not available on windows");
785 786 787 788 789
    errno = ENOTSUP;
    return -1;
}
#endif

G
Gerd Hoffmann 已提交
790
/* compatibility wrapper */
791
int unix_listen(const char *str, char *ostr, int olen, Error **errp)
G
Gerd Hoffmann 已提交
792 793 794 795 796
{
    QemuOpts *opts;
    char *path, *optstr;
    int sock, len;

797
    opts = qemu_opts_create_nofail(&socket_optslist);
G
Gerd Hoffmann 已提交
798 799 800 801 802

    optstr = strchr(str, ',');
    if (optstr) {
        len = optstr - str;
        if (len) {
803
            path = g_malloc(len+1);
G
Gerd Hoffmann 已提交
804 805
            snprintf(path, len+1, "%.*s", len, str);
            qemu_opt_set(opts, "path", path);
806
            g_free(path);
G
Gerd Hoffmann 已提交
807 808 809 810 811
        }
    } else {
        qemu_opt_set(opts, "path", str);
    }

812
    sock = unix_listen_opts(opts, errp);
G
Gerd Hoffmann 已提交
813 814 815 816 817 818 819

    if (sock != -1 && ostr)
        snprintf(ostr, olen, "%s%s", qemu_opt_get(opts, "path"), optstr ? optstr : "");
    qemu_opts_del(opts);
    return sock;
}

820
int unix_connect(const char *path, Error **errp)
G
Gerd Hoffmann 已提交
821 822 823 824
{
    QemuOpts *opts;
    int sock;

825
    opts = qemu_opts_create_nofail(&socket_optslist);
G
Gerd Hoffmann 已提交
826
    qemu_opt_set(opts, "path", path);
827 828 829 830 831 832 833 834 835 836 837 838 839 840 841
    sock = unix_connect_opts(opts, errp, NULL, NULL);
    qemu_opts_del(opts);
    return sock;
}


int unix_nonblocking_connect(const char *path,
                             NonBlockingConnectHandler *callback,
                             void *opaque, Error **errp)
{
    QemuOpts *opts;
    int sock = -1;

    g_assert(callback != NULL);

842
    opts = qemu_opts_create_nofail(&socket_optslist);
843 844
    qemu_opt_set(opts, "path", path);
    sock = unix_connect_opts(opts, errp, callback, opaque);
G
Gerd Hoffmann 已提交
845 846 847 848
    qemu_opts_del(opts);
    return sock;
}

849 850
SocketAddress *socket_parse(const char *str, Error **errp)
{
851
    SocketAddress *addr;
852

853
    addr = g_new0(SocketAddress, 1);
854 855
    if (strstart(str, "unix:", NULL)) {
        if (str[5] == '\0') {
856
            error_setg(errp, "invalid Unix socket address");
857 858 859 860 861 862 863 864
            goto fail;
        } else {
            addr->kind = SOCKET_ADDRESS_KIND_UNIX;
            addr->q_unix = g_new(UnixSocketAddress, 1);
            addr->q_unix->path = g_strdup(str + 5);
        }
    } else if (strstart(str, "fd:", NULL)) {
        if (str[3] == '\0') {
865
            error_setg(errp, "invalid file descriptor address");
866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891
            goto fail;
        } else {
            addr->kind = SOCKET_ADDRESS_KIND_FD;
            addr->fd = g_new(String, 1);
            addr->fd->str = g_strdup(str + 3);
        }
    } else {
        addr->kind = SOCKET_ADDRESS_KIND_INET;
        addr->inet = inet_parse(str, errp);
        if (addr->inet == NULL) {
            goto fail;
        }
    }
    return addr;

fail:
    qapi_free_SocketAddress(addr);
    return NULL;
}

int socket_connect(SocketAddress *addr, Error **errp,
                   NonBlockingConnectHandler *callback, void *opaque)
{
    QemuOpts *opts;
    int fd;

892
    opts = qemu_opts_create_nofail(&socket_optslist);
893 894 895 896 897 898 899 900 901 902 903 904 905 906
    switch (addr->kind) {
    case SOCKET_ADDRESS_KIND_INET:
        inet_addr_to_opts(opts, addr->inet);
        fd = inet_connect_opts(opts, errp, callback, opaque);
        break;

    case SOCKET_ADDRESS_KIND_UNIX:
        qemu_opt_set(opts, "path", addr->q_unix->path);
        fd = unix_connect_opts(opts, errp, callback, opaque);
        break;

    case SOCKET_ADDRESS_KIND_FD:
        fd = monitor_get_fd(cur_mon, addr->fd->str, errp);
        if (callback) {
907
            qemu_set_nonblock(fd);
908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923
            callback(fd, opaque);
        }
        break;

    default:
        abort();
    }
    qemu_opts_del(opts);
    return fd;
}

int socket_listen(SocketAddress *addr, Error **errp)
{
    QemuOpts *opts;
    int fd;

924
    opts = qemu_opts_create_nofail(&socket_optslist);
925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946
    switch (addr->kind) {
    case SOCKET_ADDRESS_KIND_INET:
        inet_addr_to_opts(opts, addr->inet);
        fd = inet_listen_opts(opts, 0, errp);
        break;

    case SOCKET_ADDRESS_KIND_UNIX:
        qemu_opt_set(opts, "path", addr->q_unix->path);
        fd = unix_listen_opts(opts, errp);
        break;

    case SOCKET_ADDRESS_KIND_FD:
        fd = monitor_get_fd(cur_mon, addr->fd->str, errp);
        break;

    default:
        abort();
    }
    qemu_opts_del(opts);
    return fd;
}

G
Gerd Hoffmann 已提交
947 948 949 950 951
int socket_dgram(SocketAddress *remote, SocketAddress *local, Error **errp)
{
    QemuOpts *opts;
    int fd;

952
    opts = qemu_opts_create_nofail(&socket_optslist);
G
Gerd Hoffmann 已提交
953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970
    switch (remote->kind) {
    case SOCKET_ADDRESS_KIND_INET:
        qemu_opt_set(opts, "host", remote->inet->host);
        qemu_opt_set(opts, "port", remote->inet->port);
        if (local) {
            qemu_opt_set(opts, "localaddr", local->inet->host);
            qemu_opt_set(opts, "localport", local->inet->port);
        }
        fd = inet_dgram_opts(opts, errp);
        break;

    default:
        error_setg(errp, "socket type unsupported for datagram");
        return -1;
    }
    qemu_opts_del(opts);
    return fd;
}