qemu/migration-tcp.c
<<
>>
Prefs
   1/*
   2 * QEMU live migration
   3 *
   4 * Copyright IBM, Corp. 2008
   5 *
   6 * Authors:
   7 *  Anthony Liguori   <aliguori@us.ibm.com>
   8 *
   9 * This work is licensed under the terms of the GNU GPL, version 2.  See
  10 * the COPYING file in the top-level directory.
  11 *
  12 * Contributions after 2012-01-13 are licensed under the terms of the
  13 * GNU GPL, version 2 or (at your option) any later version.
  14 */
  15
  16#include <string.h>
  17
  18#include "qemu-common.h"
  19#include "qemu/error-report.h"
  20#include "qemu/sockets.h"
  21#include "migration/migration.h"
  22#include "migration/qemu-file.h"
  23#include "block/block.h"
  24#include "qemu/main-loop.h"
  25
  26//#define DEBUG_MIGRATION_TCP
  27
  28#ifdef DEBUG_MIGRATION_TCP
  29#define DPRINTF(fmt, ...) \
  30    do { printf("migration-tcp: " fmt, ## __VA_ARGS__); } while (0)
  31#else
  32#define DPRINTF(fmt, ...) \
  33    do { } while (0)
  34#endif
  35
  36static void tcp_wait_for_connect(int fd, Error *err, void *opaque)
  37{
  38    MigrationState *s = opaque;
  39
  40    if (fd < 0) {
  41        DPRINTF("migrate connect error: %s\n", error_get_pretty(err));
  42        s->file = NULL;
  43        migrate_fd_error(s);
  44    } else {
  45        DPRINTF("migrate connect success\n");
  46        s->file = qemu_fopen_socket(fd, "wb");
  47        migrate_fd_connect(s);
  48    }
  49}
  50
  51void tcp_start_outgoing_migration(MigrationState *s, const char *host_port, Error **errp)
  52{
  53    inet_nonblocking_connect(host_port, tcp_wait_for_connect, s, errp);
  54}
  55
  56static void tcp_accept_incoming_migration(void *opaque)
  57{
  58    struct sockaddr_in addr;
  59    socklen_t addrlen = sizeof(addr);
  60    int s = (intptr_t)opaque;
  61    QEMUFile *f;
  62    int c, err;
  63
  64    do {
  65        c = qemu_accept(s, (struct sockaddr *)&addr, &addrlen);
  66        err = socket_error();
  67    } while (c < 0 && err == EINTR);
  68    qemu_set_fd_handler2(s, NULL, NULL, NULL, NULL);
  69    closesocket(s);
  70
  71    DPRINTF("accepted migration\n");
  72
  73    if (c < 0) {
  74        error_report("could not accept migration connection (%s)",
  75                     strerror(err));
  76        return;
  77    }
  78
  79    f = qemu_fopen_socket(c, "rb");
  80    if (f == NULL) {
  81        error_report("could not qemu_fopen socket");
  82        goto out;
  83    }
  84
  85    process_incoming_migration(f);
  86    return;
  87
  88out:
  89    closesocket(c);
  90}
  91
  92void tcp_start_incoming_migration(const char *host_port, Error **errp)
  93{
  94    int s;
  95
  96    s = inet_listen(host_port, NULL, 256, SOCK_STREAM, 0, errp);
  97    if (s < 0) {
  98        return;
  99    }
 100
 101    qemu_set_fd_handler2(s, NULL, tcp_accept_incoming_migration, NULL,
 102                         (void *)(intptr_t)s);
 103}
 104