You need to sign in or sign up before continuing.
qemu-sockets.c 27.5 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;
    }
}

W
Wenchao Xia 已提交
95 96 97 98 99 100 101 102 103 104
NetworkAddressFamily inet_netfamily(int family)
{
    switch (family) {
    case PF_INET6: return NETWORK_ADDRESS_FAMILY_IPV6;
    case PF_INET:  return NETWORK_ADDRESS_FAMILY_IPV4;
    case PF_UNIX:  return NETWORK_ADDRESS_FAMILY_UNIX;
    }
    return NETWORK_ADDRESS_FAMILY_UNKNOWN;
}

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
        ai.ai_family = PF_INET6;

    /* lookup */
134 135 136 137 138 139 140 141 142 143 144 145 146
    if (port_offset) {
        unsigned long long baseport;
        if (parse_uint_full(port, &baseport, 10) < 0) {
            error_setg(errp, "can't convert to a number: %s", port);
            return -1;
        }
        if (baseport > 65535 ||
            baseport + port_offset > 65535) {
            error_setg(errp, "port %s out of range", port);
            return -1;
        }
        snprintf(port, sizeof(port), "%d", (int)baseport + port_offset);
    }
147 148
    rc = getaddrinfo(strlen(addr) ? addr : NULL, port, &ai, &res);
    if (rc != 0) {
149 150
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
151 152 153 154 155
        return -1;
    }

    /* create socket + bind */
    for (e = res; e != NULL; e = e->ai_next) {
V
vibi 已提交
156 157 158
        getnameinfo((struct sockaddr*)e->ai_addr,e->ai_addrlen,
		        uaddr,INET6_ADDRSTRLEN,uport,32,
		        NI_NUMERICHOST | NI_NUMERICSERV);
K
Kevin Wolf 已提交
159
        slisten = qemu_socket(e->ai_family, e->ai_socktype, e->ai_protocol);
V
vibi 已提交
160
        if (slisten < 0) {
161
            if (!e->ai_next) {
162
                error_setg_errno(errp, errno, "Failed to create socket");
163
            }
V
vibi 已提交
164 165
            continue;
        }
166

167
        socket_set_fast_reuse(slisten);
168 169 170
#ifdef IPV6_V6ONLY
        if (e->ai_family == PF_INET6) {
            /* listen on both ipv4 and ipv6 */
171
            const int off = 0;
172 173
            qemu_setsockopt(slisten, IPPROTO_IPV6, IPV6_V6ONLY, &off,
                            sizeof(off));
174 175 176
        }
#endif

177 178 179 180
        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);
181 182 183
            if (bind(slisten, e->ai_addr, e->ai_addrlen) == 0) {
                goto listen;
            }
184
            if (p == port_max) {
185
                if (!e->ai_next) {
186
                    error_setg_errno(errp, errno, "Failed to bind socket");
187
                }
188 189 190 191 192 193 194 195 196
            }
        }
        closesocket(slisten);
    }
    freeaddrinfo(res);
    return -1;

listen:
    if (listen(slisten,1) != 0) {
197
        error_setg_errno(errp, errno, "Failed to listen on socket");
198
        closesocket(slisten);
V
vibi 已提交
199
        freeaddrinfo(res);
200 201
        return -1;
    }
G
Gerd Hoffmann 已提交
202 203 204 205 206
    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");
207 208 209 210
    freeaddrinfo(res);
    return slisten;
}

211 212 213 214 215 216 217 218
#ifdef _WIN32
#define QEMU_SOCKET_RC_INPROGRESS(rc) \
    ((rc) == -EINPROGRESS || (rc) == -EWOULDBLOCK || (rc) == -WSAEALREADY)
#else
#define QEMU_SOCKET_RC_INPROGRESS(rc) \
    ((rc) == -EINPROGRESS)
#endif

219 220 221 222 223 224 225 226 227 228
/* 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,
229
                             ConnectState *connect_state, Error **errp);
230 231

static void wait_for_connect(void *opaque)
232
{
233 234 235 236
    ConnectState *s = opaque;
    int val = 0, rc = 0;
    socklen_t valsize = sizeof(val);
    bool in_progress;
237
    Error *err = NULL;
238 239 240 241

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

    do {
242
        rc = qemu_getsockopt(s->fd, SOL_SOCKET, SO_ERROR, &val, &valsize);
243 244 245 246 247
    } while (rc == -1 && socket_error() == EINTR);

    /* update rc to contain error */
    if (!rc && val) {
        rc = -1;
248
        errno = val;
249 250 251 252
    }

    /* connect error */
    if (rc < 0) {
253
        error_setg_errno(&err, errno, "Error connecting to socket");
254 255 256 257 258
        closesocket(s->fd);
        s->fd = rc;
    }

    /* try to connect to the next address on the list */
