qemu-sockets.c 26.3 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

#ifndef AI_ADDRCONFIG
# define AI_ADDRCONFIG 0
#endif

33 34 35 36
/* used temporarily until all users are converted to QemuOpts */
QemuOptsList socket_optslist = {
    .name = "socket",
    .head = QTAILQ_HEAD_INITIALIZER(socket_optslist.head),
G
Gerd Hoffmann 已提交
37 38 39 40
    .desc = {
        {
            .name = "path",
            .type = QEMU_OPT_STRING,
G
Gerd Hoffmann 已提交
41 42 43 44 45 46 47 48 49 50 51 52 53 54 55
        },{
            .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 已提交
56 57 58 59 60
        },
        { /* end if list */ }
    },
};

61 62 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
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;
    }
}

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

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

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

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

G
Gerd Hoffmann 已提交
127 128
    to = qemu_opt_get_number(opts, "to", 0);
    if (qemu_opt_get_bool(opts, "ipv4", 0))
129
        ai.ai_family = PF_INET;
G
Gerd Hoffmann 已提交
130
    if (qemu_opt_get_bool(opts, "ipv6", 0))
131 132 133 134 135 136 137
        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) {
138 139
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
140 141 142 143 144
        return -1;
    }

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

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

166 167 168 169
        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);
170 171 172
            if (bind(slisten, e->ai_addr, e->ai_addrlen) == 0) {
                goto listen;
            }
173
            if (p == port_max) {
174
                if (!e->ai_next) {
175
                    error_set_errno(errp, errno, QERR_SOCKET_BIND_FAILED);
176
                }
177 178 179 180 181 182 183 184 185
            }
        }
        closesocket(slisten);
    }
    freeaddrinfo(res);
    return -1;

listen:
    if (listen(slisten,1) != 0) {
186
        error_set_errno(errp, errno, QERR_SOCKET_LISTEN_FAILED);
187
        closesocket(slisten);
V
vibi 已提交
188
        freeaddrinfo(res);
189 190
        return -1;
    }
G
Gerd Hoffmann 已提交
191 192 193 194 195
    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");
196 197 198 199
    freeaddrinfo(res);
    return slisten;
}

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

208 209 210 211 212 213 214 215 216 217
/* 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,
218
                             ConnectState *connect_state, Error **errp);
219 220

static void wait_for_connect(void *opaque)
221
{
222 223 224 225 226 227 228 229
    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 {
230
        rc = qemu_getsockopt(s->fd, SOL_SOCKET, SO_ERROR, &val, &valsize);
231 232 233 234 235 236 237 238 239 240 241 242 243 244
    } 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 */
245 246 247
    if (s->current_addr) {
        while (s->current_addr->ai_next != NULL && s->fd < 0) {
            s->current_addr = s->current_addr->ai_next;
248
            s->fd = inet_connect_addr(s->current_addr, &in_progress, s, NULL);
249 250 251 252
            /* connect in progress */
            if (in_progress) {
                return;
            }
253
        }
254 255

        freeaddrinfo(s->addr_list);
256
    }
257

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

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

    *in_progress = false;
270 271 272

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

288 289 290 291 292
    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;
293
    } else if (rc < 0) {
294
        error_set_errno(errp, errno, QERR_SOCKET_CONNECT_FAILED);
295 296 297 298 299 300 301 302 303 304
        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 已提交
305 306
    const char *addr;
    const char *port;
307

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

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

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

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

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

338 339 340 341 342 343
/**
 * 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
344 345
 * @callback: callback function for non-blocking connect
 * @opaque: opaque for callback function
346 347
 *
 * Returns: -1 on error, file descriptor on success.
348 349 350 351
 *
 * 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.
352
 */
353 354
int inet_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
355
{
356
    Error *local_err = NULL;
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
        error_free(local_err);
        local_err = NULL;
377 378 379
        if (connect_state != NULL) {
            connect_state->current_addr = e;
        }
380 381
        sock = inet_connect_addr(e, &in_progress, connect_state, &local_err);
        if (sock >= 0) {
382
            break;
383
        }
384
    }
385 386 387 388 389 390 391 392 393 394 395

    if (sock < 0) {
        error_propagate(errp, local_err);
    } else if (in_progress) {
        /* wait_for_connect() will do the rest */
        return sock;
    } else {
        if (callback) {
            callback(sock, opaque);
        }
    }
396
    g_free(connect_state);
397
    freeaddrinfo(res);
398
    return sock;
399 400
}

401
int inet_dgram_opts(QemuOpts *opts, Error **errp)
G
Gerd Hoffmann 已提交
402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419
{
    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) {
420
        error_setg(errp, "remote port not specified");
G
Gerd Hoffmann 已提交
421 422 423 424 425 426 427 428 429
        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))) {
430 431
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
G
Gerd Hoffmann 已提交
432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449
	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))) {
450 451
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
452
        goto err;
G
Gerd Hoffmann 已提交
453 454 455
    }

    /* create socket */
K
Kevin Wolf 已提交
456
    sock = qemu_socket(peer->ai_family, peer->ai_socktype, peer->ai_protocol);
