migration-exec.c 2.42 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * QEMU live migration
 *
 * Copyright IBM, Corp. 2008
 * Copyright Dell MessageOne 2008
 *
 * Authors:
 *  Anthony Liguori   <aliguori@us.ibm.com>
 *  Charles Duffy     <charles_duffy@messageone.com>
 *
 * This work is licensed under the terms of the GNU GPL, version 2.  See
 * the COPYING file in the top-level directory.
 *
14 15
 * Contributions after 2012-01-13 are licensed under the terms of the
 * GNU GPL, version 2 or (at your option) any later version.
16 17 18
 */

#include "qemu-common.h"
19
#include "qemu/sockets.h"
20
#include "migration/migration.h"
21
#include "buffered_file.h"
22
#include "block/block.h"
23 24
#include <sys/types.h>
#include <sys/wait.h>
25 26 27 28

//#define DEBUG_MIGRATION_EXEC

#ifdef DEBUG_MIGRATION_EXEC
malc's avatar
malc committed
29
#define DPRINTF(fmt, ...) \
30 31
    do { printf("migration-exec: " fmt, ## __VA_ARGS__); } while (0)
#else
malc's avatar
malc committed
32
#define DPRINTF(fmt, ...) \
33 34 35
    do { } while (0)
#endif

36
static int file_errno(MigrationState *s)
37 38 39 40
{
    return errno;
}

41
static int file_write(MigrationState *s, const void * buf, size_t size)
42 43 44 45
{
    return write(s->fd, buf, size);
}

46
static int exec_close(MigrationState *s)
47
{
48
    int ret = 0;
malc's avatar
malc committed
49
    DPRINTF("exec_close\n");
50 51 52 53 54 55
    ret = qemu_fclose(s->opaque);
    s->opaque = NULL;
    s->fd = -1;
    if (ret >= 0 && !(WIFEXITED(ret) && WEXITSTATUS(ret) == 0)) {
        /* close succeeded, but non-zero exit code: */
        ret = -EIO; /* fake errno value */
56
    }
57
    return ret;
58 59
}

60
void exec_start_outgoing_migration(MigrationState *s, const char *command, Error **errp)
61 62 63 64 65
{
    FILE *f;

    f = popen(command, "w");
    if (f == NULL) {
66 67
        error_setg_errno(errp, errno, "failed to popen the migration target");
        return;
68 69 70
    }

    s->fd = fileno(f);
71
    assert(s->fd != -1);
72
    socket_set_nonblock(s->fd);
73 74 75

    s->opaque = qemu_popen(f, "w");

76
    s->close = exec_close;
77 78 79 80 81 82
    s->get_error = file_errno;
    s->write = file_write;

    migrate_fd_connect(s);
}

83
static void exec_accept_incoming_migration(void *opaque)
84
{
85
    QEMUFile *f = opaque;
86

87
    qemu_set_fd_handler2(qemu_get_fd(f), NULL, NULL, NULL, NULL);
88
    process_incoming_migration(f);
89 90
}

91
void exec_start_incoming_migration(const char *command, Error **errp)
92 93 94
{
    QEMUFile *f;

malc's avatar
malc committed
95
    DPRINTF("Attempting to start an incoming migration\n");
96 97
    f = qemu_popen_cmd(command, "r");
    if(f == NULL) {
98 99
        error_setg_errno(errp, errno, "failed to popen the migration source");
        return;
100 101
    }

102
    qemu_set_fd_handler2(qemu_get_fd(f), NULL,
Juan Quintela's avatar
Juan Quintela committed
103
			 exec_accept_incoming_migration, NULL, f);
104
}