259 260 261
    if (s->current_addr) {
        while (s->current_addr->ai_next != NULL && s->fd < 0) {
            s->current_addr = s->current_addr->ai_next;
262
            s->fd = inet_connect_addr(s->current_addr, &in_progress, s, NULL);
263 264 265 266 267
            if (s->fd < 0) {
                error_free(err);
                err = NULL;
                error_setg_errno(&err, errno, "Unable to start socket connect");
            }
268 269
            /* connect in progress */
            if (in_progress) {
270
                goto out;
271
            }
272
        }
273 274

        freeaddrinfo(s->addr_list);
275
    }
276

277
    if (s->callback) {
278
        s->callback(s->fd, err, s->opaque);
279
    }
280
    g_free(s);
281 282
out:
    error_free(err);
283 284 285
}

static int inet_connect_addr(struct addrinfo *addr, bool *in_progress,
286
                             ConnectState *connect_state, Error **errp)
287 288 289 290
{
    int sock, rc;

    *in_progress = false;
291 292 293

    sock = qemu_socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol);
    if (sock < 0) {
294
        error_setg_errno(errp, errno, "Failed to create socket");
295 296
        return -1;
    }
297
    socket_set_fast_reuse(sock);
298
    if (connect_state != NULL) {
299
        qemu_set_nonblock(sock);
300 301 302 303 304 305 306 307 308
    }
    /* connect to peer */
    do {
        rc = 0;
        if (connect(sock, addr->ai_addr, addr->ai_addrlen) < 0) {
            rc = -socket_error();
        }
    } while (rc == -EINTR);

309 310 311 312 313
    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;
314
    } else if (rc < 0) {
315
        error_setg_errno(errp, errno, "Failed to connect socket");
316 317 318 319 320 321 322 323 324 325
        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 已提交
326 327
    const char *addr;
    const char *port;
328

329
    memset(&ai, 0, sizeof(ai));
330

331 332
    ai.ai_flags = AI_CANONNAME | AI_ADDRCONFIG;
    ai.ai_family = PF_UNSPEC;
G
Gerd Hoffmann 已提交
333
    ai.ai_socktype = SOCK_STREAM;
334

G
Gerd Hoffmann 已提交
335 336 337
    addr = qemu_opt_get(opts, "host");
    port = qemu_opt_get(opts, "port");
    if (addr == NULL || port == NULL) {
338
        error_setg(errp, "host and/or port not specified");
339
        return NULL;
340 341
    }

342
    if (qemu_opt_get_bool(opts, "ipv4", 0)) {
343
        ai.ai_family = PF_INET;
344 345
    }
    if (qemu_opt_get_bool(opts, "ipv6", 0)) {
346
        ai.ai_family = PF_INET6;
347
    }
348 349

    /* lookup */
350 351
    rc = getaddrinfo(addr, port, &ai, &res);
    if (rc != 0) {
352 353
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
354 355 356 357 358
        return NULL;
    }
    return res;
}

359 360 361 362 363 364
/**
 * 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
365 366
 * @callback: callback function for non-blocking connect
 * @opaque: opaque for callback function
367 368
 *
 * Returns: -1 on error, file descriptor on success.
369 370 371 372
 *
 * 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.
373
 */
374 375
int inet_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
376
{
377
    Error *local_err = NULL;
378 379
    struct addrinfo *res, *e;
    int sock = -1;
380 381
    bool in_progress;
    ConnectState *connect_state = NULL;
382 383 384 385 386 387

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

388 389 390 391 392
    if (callback != NULL) {
        connect_state = g_malloc0(sizeof(*connect_state));
        connect_state->addr_list = res;
        connect_state->callback = callback;
        connect_state->opaque = opaque;
393 394 395
    }

    for (e = res; e != NULL; e = e->ai_next) {
396 397
        error_free(local_err);
        local_err = NULL;
398 399 400
        if (connect_state != NULL) {
            connect_state->current_addr = e;
        }
401 402
        sock = inet_connect_addr(e, &in_progress, connect_state, &local_err);
        if (sock >= 0) {
403
            break;
404
        }
405
    }
406 407 408 409 410 411 412 413

    if (sock < 0) {
        error_propagate(errp, local_err);
    } else if (in_progress) {
        /* wait_for_connect() will do the rest */
        return sock;
    } else {
        if (callback) {
414
            callback(sock, NULL, opaque);
415 416
        }
    }
417
    g_free(connect_state);
418
    freeaddrinfo(res);
419
    return sock;
420 421
}