G
Gerd Hoffmann 已提交
457
    if (sock < 0) {
458
        error_set_errno(errp, errno, QERR_SOCKET_CREATE_FAILED);
G
Gerd Hoffmann 已提交
459 460
        goto err;
    }
461
    socket_set_fast_reuse(sock);
G
Gerd Hoffmann 已提交
462 463 464

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

    /* connect to peer */
    if (connect(sock,peer->ai_addr,peer->ai_addrlen) < 0) {
471
        error_set_errno(errp, errno, QERR_SOCKET_CONNECT_FAILED);
G
Gerd Hoffmann 已提交
472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488
        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 已提交
489
/* compatibility wrapper */
490
InetSocketAddress *inet_parse(const char *str, Error **errp)
G
Gerd Hoffmann 已提交
491
{
492
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
493
    const char *optstr, *h;
494
    char host[64];
G
Gerd Hoffmann 已提交
495
    char port[33];
496
    int to;
G
Gerd Hoffmann 已提交
497 498
    int pos;

499 500
    addr = g_new0(InetSocketAddress, 1);

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

    addr->host = g_strdup(host);
    addr->port = g_strdup(port);
G
Gerd Hoffmann 已提交
529 530 531 532

    /* parse options */
    optstr = str + pos;
    h = strstr(optstr, ",to=");
533
    if (h) {
534 535 536
        h += 4;
        if (sscanf(h, "%d%n", &to, &pos) != 1 ||
            (h[pos] != '\0' && h[pos] != ',')) {
537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555
            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;
}

556
static void inet_addr_to_opts(QemuOpts *opts, const InetSocketAddress *addr)
557 558 559 560 561 562 563 564 565 566 567 568 569 570 571
{
    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 已提交
572 573
}

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

582 583
    addr = inet_parse(str, errp);
    if (addr != NULL) {
584
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
585 586
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
587
        sock = inet_listen_opts(opts, port_offset, errp);
G
Gerd Hoffmann 已提交
588 589 590 591 592 593 594 595 596 597 598 599 600 601
        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 : "");
            }
        }
602
        qemu_opts_del(opts);
G
Gerd Hoffmann 已提交
603 604 605 606
    }
    return sock;
}

607 608 609 610 611 612 613 614 615
/**
 * 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 已提交
616 617 618
{
    QemuOpts *opts;
    int sock = -1;
619
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
620

621 622
    addr = inet_parse(str, errp);
    if (addr != NULL) {
623
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
624 625
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
626
        sock = inet_connect_opts(opts, errp, NULL, NULL);
627
        qemu_opts_del(opts);
628 629 630 631 632 633
    }
    return sock;
}

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

653 654
    g_assert(callback != NULL);

655 656
    addr = inet_parse(str, errp);
    if (addr != NULL) {
657
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
658 659
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
660
        sock = inet_connect_opts(opts, errp, callback, opaque);
661
        qemu_opts_del(opts);
662
    }
G
Gerd Hoffmann 已提交
663 664 665
    return sock;
}

666 667
#ifndef _WIN32

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

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

    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 已提交
696
        qemu_opt_set(opts, "path", un.sun_path);
697 698 699 700
    }

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

    return sock;

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

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

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

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

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

    /* 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) {
766
        error_set_errno(errp, -rc, QERR_SOCKET_CONNECT_FAILED);
767
        close(sock);
768
        sock = -1;
769 770
    }

771
    g_free(connect_state);
772 773 774
    return sock;
}

775 776 777 778
#else

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

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

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

800
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
801 802 803 804 805

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

815
    sock = unix_listen_opts(opts, errp);
G
Gerd Hoffmann 已提交
816 817 818 819 820 821 822

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

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

828
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
829
    qemu_opt_set(opts, "path", path);
830 831 832 833 834 835 836 837 838 839 840 841 842 843 844
    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);

845
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
846 847
    qemu_opt_set(opts, "path", path);
    sock = unix_connect_opts(opts, errp, callback, opaque);
G
Gerd Hoffmann 已提交
848 849 850 851
    qemu_opts_del(opts);
    return sock;
}

852 853
SocketAddress *socket_parse(const char *str, Error **errp)
{
854
    SocketAddress *addr;
855

856
    addr = g_new0(SocketAddress, 1);
857 858
    if (strstart(str, "unix:", NULL)) {
        if (str[5] == '\0') {
859
            error_setg(errp, "invalid Unix socket address");
860 861 862 863 864 865 866 867
            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') {
868
            error_setg(errp, "invalid file descriptor address");
869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894
            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;

895
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
896 897 898 899 900 901 902 903 904 905 906 907 908
    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);
909
        if (fd >= 0 && callback) {
910
            qemu_set_nonblock(fd);
911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926
            callback(fd, opaque);
        }
        break;

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

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

927
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949
    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 已提交
950 951 952 953 954
int socket_dgram(SocketAddress *remote, SocketAddress *local, Error **errp)
{
    QemuOpts *opts;
    int fd;

955
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
956 957 958 959 960 961 962 963 964 965 966 967 968
    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");
969
        fd = -1;
G
Gerd Hoffmann 已提交
970 971 972 973
    }
    qemu_opts_del(opts);
    return fd;
}