422
int inet_dgram_opts(QemuOpts *opts, Error **errp)
G
Gerd Hoffmann 已提交
423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440
{
    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) {
441
        error_setg(errp, "remote port not specified");
G
Gerd Hoffmann 已提交
442 443 444 445 446 447 448 449 450
        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))) {
451 452
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
G
Gerd Hoffmann 已提交
453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470
	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))) {
471 472
        error_setg(errp, "address resolution failed for %s:%s: %s", addr, port,
                   gai_strerror(rc));
473
        goto err;
G
Gerd Hoffmann 已提交
474 475 476
    }

    /* create socket */
K
Kevin Wolf 已提交
477
    sock = qemu_socket(peer->ai_family, peer->ai_socktype, peer->ai_protocol);
G
Gerd Hoffmann 已提交
478
    if (sock < 0) {
479
        error_setg_errno(errp, errno, "Failed to create socket");
G
Gerd Hoffmann 已提交
480 481
        goto err;
    }
482
    socket_set_fast_reuse(sock);
G
Gerd Hoffmann 已提交
483 484 485

    /* bind socket */
    if (bind(sock, local->ai_addr, local->ai_addrlen) < 0) {
486
        error_setg_errno(errp, errno, "Failed to bind socket");
G
Gerd Hoffmann 已提交
487 488 489 490 491
        goto err;
    }

    /* connect to peer */
    if (connect(sock,peer->ai_addr,peer->ai_addrlen) < 0) {
492
        error_setg_errno(errp, errno, "Failed to connect socket");
G
Gerd Hoffmann 已提交
493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509
        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 已提交
510
/* compatibility wrapper */
511
InetSocketAddress *inet_parse(const char *str, Error **errp)
G
Gerd Hoffmann 已提交
512
{
513
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
514
    const char *optstr, *h;
515
    char host[65];
G
Gerd Hoffmann 已提交
516
    char port[33];
517
    int to;
G
Gerd Hoffmann 已提交
518 519
    int pos;

520 521
    addr = g_new0(InetSocketAddress, 1);

G
Gerd Hoffmann 已提交
522 523 524
    /* parse address */
    if (str[0] == ':') {
        /* no host given */
525 526
        host[0] = '\0';
        if (1 != sscanf(str, ":%32[^,]%n", port, &pos)) {
527
            error_setg(errp, "error parsing port in address '%s'", str);
528
            goto fail;
G
Gerd Hoffmann 已提交
529 530 531
        }
    } else if (str[0] == '[') {
        /* IPv6 addr */
532
        if (2 != sscanf(str, "[%64[^]]]:%32[^,]%n", host, port, &pos)) {
533
            error_setg(errp, "error parsing IPv6 address '%s'", str);
534
            goto fail;
G
Gerd Hoffmann 已提交
535
        }
536
        addr->ipv6 = addr->has_ipv6 = true;
G
Gerd Hoffmann 已提交
537
    } else {
538
        /* hostname or IPv4 addr */
539
        if (2 != sscanf(str, "%64[^:]:%32[^,]%n", host, port, &pos)) {
540
            error_setg(errp, "error parsing address '%s'", str);
541
            goto fail;
G
Gerd Hoffmann 已提交
542
        }
543 544 545
        if (host[strspn(host, "0123456789.")] == '\0') {
            addr->ipv4 = addr->has_ipv4 = true;
        }
G
Gerd Hoffmann 已提交
546
    }
547 548 549

    addr->host = g_strdup(host);
    addr->port = g_strdup(port);
G
Gerd Hoffmann 已提交
550 551 552 553

    /* parse options */
    optstr = str + pos;
    h = strstr(optstr, ",to=");
554
    if (h) {
555 556 557
        h += 4;
        if (sscanf(h, "%d%n", &to, &pos) != 1 ||
            (h[pos] != '\0' && h[pos] != ',')) {
558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576
            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;
}

577
static void inet_addr_to_opts(QemuOpts *opts, const InetSocketAddress *addr)
578 579 580 581 582 583 584 585 586 587 588 589 590 591 592
{
    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 已提交
593 594
}

G
Gerd Hoffmann 已提交
595
int inet_listen(const char *str, char *ostr, int olen,
596
                int socktype, int port_offset, Error **errp)
G
Gerd Hoffmann 已提交
597 598 599 600
{
    QemuOpts *opts;
    char *optstr;
    int sock = -1;
601
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
602

603 604
    addr = inet_parse(str, errp);
    if (addr != NULL) {
605
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
606 607
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
608
        sock = inet_listen_opts(opts, port_offset, errp);
G
Gerd Hoffmann 已提交
609 610 611 612 613 614 615 616 617 618 619 620 621 622
        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 : "");
            }
        }
623
        qemu_opts_del(opts);
G
Gerd Hoffmann 已提交
624 625 626 627
    }
    return sock;
}

628 629 630 631 632 633 634 635 636
/**
 * 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 已提交
637 638 639
{
    QemuOpts *opts;
    int sock = -1;
640
    InetSocketAddress *addr;
G
Gerd Hoffmann 已提交
641

642 643
    addr = inet_parse(str, errp);
    if (addr != NULL) {
644
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
645 646
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
647
        sock = inet_connect_opts(opts, errp, NULL, NULL);
648
        qemu_opts_del(opts);
649 650 651 652 653 654
    }
    return sock;
}

/**
 * Create a non-blocking socket and connect it to an address.
655 656
 * Calls the callback function with fd in case of success or -1 in case of
 * error.
657 658
 *
 * @str: address string
659 660 661
 * @callback: callback function that is called when connect completes,
 *            cannot be NULL.
 * @opaque: opaque for callback function
662 663
 * @errp: set in case of an error
 *
664
 * Returns: -1 on immediate error, file descriptor on success.
665
 **/
666 667 668
int inet_nonblocking_connect(const char *str,
                             NonBlockingConnectHandler *callback,
                             void *opaque, Error **errp)
669 670 671
{
    QemuOpts *opts;
    int sock = -1;
672
    InetSocketAddress *addr;
673

674 675
    g_assert(callback != NULL);

676 677
    addr = inet_parse(str, errp);
    if (addr != NULL) {
678
        opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
679 680
        inet_addr_to_opts(opts, addr);
        qapi_free_InetSocketAddress(addr);
681
        sock = inet_connect_opts(opts, errp, callback, opaque);
682
        qemu_opts_del(opts);
683
    }
G
Gerd Hoffmann 已提交
684 685 686
    return sock;
}

687 688
#ifndef _WIN32

689
int unix_listen_opts(QemuOpts *opts, Error **errp)
690 691
{
    struct sockaddr_un un;
G
Gerd Hoffmann 已提交
692 693
    const char *path = qemu_opt_get(opts, "path");
    int sock, fd;
694

K
Kevin Wolf 已提交
695
    sock = qemu_socket(PF_UNIX, SOCK_STREAM, 0);
696
    if (sock < 0) {
697
        error_setg_errno(errp, errno, "Failed to create Unix socket");
V
vibi 已提交
698
        return -1;
699 700 701 702 703 704 705
    }

    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 {
706 707 708 709 710 711 712 713 714
        const char *tmpdir = getenv("TMPDIR");
        tmpdir = tmpdir ? tmpdir : "/tmp";
        if (snprintf(un.sun_path, sizeof(un.sun_path), "%s/qemu-socket-XXXXXX",
                     tmpdir) >= sizeof(un.sun_path)) {
            error_setg_errno(errp, errno,
                             "TMPDIR environment variable (%s) too large", tmpdir);
            goto err;
        }

715 716 717 718 719 720 721
        /*
         * 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.
         */
722 723 724 725 726 727 728
        fd = mkstemp(un.sun_path);
        if (fd < 0) {
            error_setg_errno(errp, errno,
                             "Failed to make a temporary socket name in %s", tmpdir);
            goto err;
        }
        close(fd);
G
Gerd Hoffmann 已提交
729
        qemu_opt_set(opts, "path", un.sun_path);
730 731 732 733
    }

    unlink(un.sun_path);
    if (bind(sock, (struct sockaddr*) &un, sizeof(un)) < 0) {
734
        error_setg_errno(errp, errno, "Failed to bind socket to %s", un.sun_path);
735 736 737
        goto err;
    }
    if (listen(sock, 1) < 0) {
738
        error_setg_errno(errp, errno, "Failed to listen on socket");
739 740 741 742 743 744 745 746 747 748
        goto err;
    }

    return sock;

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

749 750
int unix_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
751 752
{
    struct sockaddr_un un;
G
Gerd Hoffmann 已提交
753
    const char *path = qemu_opt_get(opts, "path");
754 755
    ConnectState *connect_state = NULL;
    int sock, rc;
756

G
Gonglei 已提交
757
    if (path == NULL) {
758
        error_setg(errp, "unix connect: no path specified");
G
Gerd Hoffmann 已提交
759 760 761
        return -1;
    }

K
Kevin Wolf 已提交
762
    sock = qemu_socket(PF_UNIX, SOCK_STREAM, 0);
763
    if (sock < 0) {
764
        error_setg_errno(errp, errno, "Failed to create socket");
V
vibi 已提交
765
        return -1;
766
    }
767 768 769 770
    if (callback != NULL) {
        connect_state = g_malloc0(sizeof(*connect_state));
        connect_state->callback = callback;
        connect_state->opaque = opaque;
771
        qemu_set_nonblock(sock);
772
    }
773 774 775 776

    memset(&un, 0, sizeof(un));
    un.sun_family = AF_UNIX;
    snprintf(un.sun_path, sizeof(un.sun_path), "%s", path);
777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793

    /* 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) {
794
            callback(sock, NULL, opaque);
795 796 797 798
        }
    }

    if (rc < 0) {
799
        error_setg_errno(errp, -rc, "Failed to connect socket");
800
        close(sock);
801
        sock = -1;
802 803
    }

804
    g_free(connect_state);
805 806 807
    return sock;
}

808 809 810 811
#else

int unix_listen_opts(QemuOpts *opts, Error **errp)
{
812
    error_setg(errp, "unix sockets are not available on windows");
813 814 815 816
    errno = ENOTSUP;
    return -1;
}

817 818
int unix_connect_opts(QemuOpts *opts, Error **errp,
                      NonBlockingConnectHandler *callback, void *opaque)
819
{
820
    error_setg(errp, "unix sockets are not available on windows");
821 822 823 824 825
    errno = ENOTSUP;
    return -1;
}
#endif

G
Gerd Hoffmann 已提交
826
/* compatibility wrapper */
827
int unix_listen(const char *str, char *ostr, int olen, Error **errp)
G
Gerd Hoffmann 已提交
828 829 830 831 832
{
    QemuOpts *opts;
    char *path, *optstr;
    int sock, len;

833
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
834 835 836 837 838

    optstr = strchr(str, ',');
    if (optstr) {
        len = optstr - str;
        if (len) {
839
            path = g_malloc(len+1);
G
Gerd Hoffmann 已提交
840 841
            snprintf(path, len+1, "%.*s", len, str);
            qemu_opt_set(opts, "path", path);
842
            g_free(path);
G
Gerd Hoffmann 已提交
843 844 845 846 847
        }
    } else {
        qemu_opt_set(opts, "path", str);
    }

848
    sock = unix_listen_opts(opts, errp);
G
Gerd Hoffmann 已提交
849 850 851 852 853 854 855

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

856
int unix_connect(const char *path, Error **errp)
G
Gerd Hoffmann 已提交
857 858 859 860
{
    QemuOpts *opts;
    int sock;

861
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
862
    qemu_opt_set(opts, "path", path);
863 864 865 866 867 868 869 870 871 872 873 874 875 876 877
    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);

878
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
879 880
    qemu_opt_set(opts, "path", path);
    sock = unix_connect_opts(opts, errp, callback, opaque);
G
Gerd Hoffmann 已提交
881 882 883 884
    qemu_opts_del(opts);
    return sock;
}

885 886
SocketAddress *socket_parse(const char *str, Error **errp)
{
887
    SocketAddress *addr;
888

889
    addr = g_new0(SocketAddress, 1);
890 891
    if (strstart(str, "unix:", NULL)) {
        if (str[5] == '\0') {
892
            error_setg(errp, "invalid Unix socket address");
893 894 895 896 897 898 899 900
            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') {
901
            error_setg(errp, "invalid file descriptor address");
902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927
            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;

928
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
929 930 931 932 933 934 935 936 937 938 939 940 941
    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);
942
        if (fd >= 0 && callback) {
943
            qemu_set_nonblock(fd);
944
            callback(fd, NULL, opaque);
945 946 947 948 949 950 951 952 953 954 955 956 957 958 959
        }
        break;

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

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

960
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982
    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 已提交
983 984 985 986 987
int socket_dgram(SocketAddress *remote, SocketAddress *local, Error **errp)
{
    QemuOpts *opts;
    int fd;

988
    opts = qemu_opts_create(&socket_optslist, NULL, 0, &error_abort);
G
Gerd Hoffmann 已提交
989 990
    switch (remote->kind) {
    case SOCKET_ADDRESS_KIND_INET:
991
        inet_addr_to_opts(opts, remote->inet);
G
Gerd Hoffmann 已提交
992 993 994 995 996 997 998 999 1000
        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");
1001
        fd = -1;
G
Gerd Hoffmann 已提交
1002 1003 1004 1005
    }
    qemu_opts_del(opts);
    return fd;
}