qemu_monitor.c 105.4 KB
Newer Older
1 2 3
/*
 * qemu_monitor.c: interaction with QEMU monitor console
 *
4
 * Copyright (C) 2006-2015 Red Hat, Inc.
5 6 7 8 9 10 11 12 13 14 15 16 17
 * Copyright (C) 2006 Daniel P. Berrange
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
18
 * License along with this library.  If not, see
O
Osier Yang 已提交
19
 * <http://www.gnu.org/licenses/>.
20 21 22 23 24 25 26
 *
 * Author: Daniel P. Berrange <berrange@redhat.com>
 */

#include <config.h>

#include <poll.h>
27
#include <sys/socket.h>
28 29 30 31 32
#include <sys/un.h>
#include <unistd.h>
#include <fcntl.h>

#include "qemu_monitor.h"
33
#include "qemu_monitor_text.h"
D
Daniel P. Berrange 已提交
34
#include "qemu_monitor_json.h"
35 36
#include "qemu_domain.h"
#include "qemu_process.h"
37
#include "virerror.h"
38
#include "viralloc.h"
39
#include "virlog.h"
E
Eric Blake 已提交
40
#include "virfile.h"
41
#include "virprocess.h"
42
#include "virobject.h"
43
#include "virprobe.h"
44
#include "virstring.h"
45
#include "virtime.h"
46

47 48 49 50
#ifdef WITH_DTRACE_PROBES
# include "libvirt_qemu_probes.h"
#endif

51 52
#define VIR_FROM_THIS VIR_FROM_QEMU

53 54
VIR_LOG_INIT("qemu.qemu_monitor");

55 56
#define DEBUG_IO 0
#define DEBUG_RAW_IO 0
57

58 59 60 61 62 63 64 65 66
/* We read from QEMU until seeing a \r\n pair to indicate a
 * completed reply or event. To avoid memory denial-of-service
 * though, we must have a size limit on amount of data we
 * buffer. 10 MB is large enough that it ought to cope with
 * normal QEMU replies, and small enough that we're not
 * consuming unreasonable mem.
 */
#define QEMU_MONITOR_MAX_RESPONSE (10 * 1024 * 1024)

67
struct _qemuMonitor {
68
    virObjectLockable parent;
69

70 71
    virCond notify;

72
    int fd;
73 74 75 76 77 78

    /* Represents the watch number to be used for updating and
     * unregistering the monitor @fd for events in the event loop:
     * > 0: valid watch number
     * = 0: not registered
     * < 0: an error occurred during the registration of @fd */
79 80 81 82 83
    int watch;
    int hasSendFD;

    virDomainObjPtr vm;

84
    qemuMonitorCallbacksPtr cb;
85
    void *callbackOpaque;
86 87 88 89 90 91 92 93 94 95 96 97 98

    /* If there's a command being processed this will be
     * non-NULL */
    qemuMonitorMessagePtr msg;

    /* Buffer incoming data ready for Text/QMP monitor
     * code to process & find message boundaries */
    size_t bufferOffset;
    size_t bufferLength;
    char *buffer;

    /* If anything went wrong, this will be fed back
     * the next monitor msg */
99 100 101
    virError lastError;

    int nextSerial;
102

E
Eric Blake 已提交
103 104
    bool json;
    bool waitGreeting;
105 106 107

    /* cache of query-command-line-options results */
    virJSONValuePtr options;
108 109 110 111

    /* If found, path to the virtio memballoon driver */
    char *balloonpath;
    bool ballooninit;
112

113 114 115 116
    /* Log file context of the qemu process to dig for usable info */
    qemuMonitorReportDomainLogError logFunc;
    void *logOpaque;
    virFreeCallback logDestroy;
117 118
};

119 120 121 122 123 124 125 126 127
/**
 * QEMU_CHECK_MONITOR_FULL:
 * @mon: monitor pointer variable to check, evaluated multiple times, no parentheses
 * @exit: statement that is used to exit the function
 *
 * This macro checks that the monitor is valid for given operation and exits
 * the function if not. The macro also adds a debug statement regarding the
 * monitor.
 */
128
#define QEMU_CHECK_MONITOR_FULL(mon, exit) \
129 130 131 132 133 134 135 136
    do { \
        if (!mon) { \
            virReportError(VIR_ERR_INVALID_ARG, "%s", \
                           _("monitor must not be NULL")); \
            exit; \
        } \
        VIR_DEBUG("mon:%p vm:%p json:%d fd:%d", \
                  mon, mon->vm, mon->json, mon->fd); \
137
        if (!mon->json) { \
138 139 140 141
            virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s", \
                           _("JSON monitor is required")); \
            exit; \
        } \
142
    } while (0)
143 144 145

/* Check monitor and return NULL on error */
#define QEMU_CHECK_MONITOR_NULL(mon) \
146
    QEMU_CHECK_MONITOR_FULL(mon, return NULL)
147 148 149

/* Check monitor and return -1 on error */
#define QEMU_CHECK_MONITOR(mon) \
150
    QEMU_CHECK_MONITOR_FULL(mon, return -1)
151 152 153

/* Check monitor and jump to the provided label */
#define QEMU_CHECK_MONITOR_GOTO(mon, label) \
154
    QEMU_CHECK_MONITOR_FULL(mon, goto label)
155

156 157 158 159 160
static virClassPtr qemuMonitorClass;
static void qemuMonitorDispose(void *obj);

static int qemuMonitorOnceInit(void)
{
161
    if (!VIR_CLASS_NEW(qemuMonitor, virClassForObjectLockable()))
162 163 164 165 166 167 168
        return -1;

    return 0;
}

VIR_ONCE_GLOBAL_INIT(qemuMonitor)

169

170 171
VIR_ENUM_IMPL(qemuMonitorMigrationStatus,
              QEMU_MONITOR_MIGRATION_STATUS_LAST,
172
              "inactive", "setup",
173 174
              "active", "pre-switchover",
              "device", "postcopy-active",
175 176
              "completed", "failed",
              "cancelling", "cancelled")
177

178 179 180 181
VIR_ENUM_IMPL(qemuMonitorVMStatus,
              QEMU_MONITOR_VM_STATUS_LAST,
              "debug", "inmigrate", "internal-error", "io-error", "paused",
              "postmigrate", "prelaunch", "finish-migrate", "restore-vm",
182
              "running", "save-vm", "shutdown", "watchdog", "guest-panicked")
183

184 185 186 187 188 189 190 191 192 193 194 195 196 197
typedef enum {
    QEMU_MONITOR_BLOCK_IO_STATUS_OK,
    QEMU_MONITOR_BLOCK_IO_STATUS_FAILED,
    QEMU_MONITOR_BLOCK_IO_STATUS_NOSPACE,

    QEMU_MONITOR_BLOCK_IO_STATUS_LAST
} qemuMonitorBlockIOStatus;

VIR_ENUM_DECL(qemuMonitorBlockIOStatus)

VIR_ENUM_IMPL(qemuMonitorBlockIOStatus,
              QEMU_MONITOR_BLOCK_IO_STATUS_LAST,
              "ok", "failed", "nospace")

198 199 200 201
VIR_ENUM_IMPL(qemuMonitorDumpStatus,
              QEMU_MONITOR_DUMP_STATUS_LAST,
              "none", "active", "completed", "failed")

202 203
char *
qemuMonitorEscapeArg(const char *in)
204 205
{
    int len = 0;
206
    size_t i, j;
207 208 209 210 211 212 213
    char *out;

    /* To pass through the QEMU monitor, we need to use escape
       sequences: \r, \n, \", \\
    */

    for (i = 0; in[i] != '\0'; i++) {
214
        switch (in[i]) {
215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230
        case '\r':
        case '\n':
        case '"':
        case '\\':
            len += 2;
            break;
        default:
            len += 1;
            break;
        }
    }

    if (VIR_ALLOC_N(out, len + 1) < 0)
        return NULL;

    for (i = j = 0; in[i] != '\0'; i++) {
231
        switch (in[i]) {
232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254
        case '\r':
            out[j++] = '\\';
            out[j++] = 'r';
            break;
        case '\n':
            out[j++] = '\\';
            out[j++] = 'n';
            break;
        case '"':
        case '\\':
            out[j++] = '\\';
            out[j++] = in[i];
            break;
        default:
            out[j++] = in[i];
            break;
        }
    }
    out[j] = '\0';

    return out;
}

255 256 257

char *
qemuMonitorUnescapeArg(const char *in)
258
{
259
    size_t i, j;
260
    char *out;
261
    int len = strlen(in);
262 263
    char next;

264
    if (VIR_ALLOC_N(out, len + 1) < 0)
265 266 267 268 269 270
        return NULL;

    for (i = j = 0; i < len; ++i) {
        next = in[i];
        if (in[i] == '\\') {
            ++i;
271
            switch (in[i]) {
272 273 274 275 276 277 278 279 280 281 282
            case 'r':
                next = '\r';
                break;
            case 'n':
                next = '\n';
                break;
            case '"':
            case '\\':
                next = in[i];
                break;
            default:
283
                /* invalid input (including trailing '\' at end of in) */
284 285 286 287 288 289 290 291 292 293 294
                VIR_FREE(out);
                return NULL;
            }
        }
        out[j++] = next;
    }
    out[j] = '\0';

    return out;
}

295

J
Jiri Denemark 已提交
296
#if DEBUG_RAW_IO
297
# include <c-ctype.h>
298 299
static char *
qemuMonitorEscapeNonPrintable(const char *text)
300
{
301
    size_t i;
302
    virBuffer buf = VIR_BUFFER_INITIALIZER;
303
    for (i = 0; text[i] != '\0'; i++) {
304 305
        if (c_isprint(text[i]) ||
            text[i] == '\n' ||
E
Eric Blake 已提交
306 307
            (text[i] == '\r' && text[i + 1] == '\n'))
            virBufferAddChar(&buf, text[i]);
308
        else
309
            virBufferAsprintf(&buf, "0x%02x", text[i]);
310 311 312 313 314
    }
    return virBufferContentAndReset(&buf);
}
#endif

315 316 317

static void
qemuMonitorDispose(void *obj)
318
{
319 320
    qemuMonitorPtr mon = obj;

321
    VIR_DEBUG("mon=%p", mon);
322
    if (mon->cb && mon->cb->destroy)
323
        (mon->cb->destroy)(mon, mon->vm, mon->callbackOpaque);
324 325
    virObjectUnref(mon->vm);

326
    virResetError(&mon->lastError);
327
    virCondDestroy(&mon->notify);
E
Eric Blake 已提交
328
    VIR_FREE(mon->buffer);
329
    virJSONValueFree(mon->options);
330
    VIR_FREE(mon->balloonpath);
331 332 333 334
}


static int
335 336 337
qemuMonitorOpenUnix(const char *monitor,
                    pid_t cpid,
                    unsigned long long timeout)
338 339 340
{
    struct sockaddr_un addr;
    int monfd;
341
    virTimeBackOffVar timebackoff;
342
    int ret = -1;
343 344

    if ((monfd = socket(AF_UNIX, SOCK_STREAM, 0)) < 0) {
345
        virReportSystemError(errno,
346 347 348 349 350 351 352
                             "%s", _("failed to create socket"));
        return -1;
    }

    memset(&addr, 0, sizeof(addr));
    addr.sun_family = AF_UNIX;
    if (virStrcpyStatic(addr.sun_path, monitor) == NULL) {
353 354
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Monitor path %s too big for destination"), monitor);
355 356 357
        goto error;
    }

358
    if (virTimeBackOffStart(&timebackoff, 1, timeout * 1000) < 0)
359
        goto error;
360
    while (virTimeBackOffWait(&timebackoff)) {
361
        ret = connect(monfd, (struct sockaddr *)&addr, sizeof(addr));
362 363 364 365

        if (ret == 0)
            break;

366
        if ((errno == ENOENT || errno == ECONNREFUSED) &&
367
            (!cpid || virProcessKill(cpid, 0) == 0)) {
368 369 370 371 372
            /* ENOENT       : Socket may not have shown up yet
             * ECONNREFUSED : Leftover socket hasn't been removed yet */
            continue;
        }

373
        virReportSystemError(errno, "%s",
374 375 376
                             _("failed to connect to monitor socket"));
        goto error;

377
    }
378 379

    if (ret != 0) {
380
        virReportSystemError(errno, "%s",
381
                             _("monitor socket did not show up"));
382 383 384
        goto error;
    }

385
    return monfd;
386

387
 error:
388
    VIR_FORCE_CLOSE(monfd);
389 390 391
    return -1;
}

392

393
static int
394
qemuMonitorOpenPty(const char *monitor)
395 396 397 398
{
    int monfd;

    if ((monfd = open(monitor, O_RDWR)) < 0) {
399 400
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Unable to open monitor path %s"), monitor);
401 402 403
        return -1;
    }

404
    return monfd;
405
}
406

407

408 409 410 411
/* This method processes data that has been received
 * from the monitor. Looking for async events and
 * replies/errors.
 */
412 413 414 415 416 417 418 419 420 421 422
static int
qemuMonitorIOProcess(qemuMonitorPtr mon)
{
    int len;
    qemuMonitorMessagePtr msg = NULL;

    /* See if there's a message & whether its ready for its reply
     * ie whether its completed writing all its data */
    if (mon->msg && mon->msg->txOffset == mon->msg->txLength)
        msg = mon->msg;

423
#if DEBUG_IO
424
# if DEBUG_RAW_IO
425 426
    char *str1 = qemuMonitorEscapeNonPrintable(msg ? msg->txBuffer : "");
    char *str2 = qemuMonitorEscapeNonPrintable(mon->buffer);
427
    VIR_ERROR(_("Process %d %p %p [[[[%s]]][[[%s]]]"), (int)mon->bufferOffset, mon->msg, msg, str1, str2);
428 429
    VIR_FREE(str1);
    VIR_FREE(str2);
430
# else
431
    VIR_DEBUG("Process %d", (int)mon->bufferOffset);
432
# endif
433 434
#endif

435 436
    PROBE_QUIET(QEMU_MONITOR_IO_PROCESS, "mon=%p buf=%s len=%zu",
                mon, mon->buffer, mon->bufferOffset);
437

438 439 440
    len = qemuMonitorJSONIOProcess(mon,
                                   mon->buffer, mon->bufferOffset,
                                   msg);
441
    if (len < 0)
442 443
        return -1;

E
Eric Blake 已提交
444 445
    if (len && mon->waitGreeting)
        mon->waitGreeting = false;
446

447 448 449 450 451 452 453
    if (len < mon->bufferOffset) {
        memmove(mon->buffer, mon->buffer + len, mon->bufferOffset - len);
        mon->bufferOffset -= len;
    } else {
        VIR_FREE(mon->buffer);
        mon->bufferOffset = mon->bufferLength = 0;
    }
454
#if DEBUG_IO
455
    VIR_DEBUG("Process done %d used %d", (int)mon->bufferOffset, len);
456
#endif
457 458 459 460 461 462
    if (msg && msg->finished)
        virCondBroadcast(&mon->notify);
    return len;
}


S
Stefan Berger 已提交
463
/* Call this function while holding the monitor lock. */
464 465 466 467 468 469 470 471 472 473 474 475 476
static int
qemuMonitorIOWriteWithFD(qemuMonitorPtr mon,
                         const char *data,
                         size_t len,
                         int fd)
{
    struct msghdr msg;
    struct iovec iov[1];
    int ret;
    char control[CMSG_SPACE(sizeof(int))];
    struct cmsghdr *cmsg;

    memset(&msg, 0, sizeof(msg));
477
    memset(control, 0, sizeof(control));
478 479 480 481 482 483 484 485 486 487 488

    iov[0].iov_base = (void *)data;
    iov[0].iov_len = len;

    msg.msg_iov = iov;
    msg.msg_iovlen = 1;

    msg.msg_control = control;
    msg.msg_controllen = sizeof(control);

    cmsg = CMSG_FIRSTHDR(&msg);
489 490 491
    /* Some static analyzers, like clang 2.6-0.6.pre2, fail to see
       that our use of CMSG_FIRSTHDR will not return NULL.  */
    sa_assert(cmsg);
492 493 494 495 496 497 498 499 500 501 502 503
    cmsg->cmsg_len = CMSG_LEN(sizeof(int));
    cmsg->cmsg_level = SOL_SOCKET;
    cmsg->cmsg_type = SCM_RIGHTS;
    memcpy(CMSG_DATA(cmsg), &fd, sizeof(int));

    do {
        ret = sendmsg(mon->fd, &msg, 0);
    } while (ret < 0 && errno == EINTR);

    return ret;
}

504

S
Stefan Berger 已提交
505 506 507 508
/*
 * Called when the monitor is able to write data
 * Call this function while holding the monitor lock.
 */
509 510 511 512
static int
qemuMonitorIOWrite(qemuMonitorPtr mon)
{
    int done;
J
John Ferlan 已提交
513 514
    char *buf;
    size_t len;
515 516 517 518 519

    /* If no active message, or fully transmitted, the no-op */
    if (!mon->msg || mon->msg->txOffset == mon->msg->txLength)
        return 0;

520
    if (mon->msg->txFD != -1 && !mon->hasSendFD) {
521 522
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Monitor does not support sending of file descriptors"));
523 524 525
        return -1;
    }

J
John Ferlan 已提交
526 527
    buf = mon->msg->txBuffer + mon->msg->txOffset;
    len = mon->msg->txLength - mon->msg->txOffset;
528
    if (mon->msg->txFD == -1)
J
John Ferlan 已提交
529
        done = write(mon->fd, buf, len);
530
    else
J
John Ferlan 已提交
531
        done = qemuMonitorIOWriteWithFD(mon, buf, len, mon->msg->txFD);
532

533
    PROBE(QEMU_MONITOR_IO_WRITE,
534
          "mon=%p buf=%s len=%zu ret=%d errno=%d",
535
          mon, buf, len, done, done < 0 ? errno : 0);
536

537
    if (mon->msg->txFD != -1) {
538 539
        PROBE(QEMU_MONITOR_IO_SEND_FD,
              "mon=%p fd=%d ret=%d errno=%d",
540
              mon, mon->msg->txFD, done, done < 0 ? errno : 0);
541
    }
542

543 544 545 546
    if (done < 0) {
        if (errno == EAGAIN)
            return 0;

547 548
        virReportSystemError(errno, "%s",
                             _("Unable to write to monitor"));
549 550 551 552 553 554
        return -1;
    }
    mon->msg->txOffset += done;
    return done;
}

555

556 557
/*
 * Called when the monitor has incoming data to read
S
Stefan Berger 已提交
558
 * Call this function while holding the monitor lock.
559 560 561 562 563 564 565 566 567 568
 *
 * Returns -1 on error, or number of bytes read
 */
static int
qemuMonitorIORead(qemuMonitorPtr mon)
{
    size_t avail = mon->bufferLength - mon->bufferOffset;
    int ret = 0;

    if (avail < 1024) {
569 570 571 572 573 574
        if (mon->bufferLength >= QEMU_MONITOR_MAX_RESPONSE) {
            virReportSystemError(ERANGE,
                                 _("No complete monitor response found in %d bytes"),
                                 QEMU_MONITOR_MAX_RESPONSE);
            return -1;
        }
575
        if (VIR_REALLOC_N(mon->buffer,
576
                          mon->bufferLength + 1024) < 0)
577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
            return -1;
        mon->bufferLength += 1024;
        avail += 1024;
    }

    /* Read as much as we can get into our buffer,
       until we block on EAGAIN, or hit EOF */
    while (avail > 1) {
        int got;
        got = read(mon->fd,
                   mon->buffer + mon->bufferOffset,
                   avail - 1);
        if (got < 0) {
            if (errno == EAGAIN)
                break;
592 593
            virReportSystemError(errno, "%s",
                                 _("Unable to read from monitor"));
594 595 596 597 598 599 600 601 602 603 604 605
            ret = -1;
            break;
        }
        if (got == 0)
            break;

        ret += got;
        avail -= got;
        mon->bufferOffset += got;
        mon->buffer[mon->bufferOffset] = '\0';
    }

606
#if DEBUG_IO
607
    VIR_DEBUG("Now read %d bytes of data", (int)mon->bufferOffset);
608
#endif
609 610 611 612 613

    return ret;
}


614 615
static void
qemuMonitorUpdateWatch(qemuMonitorPtr mon)
616 617 618 619 620
{
    int events =
        VIR_EVENT_HANDLE_HANGUP |
        VIR_EVENT_HANDLE_ERROR;

621 622 623
    if (!mon->watch)
        return;

624
    if (mon->lastError.code == VIR_ERR_OK) {
625 626
        events |= VIR_EVENT_HANDLE_READABLE;

627
        if ((mon->msg && mon->msg->txOffset < mon->msg->txLength) &&
E
Eric Blake 已提交
628
            !mon->waitGreeting)
629 630 631 632
            events |= VIR_EVENT_HANDLE_WRITABLE;
    }

    virEventUpdateHandle(mon->watch, events);
633 634
}

635 636

static void
637 638
qemuMonitorIO(int watch, int fd, int events, void *opaque)
{
639
    qemuMonitorPtr mon = opaque;
640 641
    bool error = false;
    bool eof = false;
642
    bool hangup = false;
643

644 645
    virObjectRef(mon);

S
Stefan Berger 已提交
646
    /* lock access to the monitor and protect fd */
647
    virObjectLock(mon);
648
#if DEBUG_IO
649
    VIR_DEBUG("Monitor %p I/O on watch %d fd %d events %d", mon, watch, fd, events);
650
#endif
651
    if (mon->fd == -1 || mon->watch == 0) {
652
        virObjectUnlock(mon);
653 654 655
        virObjectUnref(mon);
        return;
    }
656

657
    if (mon->fd != fd || mon->watch != watch) {
658
        if (events & (VIR_EVENT_HANDLE_HANGUP | VIR_EVENT_HANDLE_ERROR))
659
            eof = true;
660 661 662
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("event from unexpected fd %d!=%d / watch %d!=%d"),
                       mon->fd, fd, mon->watch, watch);
663 664
        error = true;
    } else if (mon->lastError.code != VIR_ERR_OK) {
665
        if (events & (VIR_EVENT_HANDLE_HANGUP | VIR_EVENT_HANDLE_ERROR))
666
            eof = true;
667
        error = true;
668
    } else {
669
        if (events & VIR_EVENT_HANDLE_WRITABLE) {
670
            if (qemuMonitorIOWrite(mon) < 0) {
671
                error = true;
672 673 674
                if (errno == ECONNRESET)
                    hangup = true;
            }
675 676
            events &= ~VIR_EVENT_HANDLE_WRITABLE;
        }
677 678

        if (!error &&
679 680
            events & VIR_EVENT_HANDLE_READABLE) {
            int got = qemuMonitorIORead(mon);
681 682
            events &= ~VIR_EVENT_HANDLE_READABLE;
            if (got < 0) {
683
                error = true;
684 685
                if (errno == ECONNRESET)
                    hangup = true;
686 687 688 689 690
            } else if (got == 0) {
                eof = true;
            } else {
                /* Ignore hangup/error events if we read some data, to
                 * give time for that data to be consumed */
691 692 693
                events = 0;

                if (qemuMonitorIOProcess(mon) < 0)
694
                    error = true;
695
            }
696 697
        }

698 699 700
        if (events & VIR_EVENT_HANDLE_HANGUP) {
            hangup = true;
            if (!error) {
701 702
                virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                               _("End of file from qemu monitor"));
703 704 705
                eof = true;
                events &= ~VIR_EVENT_HANDLE_HANGUP;
            }
706 707
        }

708 709
        if (!error && !eof &&
            events & VIR_EVENT_HANDLE_ERROR) {
710 711
            virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                           _("Invalid file descriptor while waiting for monitor"));
712
            eof = true;
713
            events &= ~VIR_EVENT_HANDLE_ERROR;
714 715
        }
        if (!error && events) {
716 717 718
            virReportError(VIR_ERR_INTERNAL_ERROR,
                           _("Unhandled event %d for monitor fd %d"),
                           events, mon->fd);
719
            error = true;
720 721 722
        }
    }

723
    if (error || eof) {
724
        if (hangup && mon->logFunc != NULL) {
725 726
            /* Check if an error message from qemu is available and if so, use
             * it to overwrite the actual message. It's done only in early
727 728 729
             * startup phases or during incoming migration when the message
             * from qemu is certainly more interesting than a
             * "connection reset by peer" message.
730
             */
731
            mon->logFunc(mon,
732
                         _("qemu unexpectedly closed the monitor"),
733
                         mon->logOpaque);
734 735
            virCopyLastError(&mon->lastError);
            virResetLastError();
736 737
        }

738 739 740
        if (mon->lastError.code != VIR_ERR_OK) {
            /* Already have an error, so clear any new error */
            virResetLastError();
741
        } else {
742 743
            virErrorPtr err = virGetLastError();
            if (!err)
744 745
                virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                               _("Error while processing monitor IO"));
746 747 748 749 750 751 752 753 754 755 756 757
            virCopyLastError(&mon->lastError);
            virResetLastError();
        }

        VIR_DEBUG("Error on monitor %s", NULLSTR(mon->lastError.message));
        /* If IO process resulted in an error & we have a message,
         * then wakeup that waiter */
        if (mon->msg && !mon->msg->finished) {
            mon->msg->finished = 1;
            virCondSignal(&mon->notify);
        }
    }
758 759 760

    qemuMonitorUpdateWatch(mon);

761 762 763
    /* We have to unlock to avoid deadlock against command thread,
     * but is this safe ?  I think it is, because the callback
     * will try to acquire the virDomainObjPtr mutex next */
764
    if (eof) {
J
Jiri Denemark 已提交
765
        qemuMonitorEofNotifyCallback eofNotify = mon->cb->eofNotify;
766
        virDomainObjPtr vm = mon->vm;
767

768 769
        /* Make sure anyone waiting wakes up now */
        virCondSignal(&mon->notify);
770
        virObjectUnlock(mon);
771
        VIR_DEBUG("Triggering EOF callback");
772
        (eofNotify)(mon, vm, mon->callbackOpaque);
773
        virObjectUnref(mon);
774
    } else if (error) {
J
Jiri Denemark 已提交
775
        qemuMonitorErrorNotifyCallback errorNotify = mon->cb->errorNotify;
776
        virDomainObjPtr vm = mon->vm;
777

778 779
        /* Make sure anyone waiting wakes up now */
        virCondSignal(&mon->notify);
780
        virObjectUnlock(mon);
781
        VIR_DEBUG("Triggering error callback");
782
        (errorNotify)(mon, vm, mon->callbackOpaque);
783
        virObjectUnref(mon);
784
    } else {
785
        virObjectUnlock(mon);
786
        virObjectUnref(mon);
787
    }
788 789 790
}


791 792 793 794
static qemuMonitorPtr
qemuMonitorOpenInternal(virDomainObjPtr vm,
                        int fd,
                        bool hasSendFD,
E
Eric Blake 已提交
795
                        bool json,
796 797
                        qemuMonitorCallbacksPtr cb,
                        void *opaque)
798
{
799 800
    qemuMonitorPtr mon;

801
    if (!cb->eofNotify) {
802 803
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("EOF notify callback must be supplied"));
804 805
        return NULL;
    }
806 807 808 809 810
    if (!cb->errorNotify) {
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Error notify callback must be supplied"));
        return NULL;
    }
811

812 813 814
    if (qemuMonitorInitialize() < 0)
        return NULL;

815
    if (!(mon = virObjectLockableNew(qemuMonitorClass)))
816 817
        return NULL;

818
    if (virCondInit(&mon->notify) < 0) {
819 820
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("cannot initialize monitor condition"));
821
        goto cleanup;
822
    }
823 824
    mon->fd = fd;
    mon->hasSendFD = hasSendFD;
825
    mon->vm = virObjectRef(vm);
D
Daniel P. Berrange 已提交
826
    mon->json = json;
827
    if (json)
E
Eric Blake 已提交
828
        mon->waitGreeting = true;
829
    mon->cb = cb;
830
    mon->callbackOpaque = opaque;
831

832
    if (virSetCloseExec(mon->fd) < 0) {
833 834
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("Unable to set monitor close-on-exec flag"));
835 836 837
        goto cleanup;
    }
    if (virSetNonBlock(mon->fd) < 0) {
838 839
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("Unable to put monitor into non-blocking mode"));
840 841 842 843
        goto cleanup;
    }


E
Eric Blake 已提交
844
    virObjectLock(mon);
845
    if (!qemuMonitorRegister(mon)) {
E
Eric Blake 已提交
846
        virObjectUnlock(mon);
847 848
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("unable to register monitor events"));
849 850 851
        goto cleanup;
    }

852 853
    PROBE(QEMU_MONITOR_NEW,
          "mon=%p refs=%d fd=%d",
854
          mon, mon->parent.parent.u.s.refs, mon->fd);
855
    virObjectUnlock(mon);
856

857 858
    return mon;

859
 cleanup:
860 861 862 863 864 865
    /* We don't want the 'destroy' callback invoked during
     * cleanup from construction failure, because that can
     * give a double-unref on virDomainObjPtr in the caller,
     * so kill the callbacks now.
     */
    mon->cb = NULL;
866 867
    /* The caller owns 'fd' on failure */
    mon->fd = -1;
868 869 870 871
    qemuMonitorClose(mon);
    return NULL;
}

872

873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891
#define QEMU_DEFAULT_MONITOR_WAIT 30

/**
 * qemuMonitorOpen:
 * @vm: domain object
 * @config: monitor configuration
 * @json: enable JSON on the monitor
 * @timeout: number of seconds to add to default timeout
 * @cb: monitor event handles
 * @opaque: opaque data for @cb
 *
 * Opens the monitor for running qemu. It may happen that it
 * takes some time for qemu to create the monitor socket (e.g.
 * because kernel is zeroing configured hugepages), therefore we
 * wait up to default + timeout seconds for the monitor to show
 * up after which a failure is claimed.
 *
 * Returns monitor object, NULL on error.
 */
892 893 894
qemuMonitorPtr
qemuMonitorOpen(virDomainObjPtr vm,
                virDomainChrSourceDefPtr config,
E
Eric Blake 已提交
895
                bool json,
896
                unsigned long long timeout,
897 898
                qemuMonitorCallbacksPtr cb,
                void *opaque)
899 900 901 902 903
{
    int fd;
    bool hasSendFD = false;
    qemuMonitorPtr ret;

904 905
    timeout += QEMU_DEFAULT_MONITOR_WAIT;

906 907 908
    switch (config->type) {
    case VIR_DOMAIN_CHR_TYPE_UNIX:
        hasSendFD = true;
909
        if ((fd = qemuMonitorOpenUnix(config->data.nix.path,
910
                                      vm->pid, timeout)) < 0)
911 912 913 914 915 916 917 918 919 920 921 922 923 924 925
            return NULL;
        break;

    case VIR_DOMAIN_CHR_TYPE_PTY:
        if ((fd = qemuMonitorOpenPty(config->data.file.path)) < 0)
            return NULL;
        break;

    default:
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("unable to handle monitor type: %s"),
                       virDomainChrTypeToString(config->type));
        return NULL;
    }

926
    ret = qemuMonitorOpenInternal(vm, fd, hasSendFD, json, cb, opaque);
927 928 929 930 931 932
    if (!ret)
        VIR_FORCE_CLOSE(fd);
    return ret;
}


933 934 935 936 937 938
qemuMonitorPtr
qemuMonitorOpenFD(virDomainObjPtr vm,
                  int sockfd,
                  bool json,
                  qemuMonitorCallbacksPtr cb,
                  void *opaque)
939
{
940
    return qemuMonitorOpenInternal(vm, sockfd, true, json, cb, opaque);
941 942
}

943

944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971
/**
 * qemuMonitorRegister:
 * @mon: QEMU monitor
 *
 * Registers the monitor in the event loop. The caller has to hold the
 * lock for @mon.
 *
 * Returns true in case of success, false otherwise
 */
bool
qemuMonitorRegister(qemuMonitorPtr mon)
{
    virObjectRef(mon);
    if ((mon->watch = virEventAddHandle(mon->fd,
                                        VIR_EVENT_HANDLE_HANGUP |
                                        VIR_EVENT_HANDLE_ERROR |
                                        VIR_EVENT_HANDLE_READABLE,
                                        qemuMonitorIO,
                                        mon,
                                        virObjectFreeCallback)) < 0) {
        virObjectUnref(mon);
        return false;
    }

    return true;
}


972 973 974 975 976 977 978 979 980
void
qemuMonitorUnregister(qemuMonitorPtr mon)
{
    if (mon->watch) {
        virEventRemoveHandle(mon->watch);
        mon->watch = 0;
    }
}

981 982
void
qemuMonitorClose(qemuMonitorPtr mon)
983 984
{
    if (!mon)
985
        return;
986

987
    virObjectLock(mon);
988
    PROBE(QEMU_MONITOR_CLOSE,
989
          "mon=%p refs=%d", mon, mon->parent.parent.u.s.refs);
S
Stefan Berger 已提交
990

991
    qemuMonitorSetDomainLogLocked(mon, NULL, NULL, NULL);
992

S
Stefan Berger 已提交
993
    if (mon->fd >= 0) {
994
        qemuMonitorUnregister(mon);
S
Stefan Berger 已提交
995
        VIR_FORCE_CLOSE(mon->fd);
996
    }
997

998 999 1000 1001 1002 1003 1004
    /* In case another thread is waiting for its monitor command to be
     * processed, we need to wake it up with appropriate error set.
     */
    if (mon->msg) {
        if (mon->lastError.code == VIR_ERR_OK) {
            virErrorPtr err = virSaveLastError();

1005
            virReportError(VIR_ERR_OPERATION_FAILED, "%s",
P
Philipp Hahn 已提交
1006
                           _("QEMU monitor was closed"));
1007 1008 1009 1010 1011 1012 1013 1014 1015 1016 1017 1018
            virCopyLastError(&mon->lastError);
            if (err) {
                virSetError(err);
                virFreeError(err);
            } else {
                virResetLastError();
            }
        }
        mon->msg->finished = 1;
        virCondSignal(&mon->notify);
    }

1019 1020 1021 1022 1023 1024
    /* Propagate existing monitor error in case the current thread has no
     * error set.
     */
    if (mon->lastError.code != VIR_ERR_OK && !virGetLastError())
        virSetError(&mon->lastError);

1025
    virObjectUnlock(mon);
1026
    virObjectUnref(mon);
1027 1028 1029
}


1030 1031
char *
qemuMonitorNextCommandID(qemuMonitorPtr mon)
1032 1033 1034
{
    char *id;

1035
    ignore_value(virAsprintf(&id, "libvirt-%d", ++mon->nextSerial));
1036 1037 1038 1039
    return id;
}


1040 1041 1042
int
qemuMonitorSend(qemuMonitorPtr mon,
                qemuMonitorMessagePtr msg)
1043
{
1044
    int ret = -1;
1045

E
Eric Blake 已提交
1046
    /* Check whether qemu quit unexpectedly */
1047 1048 1049 1050
    if (mon->lastError.code != VIR_ERR_OK) {
        VIR_DEBUG("Attempt to send command while error is set %s",
                  NULLSTR(mon->lastError.message));
        virSetError(&mon->lastError);
1051 1052 1053
        return -1;
    }

1054 1055
    mon->msg = msg;
    qemuMonitorUpdateWatch(mon);
1056

1057 1058 1059 1060
    PROBE(QEMU_MONITOR_SEND_MSG,
          "mon=%p msg=%s fd=%d",
          mon, mon->msg->txBuffer, mon->msg->txFD);

1061
    while (!mon->msg->finished) {
1062
        if (virCondWait(&mon->notify, &mon->parent.lock) < 0) {
1063 1064
            virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                           _("Unable to wait on monitor condition"));
1065
            goto cleanup;
1066 1067 1068 1069 1070 1071 1072 1073
        }
    }

    if (mon->lastError.code != VIR_ERR_OK) {
        VIR_DEBUG("Send command resulted in error %s",
                  NULLSTR(mon->lastError.message));
        virSetError(&mon->lastError);
        goto cleanup;
1074
    }
1075

1076
    ret = 0;
1077

1078
 cleanup:
1079 1080
    mon->msg = NULL;
    qemuMonitorUpdateWatch(mon);
1081

1082
    return ret;
1083
}
1084 1085


1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099
/**
 * This function returns a new virError object; the caller is responsible
 * for freeing it.
 */
virErrorPtr
qemuMonitorLastError(qemuMonitorPtr mon)
{
    if (mon->lastError.code == VIR_ERR_OK)
        return NULL;

    return virErrorCopyNew(&mon->lastError);
}


1100 1101 1102 1103 1104 1105
virJSONValuePtr
qemuMonitorGetOptions(qemuMonitorPtr mon)
{
    return mon->options;
}

1106

1107 1108 1109 1110 1111 1112
void
qemuMonitorSetOptions(qemuMonitorPtr mon, virJSONValuePtr options)
{
    mon->options = options;
}

1113 1114

/**
1115 1116 1117
 * Search the qom objects for the balloon driver object by its known names
 * of "virtio-balloon-pci" or "virtio-balloon-ccw". The entry for the driver
 * will be found by using function "qemuMonitorJSONFindLinkPath".
1118 1119 1120 1121 1122
 *
 * Once found, check the entry to ensure it has the correct property listed.
 * If it does not, then obtaining statistics from QEMU will not be possible.
 * This feature was added to QEMU 1.5.
 */
1123
static void
1124 1125
qemuMonitorInitBalloonObjectPath(qemuMonitorPtr mon,
                                 virDomainMemballoonDefPtr balloon)
1126 1127 1128
{
    ssize_t i, nprops = 0;
    char *path = NULL;
1129
    const char *name;
1130 1131 1132
    qemuMonitorJSONListPathPtr *bprops = NULL;

    if (mon->balloonpath) {
1133
        return;
1134 1135 1136
    } else if (mon->ballooninit) {
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Cannot determine balloon device path"));
1137
        return;
1138
    }
1139
    mon->ballooninit = true;
1140

1141 1142 1143 1144 1145 1146 1147 1148
    switch (balloon->info.type) {
    case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_PCI:
        name = "virtio-balloon-pci";
        break;
    case VIR_DOMAIN_DEVICE_ADDRESS_TYPE_CCW:
        name = "virtio-balloon-ccw";
        break;
    default:
1149
        return;
1150
    }
1151

1152 1153 1154
    if (qemuMonitorJSONFindLinkPath(mon, name, balloon->info.alias, &path) < 0)
        return;

1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178
    nprops = qemuMonitorJSONGetObjectListPaths(mon, path, &bprops);
    if (nprops < 0)
        goto cleanup;

    for (i = 0; i < nprops; i++) {
        if (STREQ(bprops[i]->name, "guest-stats-polling-interval")) {
            VIR_DEBUG("Found Balloon Object Path %s", path);
            mon->balloonpath = path;
            path = NULL;
            goto cleanup;
        }
    }


    /* If we get here, we found the path, but not the property */
    virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                   _("Property 'guest-stats-polling-interval' "
                     "not found on memory balloon driver."));

 cleanup:
    for (i = 0; i < nprops; i++)
        qemuMonitorJSONListPathFree(bprops[i]);
    VIR_FREE(bprops);
    VIR_FREE(path);
1179
    return;
1180 1181
}

1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196

/**
 * To update video memory size in status XML we need to load correct values from
 * QEMU.  This is supported only with JSON monitor.
 *
 * Returns 0 on success, -1 on failure and sets proper error message.
 */
int
qemuMonitorUpdateVideoMemorySize(qemuMonitorPtr mon,
                                 virDomainVideoDefPtr video,
                                 const char *videoName)
{
    int ret = -1;
    char *path = NULL;

1197 1198
    QEMU_CHECK_MONITOR(mon);

1199
    if (mon->json) {
1200 1201
        ret = qemuMonitorJSONFindLinkPath(mon, videoName,
                                          video->info.alias, &path);
1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215 1216 1217 1218
        if (ret < 0) {
            if (ret == -2)
                virReportError(VIR_ERR_INTERNAL_ERROR,
                               _("Failed to find QOM Object path for "
                                 "device '%s'"), videoName);
            return -1;
        }

        ret = qemuMonitorJSONUpdateVideoMemorySize(mon, video, path);
        VIR_FREE(path);
        return ret;
    }

    return 0;
}


1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235
/**
 * To update video vram64 size in status XML we need to load correct value from
 * QEMU.  This is supported only with JSON monitor.
 *
 * Returns 0 on success, -1 on failure and sets proper error message.
 */
int
qemuMonitorUpdateVideoVram64Size(qemuMonitorPtr mon,
                                 virDomainVideoDefPtr video,
                                 const char *videoName)
{
    int ret = -1;
    char *path = NULL;

    QEMU_CHECK_MONITOR(mon);

    if (mon->json) {
1236 1237
        ret = qemuMonitorJSONFindLinkPath(mon, videoName,
                                          video->info.alias, &path);
1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254
        if (ret < 0) {
            if (ret == -2)
                virReportError(VIR_ERR_INTERNAL_ERROR,
                               _("Failed to find QOM Object path for "
                                 "device '%s'"), videoName);
            return -1;
        }

        ret = qemuMonitorJSONUpdateVideoVram64Size(mon, video, path);
        VIR_FREE(path);
        return ret;
    }

    return 0;
}


1255 1256 1257 1258 1259
int
qemuMonitorHMPCommandWithFd(qemuMonitorPtr mon,
                            const char *cmd,
                            int scm_fd,
                            char **reply)
1260
{
1261 1262 1263
    char *json_cmd = NULL;
    int ret = -1;

1264 1265
    QEMU_CHECK_MONITOR(mon);

1266 1267 1268 1269 1270 1271 1272
    /* hack to avoid complicating each call to text monitor functions */
    json_cmd = qemuMonitorUnescapeArg(cmd);
    if (!json_cmd) {
        VIR_DEBUG("Could not unescape command: %s", cmd);
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Unable to unescape command"));
        goto cleanup;
1273
    }
1274
    ret = qemuMonitorJSONHumanCommandWithFd(mon, json_cmd, scm_fd, reply);
1275

1276
 cleanup:
1277 1278
    VIR_FREE(json_cmd);
    return ret;
1279 1280
}

1281

E
Eric Blake 已提交
1282
/* Ensure proper locking around callbacks.  */
1283 1284 1285 1286 1287 1288
#define QEMU_MONITOR_CALLBACK(mon, ret, callback, ...) \
    do { \
        virObjectRef(mon); \
        virObjectUnlock(mon); \
        if ((mon)->cb && (mon)->cb->callback) \
            (ret) = (mon)->cb->callback(mon, __VA_ARGS__, \
1289
                                        (mon)->callbackOpaque); \
1290 1291
        virObjectLock(mon); \
        virObjectUnref(mon); \
E
Eric Blake 已提交
1292
    } while (0)
1293

1294

1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308
int
qemuMonitorEmitEvent(qemuMonitorPtr mon, const char *event,
                     long long seconds, unsigned int micros,
                     const char *details)
{
    int ret = -1;
    VIR_DEBUG("mon=%p event=%s", mon, event);

    QEMU_MONITOR_CALLBACK(mon, ret, domainEvent, mon->vm, event, seconds,
                          micros, details);
    return ret;
}


1309
int
1310
qemuMonitorEmitShutdown(qemuMonitorPtr mon, virTristateBool guest)
1311 1312
{
    int ret = -1;
1313
    VIR_DEBUG("mon=%p guest=%u", mon, guest);
1314

1315
    QEMU_MONITOR_CALLBACK(mon, ret, domainShutdown, mon->vm, guest);
1316 1317 1318 1319
    return ret;
}


1320 1321
int
qemuMonitorEmitReset(qemuMonitorPtr mon)
1322 1323 1324 1325
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1326
    QEMU_MONITOR_CALLBACK(mon, ret, domainReset, mon->vm);
1327 1328 1329 1330
    return ret;
}


1331 1332
int
qemuMonitorEmitPowerdown(qemuMonitorPtr mon)
1333 1334 1335 1336
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1337
    QEMU_MONITOR_CALLBACK(mon, ret, domainPowerdown, mon->vm);
1338 1339 1340 1341
    return ret;
}


1342 1343
int
qemuMonitorEmitStop(qemuMonitorPtr mon)
1344 1345 1346 1347
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1348
    QEMU_MONITOR_CALLBACK(mon, ret, domainStop, mon->vm);
1349 1350 1351 1352
    return ret;
}


1353 1354
int
qemuMonitorEmitResume(qemuMonitorPtr mon)
1355 1356 1357 1358 1359 1360 1361 1362 1363
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainResume, mon->vm);
    return ret;
}


1364
int
1365 1366
qemuMonitorEmitGuestPanic(qemuMonitorPtr mon,
                          qemuMonitorEventPanicInfoPtr info)
1367 1368 1369
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);
1370
    QEMU_MONITOR_CALLBACK(mon, ret, domainGuestPanic, mon->vm, info);
1371 1372 1373 1374
    return ret;
}


1375 1376
int
qemuMonitorEmitRTCChange(qemuMonitorPtr mon, long long offset)
1377 1378 1379 1380
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1381
    QEMU_MONITOR_CALLBACK(mon, ret, domainRTCChange, mon->vm, offset);
1382 1383 1384 1385
    return ret;
}


1386 1387
int
qemuMonitorEmitWatchdog(qemuMonitorPtr mon, int action)
1388 1389 1390 1391
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1392
    QEMU_MONITOR_CALLBACK(mon, ret, domainWatchdog, mon->vm, action);
1393 1394 1395 1396
    return ret;
}


1397 1398 1399 1400 1401
int
qemuMonitorEmitIOError(qemuMonitorPtr mon,
                       const char *diskAlias,
                       int action,
                       const char *reason)
1402 1403 1404 1405
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1406 1407
    QEMU_MONITOR_CALLBACK(mon, ret, domainIOError, mon->vm,
                          diskAlias, action, reason);
1408 1409 1410 1411
    return ret;
}


1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423
int
qemuMonitorEmitGraphics(qemuMonitorPtr mon,
                        int phase,
                        int localFamily,
                        const char *localNode,
                        const char *localService,
                        int remoteFamily,
                        const char *remoteNode,
                        const char *remoteService,
                        const char *authScheme,
                        const char *x509dname,
                        const char *saslUsername)
1424 1425 1426 1427
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

E
Eric Blake 已提交
1428 1429 1430 1431
    QEMU_MONITOR_CALLBACK(mon, ret, domainGraphics, mon->vm, phase,
                          localFamily, localNode, localService,
                          remoteFamily, remoteNode, remoteService,
                          authScheme, x509dname, saslUsername);
1432 1433 1434
    return ret;
}

1435 1436 1437 1438 1439

int
qemuMonitorEmitTrayChange(qemuMonitorPtr mon,
                          const char *devAlias,
                          int reason)
1440 1441 1442 1443 1444 1445 1446 1447 1448 1449
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainTrayChange, mon->vm,
                          devAlias, reason);

    return ret;
}

1450 1451 1452

int
qemuMonitorEmitPMWakeup(qemuMonitorPtr mon)
O
Osier Yang 已提交
1453 1454 1455 1456 1457 1458 1459 1460 1461
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainPMWakeup, mon->vm);

    return ret;
}

1462 1463 1464

int
qemuMonitorEmitPMSuspend(qemuMonitorPtr mon)
O
Osier Yang 已提交
1465 1466 1467 1468 1469 1470 1471 1472 1473
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainPMSuspend, mon->vm);

    return ret;
}

1474 1475 1476

int
qemuMonitorEmitPMSuspendDisk(qemuMonitorPtr mon)
1477 1478 1479 1480 1481 1482 1483 1484 1485
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainPMSuspendDisk, mon->vm);

    return ret;
}

1486 1487 1488 1489 1490

int
qemuMonitorEmitBlockJob(qemuMonitorPtr mon,
                        const char *diskAlias,
                        int type,
1491 1492
                        int status,
                        const char *error)
1493 1494 1495 1496 1497
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainBlockJob, mon->vm,
1498
                          diskAlias, type, status, error);
1499 1500 1501
    return ret;
}

1502

1503 1504 1505
int
qemuMonitorEmitBalloonChange(qemuMonitorPtr mon,
                             unsigned long long actual)
1506 1507 1508 1509 1510 1511 1512 1513
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainBalloonChange, mon->vm, actual);
    return ret;
}

1514

1515 1516 1517 1518 1519 1520 1521 1522 1523 1524 1525 1526 1527
int
qemuMonitorEmitDeviceDeleted(qemuMonitorPtr mon,
                             const char *devAlias)
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainDeviceDeleted, mon->vm, devAlias);

    return ret;
}


1528 1529 1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540
int
qemuMonitorEmitNicRxFilterChanged(qemuMonitorPtr mon,
                                  const char *devAlias)
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainNicRxFilterChanged, mon->vm, devAlias);

    return ret;
}


1541 1542 1543 1544 1545 1546 1547 1548 1549 1550 1551 1552 1553 1554
int
qemuMonitorEmitSerialChange(qemuMonitorPtr mon,
                            const char *devAlias,
                            bool connected)
{
    int ret = -1;
    VIR_DEBUG("mon=%p, devAlias='%s', connected=%d", mon, devAlias, connected);

    QEMU_MONITOR_CALLBACK(mon, ret, domainSerialChange, mon->vm, devAlias, connected);

    return ret;
}


1555 1556 1557 1558 1559 1560 1561 1562 1563 1564 1565 1566
int
qemuMonitorEmitSpiceMigrated(qemuMonitorPtr mon)
{
    int ret = -1;
    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainSpiceMigrated, mon->vm);

    return ret;
}


1567 1568 1569 1570 1571 1572 1573 1574 1575 1576 1577 1578 1579 1580
int
qemuMonitorEmitMigrationStatus(qemuMonitorPtr mon,
                               int status)
{
    int ret = -1;
    VIR_DEBUG("mon=%p, status=%s",
              mon, NULLSTR(qemuMonitorMigrationStatusTypeToString(status)));

    QEMU_MONITOR_CALLBACK(mon, ret, domainMigrationStatus, mon->vm, status);

    return ret;
}


1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593
int
qemuMonitorEmitMigrationPass(qemuMonitorPtr mon,
                             int pass)
{
    int ret = -1;
    VIR_DEBUG("mon=%p, pass=%d", mon, pass);

    QEMU_MONITOR_CALLBACK(mon, ret, domainMigrationPass, mon->vm, pass);

    return ret;
}


1594 1595 1596 1597 1598 1599 1600 1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612
int
qemuMonitorEmitAcpiOstInfo(qemuMonitorPtr mon,
                           const char *alias,
                           const char *slotType,
                           const char *slot,
                           unsigned int source,
                           unsigned int status)
{
    int ret = -1;
    VIR_DEBUG("mon=%p, alias='%s', slotType='%s', slot='%s', source='%u' status=%u",
              mon, NULLSTR(alias), slotType, slot, source, status);

    QEMU_MONITOR_CALLBACK(mon, ret, domainAcpiOstInfo, mon->vm,
                          alias, slotType, slot, source, status);

    return ret;
}


1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626 1627 1628 1629 1630
int
qemuMonitorEmitBlockThreshold(qemuMonitorPtr mon,
                              const char *nodename,
                              unsigned long long threshold,
                              unsigned long long excess)
{
    int ret = -1;

    VIR_DEBUG("mon=%p, node-name='%s', threshold='%llu', excess='%llu'",
              mon, nodename, threshold, excess);

    QEMU_MONITOR_CALLBACK(mon, ret, domainBlockThreshold, mon->vm,
                          nodename, threshold, excess);

    return ret;
}


1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647
int
qemuMonitorEmitDumpCompleted(qemuMonitorPtr mon,
                             int status,
                             qemuMonitorDumpStatsPtr stats,
                             const char *error)
{
    int ret = -1;

    VIR_DEBUG("mon=%p", mon);

    QEMU_MONITOR_CALLBACK(mon, ret, domainDumpCompleted, mon->vm,
                          status, stats, error);

    return ret;
}


1648 1649
int
qemuMonitorSetCapabilities(qemuMonitorPtr mon)
1650
{
1651
    QEMU_CHECK_MONITOR(mon);
1652

1653 1654
    if (!mon->json)
        return 0;
1655

1656
    return qemuMonitorJSONSetCapabilities(mon);
1657 1658 1659
}


1660
int
1661
qemuMonitorStartCPUs(qemuMonitorPtr mon)
1662
{
1663
    QEMU_CHECK_MONITOR(mon);
1664

1665
    return qemuMonitorJSONStartCPUs(mon);
1666 1667 1668 1669 1670 1671
}


int
qemuMonitorStopCPUs(qemuMonitorPtr mon)
{
1672
    QEMU_CHECK_MONITOR(mon);
1673

1674
    return qemuMonitorJSONStopCPUs(mon);
1675 1676 1677
}


1678 1679 1680 1681 1682 1683 1684 1685
int
qemuMonitorCheck(qemuMonitorPtr mon)
{
    bool running;
    return qemuMonitorGetStatus(mon, &running, NULL);
}


1686
int
1687 1688 1689
qemuMonitorGetStatus(qemuMonitorPtr mon,
                     bool *running,
                     virDomainPausedReason *reason)
1690
{
1691
    VIR_DEBUG("running=%p, reason=%p", running, reason);
1692

1693
    QEMU_CHECK_MONITOR(mon);
1694

1695
    return qemuMonitorJSONGetStatus(mon, running, reason);
1696 1697 1698
}


1699 1700
int
qemuMonitorSystemPowerdown(qemuMonitorPtr mon)
1701
{
1702
    QEMU_CHECK_MONITOR(mon);
1703

1704
    return qemuMonitorJSONSystemPowerdown(mon);
1705 1706 1707
}


1708 1709
int
qemuMonitorSystemReset(qemuMonitorPtr mon)
1710
{
1711
    QEMU_CHECK_MONITOR(mon);
1712

1713
    return qemuMonitorJSONSystemReset(mon);
1714 1715 1716
}


1717 1718 1719 1720 1721 1722 1723 1724
static void
qemuMonitorCPUInfoClear(qemuMonitorCPUInfoPtr cpus,
                        size_t ncpus)
{
    size_t i;

    for (i = 0; i < ncpus; i++) {
        cpus[i].id = 0;
1725
        cpus[i].qemu_id = -1;
1726 1727 1728
        cpus[i].socket_id = -1;
        cpus[i].core_id = -1;
        cpus[i].thread_id = -1;
1729
        cpus[i].node_id = -1;
1730 1731
        cpus[i].vcpus = 0;
        cpus[i].tid = 0;
1732
        cpus[i].halted = false;
1733 1734 1735 1736 1737 1738 1739 1740

        VIR_FREE(cpus[i].qom_path);
        VIR_FREE(cpus[i].alias);
        VIR_FREE(cpus[i].type);
    }
}


1741 1742
void
qemuMonitorCPUInfoFree(qemuMonitorCPUInfoPtr cpus,
1743
                       size_t ncpus)
1744 1745 1746 1747
{
    if (!cpus)
        return;

1748 1749
    qemuMonitorCPUInfoClear(cpus, ncpus);

1750 1751 1752
    VIR_FREE(cpus);
}

1753 1754
void
qemuMonitorQueryCpusFree(struct qemuMonitorQueryCpusEntry *entries,
1755
                         size_t nentries)
1756
{
1757 1758
    size_t i;

1759 1760 1761
    if (!entries)
        return;

1762 1763 1764
    for (i = 0; i < nentries; i++)
        VIR_FREE(entries[i].qom_path);

1765 1766 1767
    VIR_FREE(entries);
}

1768

1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780
/**
 * Legacy approach doesn't allow out of order cpus, thus no complex matching
 * algorithm is necessary */
static void
qemuMonitorGetCPUInfoLegacy(struct qemuMonitorQueryCpusEntry *cpuentries,
                            size_t ncpuentries,
                            qemuMonitorCPUInfoPtr vcpus,
                            size_t maxvcpus)
{
    size_t i;

    for (i = 0; i < maxvcpus; i++) {
1781
        if (i < ncpuentries) {
1782
            vcpus[i].tid = cpuentries[i].tid;
1783
            vcpus[i].halted = cpuentries[i].halted;
1784
            vcpus[i].qemu_id = cpuentries[i].qemu_id;
1785
        }
1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798

        /* for legacy hotplug to work we need to fake the vcpu count added by
         * enabling a given vcpu */
        vcpus[i].vcpus = 1;
    }
}


/**
 * qemuMonitorGetCPUInfoHotplug:
 *
 * This function stitches together data retrieved via query-hotpluggable-cpus
 * which returns entities on the hotpluggable level (which may describe more
1799 1800
 * than one guest logical vcpu) with the output of query-cpus (or
 * query-cpus-fast), having an entry per enabled guest logical vcpu.
1801 1802 1803 1804
 *
 * query-hotpluggable-cpus conveys following information:
 * - topology information and number of logical vcpus this entry creates
 * - device type name of the entry that needs to be used when hotplugging
1805 1806
 * - qom path in qemu which can be used to map the entry against
 *   query-cpus[-fast]
1807
 *
1808
 * query-cpus[-fast] conveys following information:
1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828 1829 1830 1831
 * - thread id of a given guest logical vcpu
 * - order in which the vcpus were inserted
 * - qom path to allow mapping the two together
 *
 * The libvirt's internal structure has an entry for each possible (even
 * disabled) guest vcpu. The purpose is to map the data together so that we are
 * certain of the thread id mapping and the information required for vcpu
 * hotplug.
 *
 * This function returns 0 on success and -1 on error, but does not report
 * libvirt errors so that fallback approach can be used.
 */
static int
qemuMonitorGetCPUInfoHotplug(struct qemuMonitorQueryHotpluggableCpusEntry *hotplugvcpus,
                             size_t nhotplugvcpus,
                             struct qemuMonitorQueryCpusEntry *cpuentries,
                             size_t ncpuentries,
                             qemuMonitorCPUInfoPtr vcpus,
                             size_t maxvcpus)
{
    char *tmp;
    int order = 1;
    size_t totalvcpus = 0;
1832
    size_t mastervcpu; /* this iterator is used for iterating hotpluggable entities */
1833
    size_t slavevcpu; /* this corresponds to subentries of a hotpluggable entry */
1834
    size_t anyvcpu; /* this iterator is used for any vcpu entry in the result */
1835 1836 1837 1838 1839 1840 1841 1842
    size_t i;
    size_t j;

    /* ensure that the total vcpu count reported by query-hotpluggable-cpus equals
     * to the libvirt maximum cpu count */
    for (i = 0; i < nhotplugvcpus; i++)
        totalvcpus += hotplugvcpus[i].vcpus;

1843
    /* trim '/thread...' suffix from the data returned by query-cpus[-fast] */
1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854 1855
    for (i = 0; i < ncpuentries; i++) {
        if (cpuentries[i].qom_path &&
            (tmp = strstr(cpuentries[i].qom_path, "/thread")))
            *tmp = '\0';
    }

    if (totalvcpus != maxvcpus) {
        VIR_DEBUG("expected '%zu' total vcpus got '%zu'", maxvcpus, totalvcpus);
        return -1;
    }

    /* Note the order in which the hotpluggable entities are inserted by
1856
     * matching them to the query-cpus[-fast] entries */
1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874
    for (i = 0; i < ncpuentries; i++) {
        for (j = 0; j < nhotplugvcpus; j++) {
            if (!cpuentries[i].qom_path ||
                !hotplugvcpus[j].qom_path ||
                STRNEQ(cpuentries[i].qom_path, hotplugvcpus[j].qom_path))
                continue;

            /* add ordering info for hotpluggable entries */
            if (hotplugvcpus[j].enable_id == 0)
                hotplugvcpus[j].enable_id = order++;

            break;
        }
    }

    /* transfer appropriate data from the hotpluggable list to corresponding
     * entries. the entries returned by qemu may in fact describe multiple
     * logical vcpus in the guest */
1875
    mastervcpu = 0;
1876
    for (i = 0; i < nhotplugvcpus; i++) {
1877 1878 1879
        vcpus[mastervcpu].online = !!hotplugvcpus[i].qom_path;
        vcpus[mastervcpu].hotpluggable = !!hotplugvcpus[i].alias ||
                                         !vcpus[mastervcpu].online;
1880 1881 1882
        vcpus[mastervcpu].socket_id = hotplugvcpus[i].socket_id;
        vcpus[mastervcpu].core_id = hotplugvcpus[i].core_id;
        vcpus[mastervcpu].thread_id = hotplugvcpus[i].thread_id;
1883
        vcpus[mastervcpu].node_id = hotplugvcpus[i].node_id;
1884 1885 1886 1887 1888 1889
        vcpus[mastervcpu].vcpus = hotplugvcpus[i].vcpus;
        VIR_STEAL_PTR(vcpus[mastervcpu].qom_path, hotplugvcpus[i].qom_path);
        VIR_STEAL_PTR(vcpus[mastervcpu].alias, hotplugvcpus[i].alias);
        VIR_STEAL_PTR(vcpus[mastervcpu].type, hotplugvcpus[i].type);
        vcpus[mastervcpu].id = hotplugvcpus[i].enable_id;

1890 1891 1892 1893 1894 1895
        /* copy state information to slave vcpus */
        for (slavevcpu = mastervcpu + 1; slavevcpu < mastervcpu + hotplugvcpus[i].vcpus; slavevcpu++) {
            vcpus[slavevcpu].online = vcpus[mastervcpu].online;
            vcpus[slavevcpu].hotpluggable = vcpus[mastervcpu].hotpluggable;
        }

1896 1897
        /* calculate next master vcpu (hotpluggable unit) entry */
        mastervcpu += hotplugvcpus[i].vcpus;
1898 1899 1900 1901 1902 1903
    }

    /* match entries from query cpus to the output array taking into account
     * multi-vcpu objects */
    for (j = 0; j < ncpuentries; j++) {
        /* find the correct entry or beginning of group of entries */
1904 1905 1906
        for (anyvcpu = 0; anyvcpu < maxvcpus; anyvcpu++) {
            if (cpuentries[j].qom_path && vcpus[anyvcpu].qom_path &&
                STREQ(cpuentries[j].qom_path, vcpus[anyvcpu].qom_path))
1907 1908 1909
                break;
        }

1910
        if (anyvcpu == maxvcpus) {
1911
            VIR_DEBUG("too many query-cpus[-fast] entries for a given "
1912 1913 1914 1915
                      "query-hotpluggable-cpus entry");
            return -1;
        }

1916
        if (vcpus[anyvcpu].vcpus != 1) {
1917
            /* find a possibly empty vcpu thread for core granularity systems */
1918 1919
            for (; anyvcpu < maxvcpus; anyvcpu++) {
                if (vcpus[anyvcpu].tid == 0)
1920 1921 1922 1923
                    break;
            }
        }

1924
        vcpus[anyvcpu].qemu_id = cpuentries[j].qemu_id;
1925
        vcpus[anyvcpu].tid = cpuentries[j].tid;
1926
        vcpus[anyvcpu].halted = cpuentries[j].halted;
1927 1928 1929 1930 1931 1932
    }

    return 0;
}


1933 1934 1935
/**
 * qemuMonitorGetCPUInfo:
 * @mon: monitor
1936 1937
 * @vcpus: pointer filled by array of qemuMonitorCPUInfo structures
 * @maxvcpus: total possible number of vcpus
1938
 * @hotplug: query data relevant for hotplug support
1939
 * @fast: use QMP query-cpus-fast if supported
1940 1941 1942 1943
 *
 * Detects VCPU information. If qemu doesn't support or fails reporting
 * information this function will return success as other parts of libvirt
 * are able to cope with that.
1944
 *
1945 1946
 * Returns 0 on success (including if qemu didn't report any data) and
 *  -1 on error (reports libvirt error).
1947
 */
1948 1949
int
qemuMonitorGetCPUInfo(qemuMonitorPtr mon,
1950
                      qemuMonitorCPUInfoPtr *vcpus,
1951
                      size_t maxvcpus,
1952 1953
                      bool hotplug,
                      bool fast)
1954
{
1955 1956
    struct qemuMonitorQueryHotpluggableCpusEntry *hotplugcpus = NULL;
    size_t nhotplugcpus = 0;
1957 1958
    struct qemuMonitorQueryCpusEntry *cpuentries = NULL;
    size_t ncpuentries = 0;
1959 1960
    int ret = -1;
    int rc;
1961
    qemuMonitorCPUInfoPtr info = NULL;
1962

1963
    QEMU_CHECK_MONITOR(mon);
1964

1965 1966 1967
    if (VIR_ALLOC_N(info, maxvcpus) < 0)
        return -1;

1968 1969 1970
    if (!mon->json)
        hotplug = false;

1971 1972 1973 1974 1975 1976 1977
    /* initialize a few non-zero defaults */
    qemuMonitorCPUInfoClear(info, maxvcpus);

    if (hotplug &&
        (qemuMonitorJSONGetHotpluggableCPUs(mon, &hotplugcpus, &nhotplugcpus)) < 0)
        goto cleanup;

1978 1979
    rc = qemuMonitorJSONQueryCPUs(mon, &cpuentries, &ncpuentries, hotplug,
                                  fast);
1980 1981

    if (rc < 0) {
1982
        if (!hotplug && rc == -2) {
1983 1984 1985 1986
            VIR_STEAL_PTR(*vcpus, info);
            ret = 0;
        }

1987 1988 1989
        goto cleanup;
    }

1990 1991 1992 1993 1994 1995 1996 1997 1998
    if (!hotplugcpus ||
        qemuMonitorGetCPUInfoHotplug(hotplugcpus, nhotplugcpus,
                                     cpuentries, ncpuentries,
                                     info, maxvcpus) < 0) {
        /* Fallback to the legacy algorithm. Hotplug paths will make sure that
         * the apropriate data is present */
        qemuMonitorCPUInfoClear(info, maxvcpus);
        qemuMonitorGetCPUInfoLegacy(cpuentries, ncpuentries, info, maxvcpus);
    }
1999 2000 2001 2002 2003

    VIR_STEAL_PTR(*vcpus, info);
    ret = 0;

 cleanup:
2004
    qemuMonitorQueryHotpluggableCpusFree(hotplugcpus, nhotplugcpus);
2005
    qemuMonitorQueryCpusFree(cpuentries, ncpuentries);
2006
    qemuMonitorCPUInfoFree(info, maxvcpus);
2007
    return ret;
2008 2009
}

2010

2011 2012 2013 2014
/**
 * qemuMonitorGetCpuHalted:
 *
 * Returns a bitmap of vcpu id's that are halted. The id's correspond to the
2015
 * 'CPU' field as reported by query-cpus[-fast]'.
2016 2017 2018
 */
virBitmapPtr
qemuMonitorGetCpuHalted(qemuMonitorPtr mon,
2019
                        size_t maxvcpus,
2020
                        bool fast)
2021 2022 2023 2024 2025 2026 2027 2028 2029
{
    struct qemuMonitorQueryCpusEntry *cpuentries = NULL;
    size_t ncpuentries = 0;
    size_t i;
    int rc;
    virBitmapPtr ret = NULL;

    QEMU_CHECK_MONITOR_NULL(mon);

2030 2031
    rc = qemuMonitorJSONQueryCPUs(mon, &cpuentries, &ncpuentries, false,
                                  fast);
2032 2033 2034 2035 2036 2037 2038 2039 2040 2041 2042 2043 2044 2045 2046 2047 2048 2049

    if (rc < 0)
        goto cleanup;

    if (!(ret = virBitmapNew(maxvcpus)))
        goto cleanup;

    for (i = 0; i < ncpuentries; i++) {
        if (cpuentries[i].halted)
            ignore_value(virBitmapSetBit(ret, cpuentries[i].qemu_id));
    }

 cleanup:
    qemuMonitorQueryCpusFree(cpuentries, ncpuentries);
    return ret;
}


2050 2051 2052 2053
int
qemuMonitorSetLink(qemuMonitorPtr mon,
                   const char *name,
                   virDomainNetInterfaceLinkState state)
2054
{
2055
    VIR_DEBUG("name=%s, state=%u", name, state);
2056

2057
    QEMU_CHECK_MONITOR(mon);
2058

2059
    return qemuMonitorJSONSetLink(mon, name, state);
2060
}
2061

2062 2063 2064

int
qemuMonitorGetVirtType(qemuMonitorPtr mon,
2065
                       virDomainVirtType *virtType)
2066
{
2067
    QEMU_CHECK_MONITOR(mon);
2068

2069
    return qemuMonitorJSONGetVirtType(mon, virtType);
2070 2071 2072
}


2073 2074 2075 2076
/**
 * Returns: 0 if balloon not supported, +1 if balloon query worked
 * or -1 on failure
 */
2077 2078 2079
int
qemuMonitorGetBalloonInfo(qemuMonitorPtr mon,
                          unsigned long long *currmem)
2080
{
2081
    QEMU_CHECK_MONITOR(mon);
2082

2083
    return qemuMonitorJSONGetBalloonInfo(mon, currmem);
2084 2085 2086
}


2087 2088
int
qemuMonitorGetMemoryStats(qemuMonitorPtr mon,
2089
                          virDomainMemballoonDefPtr balloon,
2090 2091
                          virDomainMemoryStatPtr stats,
                          unsigned int nr_stats)
2092
{
2093
    VIR_DEBUG("stats=%p nstats=%u", stats, nr_stats);
2094

2095
    QEMU_CHECK_MONITOR(mon);
2096

2097 2098 2099
    qemuMonitorInitBalloonObjectPath(mon, balloon);
    return qemuMonitorJSONGetMemoryStats(mon, mon->balloonpath,
                                         stats, nr_stats);
2100 2101
}

2102

2103 2104 2105 2106 2107 2108 2109
/**
 * qemuMonitorSetMemoryStatsPeriod:
 *
 * This function sets balloon stats update period.
 *
 * Returns 0 on success and -1 on error, but does *not* set an error.
 */
2110 2111
int
qemuMonitorSetMemoryStatsPeriod(qemuMonitorPtr mon,
2112
                                virDomainMemballoonDefPtr balloon,
2113
                                int period)
2114 2115 2116 2117
{
    int ret = -1;
    VIR_DEBUG("mon=%p period=%d", mon, period);

2118
    if (!mon)
2119 2120
        return -1;

2121 2122 2123 2124
    if (!mon->json)
        return -1;

    if (period < 0)
2125 2126
        return -1;

2127
    qemuMonitorInitBalloonObjectPath(mon, balloon);
2128
    if (mon->balloonpath) {
2129 2130
        ret = qemuMonitorJSONSetMemoryStatsPeriod(mon, mon->balloonpath,
                                                  period);
2131 2132 2133 2134 2135 2136 2137 2138 2139

        /*
         * Most of the calls to this function are supposed to be
         * non-fatal and the only one that should be fatal wants its
         * own error message.  More details for debugging will be in
         * the log file.
         */
        if (ret < 0)
            virResetLastError();
2140 2141 2142 2143
    }
    return ret;
}

2144

2145 2146 2147 2148 2149 2150
int
qemuMonitorBlockIOStatusToError(const char *status)
{
    int st = qemuMonitorBlockIOStatusTypeFromString(status);

    if (st < 0) {
2151 2152
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("unknown block IO status: %s"), status);
2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170
        return -1;
    }

    switch ((qemuMonitorBlockIOStatus) st) {
    case QEMU_MONITOR_BLOCK_IO_STATUS_OK:
        return VIR_DOMAIN_DISK_ERROR_NONE;
    case QEMU_MONITOR_BLOCK_IO_STATUS_FAILED:
        return VIR_DOMAIN_DISK_ERROR_UNSPEC;
    case QEMU_MONITOR_BLOCK_IO_STATUS_NOSPACE:
        return VIR_DOMAIN_DISK_ERROR_NO_SPACE;

    /* unreachable */
    case QEMU_MONITOR_BLOCK_IO_STATUS_LAST:
        break;
    }
    return -1;
}

2171

2172 2173 2174 2175 2176 2177 2178 2179 2180 2181
static void
qemuDomainDiskInfoFree(void *value, const void *name ATTRIBUTE_UNUSED)
{
    struct qemuDomainDiskInfo *info = value;

    VIR_FREE(info->nodename);
    VIR_FREE(info);
}


2182 2183
virHashTablePtr
qemuMonitorGetBlockInfo(qemuMonitorPtr mon)
2184 2185
{
    int ret;
2186 2187
    virHashTablePtr table;

2188
    QEMU_CHECK_MONITOR_NULL(mon);
2189

2190
    if (!(table = virHashCreate(32, qemuDomainDiskInfoFree)))
2191 2192
        return NULL;

2193
    ret = qemuMonitorJSONGetBlockInfo(mon, table);
2194 2195 2196 2197 2198 2199 2200 2201 2202

    if (ret < 0) {
        virHashFree(table);
        return NULL;
    }

    return table;
}

2203

2204 2205 2206 2207 2208 2209 2210 2211 2212
/**
 * qemuMonitorQueryBlockstats:
 * @mon: monitor object
 *
 * Returns data from a call to 'query-blockstats'.
 */
virJSONValuePtr
qemuMonitorQueryBlockstats(qemuMonitorPtr mon)
{
2213
    QEMU_CHECK_MONITOR_NULL(mon);
2214 2215 2216 2217 2218

    return qemuMonitorJSONQueryBlockstats(mon);
}


2219 2220 2221 2222 2223 2224 2225 2226 2227 2228 2229
/**
 * qemuMonitorGetAllBlockStatsInfo:
 * @mon: monitor object
 * @ret_stats: pointer that is filled with a hash table containing the stats
 * @backingChain: recurse into the backing chain of devices
 *
 * Creates a hash table in @ret_stats with block stats of all devices. In case
 * @backingChain is true @ret_stats will additionally contain stats for
 * backing chain members of block devices.
 *
 * Returns < 0 on error, count of supported block stats fields on success.
2230 2231 2232
 */
int
qemuMonitorGetAllBlockStatsInfo(qemuMonitorPtr mon,
2233 2234
                                virHashTablePtr *ret_stats,
                                bool backingChain)
2235
{
2236
    int ret = -1;
2237 2238 2239
    VIR_DEBUG("ret_stats=%p, backing=%d", ret_stats, backingChain);

    QEMU_CHECK_MONITOR(mon);
2240

2241 2242 2243
    if (!(*ret_stats = virHashCreate(10, virHashValueFree)))
        goto error;

2244 2245
    ret = qemuMonitorJSONGetAllBlockStatsInfo(mon, *ret_stats,
                                              backingChain);
2246

2247 2248 2249 2250
    if (ret < 0)
        goto error;

    return ret;
2251 2252 2253 2254 2255

 error:
    virHashFree(*ret_stats);
    *ret_stats = NULL;
    return -1;
2256 2257
}

2258 2259

/* Updates "stats" to fill virtual and physical size of the image */
2260 2261 2262 2263
int
qemuMonitorBlockStatsUpdateCapacity(qemuMonitorPtr mon,
                                    virHashTablePtr stats,
                                    bool backingChain)
2264
{
2265
    VIR_DEBUG("stats=%p, backing=%d", stats, backingChain);
2266

2267
    QEMU_CHECK_MONITOR(mon);
2268

2269
    return qemuMonitorJSONBlockStatsUpdateCapacity(mon, stats, backingChain);
2270 2271 2272
}


2273 2274 2275 2276
int
qemuMonitorBlockResize(qemuMonitorPtr mon,
                       const char *device,
                       unsigned long long size)
2277
{
2278 2279 2280
    VIR_DEBUG("device=%s size=%llu", device, size);

    QEMU_CHECK_MONITOR(mon);
2281

2282
    return qemuMonitorJSONBlockResize(mon, device, size);
2283
}
2284

2285 2286 2287 2288

int
qemuMonitorSetVNCPassword(qemuMonitorPtr mon,
                          const char *password)
2289
{
2290
    VIR_DEBUG("password=%p", password);
2291

2292
    QEMU_CHECK_MONITOR(mon);
2293

2294 2295 2296
    if (!password)
        password = "";

2297
    return qemuMonitorJSONSetVNCPassword(mon, password);
2298 2299
}

2300 2301 2302

static const char *
qemuMonitorTypeToProtocol(int type)
2303 2304 2305 2306 2307 2308 2309
{
    switch (type) {
    case VIR_DOMAIN_GRAPHICS_TYPE_VNC:
        return "vnc";
    case VIR_DOMAIN_GRAPHICS_TYPE_SPICE:
        return "spice";
    default:
2310 2311 2312
        virReportError(VIR_ERR_INVALID_ARG,
                       _("unsupported protocol type %s"),
                       virDomainGraphicsTypeToString(type));
2313 2314 2315 2316
        return NULL;
    }
}

2317

2318
/* Returns -2 if not supported with this monitor connection */
2319 2320 2321 2322 2323
int
qemuMonitorSetPassword(qemuMonitorPtr mon,
                       int type,
                       const char *password,
                       const char *action_if_connected)
2324 2325 2326 2327 2328 2329
{
    const char *protocol = qemuMonitorTypeToProtocol(type);

    if (!protocol)
        return -1;

2330 2331
    VIR_DEBUG("protocol=%s, password=%p, action_if_connected=%s",
              protocol, password, action_if_connected);
2332

2333
    QEMU_CHECK_MONITOR(mon);
2334 2335 2336 2337 2338 2339 2340

    if (!password)
        password = "";

    if (!action_if_connected)
        action_if_connected = "keep";

2341
    return qemuMonitorJSONSetPassword(mon, protocol, password, action_if_connected);
2342 2343
}

2344 2345 2346 2347 2348

int
qemuMonitorExpirePassword(qemuMonitorPtr mon,
                          int type,
                          const char *expire_time)
2349 2350 2351 2352 2353 2354
{
    const char *protocol = qemuMonitorTypeToProtocol(type);

    if (!protocol)
        return -1;

2355
    VIR_DEBUG("protocol=%s, expire_time=%s", protocol, expire_time);
2356

2357
    QEMU_CHECK_MONITOR(mon);
2358 2359 2360 2361

    if (!expire_time)
        expire_time = "now";

2362
    return qemuMonitorJSONExpirePassword(mon, protocol, expire_time);
2363
}
2364

2365

2366 2367 2368 2369
/*
 * Returns: 0 if balloon not supported, +1 if balloon adjust worked
 * or -1 on failure
 */
2370 2371
int
qemuMonitorSetBalloon(qemuMonitorPtr mon,
2372
                      unsigned long long newmem)
2373
{
2374
    VIR_DEBUG("newmem=%llu", newmem);
2375

2376
    QEMU_CHECK_MONITOR(mon);
2377

2378
    return qemuMonitorJSONSetBalloon(mon, newmem);
2379 2380
}

2381

2382 2383 2384
/*
 * Returns: 0 if CPU modification was successful or -1 on failure
 */
2385 2386
int
qemuMonitorSetCPU(qemuMonitorPtr mon, int cpu, bool online)
2387
{
2388
    VIR_DEBUG("cpu=%d online=%d", cpu, online);
2389

2390
    QEMU_CHECK_MONITOR(mon);
2391

2392
    return qemuMonitorJSONSetCPU(mon, cpu, online);
2393 2394 2395
}


2396 2397 2398 2399
int
qemuMonitorEjectMedia(qemuMonitorPtr mon,
                      const char *dev_name,
                      bool force)
2400
{
2401
    VIR_DEBUG("dev_name=%s force=%d", dev_name, force);
2402

2403
    QEMU_CHECK_MONITOR(mon);
2404

2405
    return qemuMonitorJSONEjectMedia(mon, dev_name, force);
2406 2407 2408
}


2409 2410 2411 2412 2413
int
qemuMonitorChangeMedia(qemuMonitorPtr mon,
                       const char *dev_name,
                       const char *newmedia,
                       const char *format)
2414
{
2415
    VIR_DEBUG("dev_name=%s newmedia=%s format=%s", dev_name, newmedia, format);
2416

2417
    QEMU_CHECK_MONITOR(mon);
2418

2419
    return qemuMonitorJSONChangeMedia(mon, dev_name, newmedia, format);
2420 2421 2422
}


2423 2424 2425 2426 2427
int
qemuMonitorSaveVirtualMemory(qemuMonitorPtr mon,
                             unsigned long long offset,
                             size_t length,
                             const char *path)
2428
{
2429
    VIR_DEBUG("offset=%llu length=%zu path=%s", offset, length, path);
2430

2431
    QEMU_CHECK_MONITOR(mon);
2432

2433
    return qemuMonitorJSONSaveVirtualMemory(mon, offset, length, path);
2434 2435
}

2436 2437 2438 2439 2440 2441

int
qemuMonitorSavePhysicalMemory(qemuMonitorPtr mon,
                              unsigned long long offset,
                              size_t length,
                              const char *path)
2442
{
2443
    VIR_DEBUG("offset=%llu length=%zu path=%s", offset, length, path);
2444

2445
    QEMU_CHECK_MONITOR(mon);
2446

2447
    return qemuMonitorJSONSavePhysicalMemory(mon, offset, length, path);
2448 2449 2450
}


2451 2452 2453
int
qemuMonitorSetMigrationSpeed(qemuMonitorPtr mon,
                             unsigned long bandwidth)
2454
{
2455
    VIR_DEBUG("bandwidth=%lu", bandwidth);
2456

2457
    QEMU_CHECK_MONITOR(mon);
2458

2459 2460 2461 2462 2463 2464 2465
    if (bandwidth > QEMU_DOMAIN_MIG_BANDWIDTH_MAX) {
        virReportError(VIR_ERR_OVERFLOW,
                       _("bandwidth must be less than %llu"),
                       QEMU_DOMAIN_MIG_BANDWIDTH_MAX + 1ULL);
        return -1;
    }

2466
    return qemuMonitorJSONSetMigrationSpeed(mon, bandwidth);
2467 2468
}

2469

2470 2471 2472
int
qemuMonitorSetMigrationDowntime(qemuMonitorPtr mon,
                                unsigned long long downtime)
2473
{
2474
    VIR_DEBUG("downtime=%llu", downtime);
2475

2476
    QEMU_CHECK_MONITOR(mon);
2477

2478
    return qemuMonitorJSONSetMigrationDowntime(mon, downtime);
2479 2480 2481
}


2482 2483 2484 2485
int
qemuMonitorGetMigrationCacheSize(qemuMonitorPtr mon,
                                 unsigned long long *cacheSize)
{
2486
    VIR_DEBUG("cacheSize=%p", cacheSize);
2487

2488
    QEMU_CHECK_MONITOR(mon);
2489 2490 2491 2492

    return qemuMonitorJSONGetMigrationCacheSize(mon, cacheSize);
}

2493

2494 2495 2496 2497
int
qemuMonitorSetMigrationCacheSize(qemuMonitorPtr mon,
                                 unsigned long long cacheSize)
{
2498
    VIR_DEBUG("cacheSize=%llu", cacheSize);
2499

2500
    QEMU_CHECK_MONITOR(mon);
2501 2502 2503 2504 2505

    return qemuMonitorJSONSetMigrationCacheSize(mon, cacheSize);
}


2506 2507 2508 2509 2510 2511 2512 2513 2514 2515 2516
/**
 * qemuMonitorGetMigrationParams:
 * @mon: Pointer to the monitor object.
 * @params: Where to store migration parameters.
 *
 * If QEMU does not support querying migration parameters, the function will
 * set @params to NULL and return 0 (success). The caller is responsible for
 * freeing @params.
 *
 * Returns 0 on success, -1 on error.
 */
2517
int
2518
qemuMonitorGetMigrationParams(qemuMonitorPtr mon,
2519
                              virJSONValuePtr *params)
2520
{
2521
    QEMU_CHECK_MONITOR(mon);
2522

2523
    return qemuMonitorJSONGetMigrationParams(mon, params);
2524 2525
}

2526 2527 2528 2529 2530 2531 2532 2533 2534 2535 2536

/**
 * qemuMonitorSetMigrationParams:
 * @mon: Pointer to the monitor object.
 * @params: Migration parameters.
 *
 * The @params object is consumed and should not be referenced by the caller
 * after this function returns.
 *
 * Returns 0 on success, -1 on error.
 */
2537
int
2538
qemuMonitorSetMigrationParams(qemuMonitorPtr mon,
2539 2540
                              virJSONValuePtr params)
{
2541
    QEMU_CHECK_MONITOR_GOTO(mon, error);
2542

2543
    return qemuMonitorJSONSetMigrationParams(mon, params);
2544 2545 2546 2547

 error:
    virJSONValueFree(params);
    return -1;
2548 2549 2550
}


2551
int
2552
qemuMonitorGetMigrationStats(qemuMonitorPtr mon,
2553 2554
                             qemuMonitorMigrationStatsPtr stats,
                             char **error)
2555
{
2556
    QEMU_CHECK_MONITOR(mon);
2557

2558 2559 2560
    if (error)
        *error = NULL;

2561
    return qemuMonitorJSONGetMigrationStats(mon, stats, error);
2562 2563 2564
}


2565 2566 2567 2568
int
qemuMonitorMigrateToFd(qemuMonitorPtr mon,
                       unsigned int flags,
                       int fd)
E
Eric Blake 已提交
2569 2570
{
    int ret;
2571
    VIR_DEBUG("fd=%d flags=0x%x", fd, flags);
E
Eric Blake 已提交
2572

2573
    QEMU_CHECK_MONITOR(mon);
E
Eric Blake 已提交
2574 2575 2576 2577

    if (qemuMonitorSendFileHandle(mon, "migrate", fd) < 0)
        return -1;

2578
    ret = qemuMonitorJSONMigrate(mon, flags, "fd:migrate");
E
Eric Blake 已提交
2579 2580 2581

    if (ret < 0) {
        if (qemuMonitorCloseFileHandle(mon, "migrate") < 0)
2582
            VIR_WARN("failed to close migration handle");
E
Eric Blake 已提交
2583 2584 2585 2586 2587 2588
    }

    return ret;
}


2589 2590 2591 2592 2593 2594
int
qemuMonitorMigrateToHost(qemuMonitorPtr mon,
                         unsigned int flags,
                         const char *protocol,
                         const char *hostname,
                         int port)
2595
{
D
Daniel P. Berrange 已提交
2596
    int ret;
2597
    char *uri = NULL;
2598
    VIR_DEBUG("hostname=%s port=%d flags=0x%x", hostname, port, flags);
2599

2600
    QEMU_CHECK_MONITOR(mon);
2601

2602 2603 2604 2605
    if (strchr(hostname, ':')) {
        if (virAsprintf(&uri, "%s:[%s]:%d", protocol, hostname, port) < 0)
            return -1;
    } else if (virAsprintf(&uri, "%s:%s:%d", protocol, hostname, port) < 0) {
2606
        return -1;
2607
    }
2608

2609
    ret = qemuMonitorJSONMigrate(mon, flags, uri);
2610 2611

    VIR_FREE(uri);
D
Daniel P. Berrange 已提交
2612
    return ret;
2613 2614 2615
}


2616 2617
int
qemuMonitorMigrateCancel(qemuMonitorPtr mon)
2618
{
2619
    QEMU_CHECK_MONITOR(mon);
2620

2621
    return qemuMonitorJSONMigrateCancel(mon);
2622 2623
}

2624

2625 2626 2627 2628
int
qemuMonitorQueryDump(qemuMonitorPtr mon,
                     qemuMonitorDumpStatsPtr stats)
{
2629
    QEMU_CHECK_MONITOR(mon);
2630 2631 2632 2633 2634

    return qemuMonitorJSONQueryDump(mon, stats);
}


2635 2636 2637
/**
 * Returns 1 if @capability is supported, 0 if it's not, or -1 on error.
 */
2638 2639 2640
int
qemuMonitorGetDumpGuestMemoryCapability(qemuMonitorPtr mon,
                                        const char *capability)
2641
{
2642
    VIR_DEBUG("capability=%s", capability);
2643

2644
    QEMU_CHECK_MONITOR(mon);
2645 2646 2647 2648 2649 2650 2651 2652

    /* No capability is supported without JSON monitor */
    if (!mon->json)
        return 0;

    return qemuMonitorJSONGetDumpGuestMemoryCapability(mon, capability);
}

2653

2654
int
2655 2656 2657 2658
qemuMonitorDumpToFd(qemuMonitorPtr mon,
                    int fd,
                    const char *dumpformat,
                    bool detach)
2659 2660
{
    int ret;
2661
    VIR_DEBUG("fd=%d dumpformat=%s", fd, dumpformat);
2662

2663
    QEMU_CHECK_MONITOR(mon);
2664 2665 2666 2667

    if (qemuMonitorSendFileHandle(mon, "dump", fd) < 0)
        return -1;

2668
    ret = qemuMonitorJSONDump(mon, "fd:dump", dumpformat, detach);
2669 2670 2671 2672 2673 2674 2675 2676

    if (ret < 0) {
        if (qemuMonitorCloseFileHandle(mon, "dump") < 0)
            VIR_WARN("failed to close dumping handle");
    }

    return ret;
}
2677

2678 2679 2680 2681 2682 2683 2684 2685

int
qemuMonitorGraphicsRelocate(qemuMonitorPtr mon,
                            int type,
                            const char *hostname,
                            int port,
                            int tlsPort,
                            const char *tlsSubject)
2686
{
2687 2688 2689 2690
    VIR_DEBUG("type=%d hostname=%s port=%d tlsPort=%d tlsSubject=%s",
              type, hostname, port, tlsPort, NULLSTR(tlsSubject));

    QEMU_CHECK_MONITOR(mon);
2691

2692 2693 2694 2695 2696 2697
    return qemuMonitorJSONGraphicsRelocate(mon,
                                           type,
                                           hostname,
                                           port,
                                           tlsPort,
                                           tlsSubject);
2698 2699 2700
}


2701 2702 2703 2704
int
qemuMonitorSendFileHandle(qemuMonitorPtr mon,
                          const char *fdname,
                          int fd)
2705
{
2706
    VIR_DEBUG("fdname=%s fd=%d", fdname, fd);
2707

2708
    QEMU_CHECK_MONITOR(mon);
2709

2710
    if (fd < 0) {
2711 2712
        virReportError(VIR_ERR_INVALID_ARG, "%s",
                       _("fd must be valid"));
2713 2714 2715 2716
        return -1;
    }

    if (!mon->hasSendFD) {
2717
        virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
2718 2719
                       _("qemu is not using a unix socket monitor, "
                         "cannot send fd %s"), fdname);
2720 2721 2722
        return -1;
    }

2723
    return qemuMonitorJSONSendFileHandle(mon, fdname, fd);
2724 2725 2726
}


2727 2728 2729
int
qemuMonitorCloseFileHandle(qemuMonitorPtr mon,
                           const char *fdname)
2730
{
2731 2732 2733
    int ret = -1;
    virErrorPtr error;

2734
    VIR_DEBUG("fdname=%s", fdname);
2735

2736 2737
    error = virSaveLastError();

2738
    QEMU_CHECK_MONITOR_GOTO(mon, cleanup);
2739

2740
    ret = qemuMonitorJSONCloseFileHandle(mon, fdname);
2741

2742
 cleanup:
2743 2744 2745 2746
    if (error) {
        virSetError(error);
        virFreeError(error);
    }
D
Daniel P. Berrange 已提交
2747
    return ret;
2748 2749 2750
}


2751 2752 2753 2754 2755 2756
/* Add the open file descriptor FD into the non-negative set FDSET.
 * If NAME is present, it will be passed along for logging purposes.
 * Returns the counterpart fd that qemu received, or -1 on error.  */
int
qemuMonitorAddFd(qemuMonitorPtr mon, int fdset, int fd, const char *name)
{
2757
    VIR_DEBUG("fdset=%d, fd=%d, name=%s", fdset, fd, NULLSTR(name));
2758

2759
    QEMU_CHECK_MONITOR(mon);
2760 2761 2762 2763 2764 2765 2766 2767 2768 2769 2770 2771 2772 2773

    if (fd < 0 || fdset < 0) {
        virReportError(VIR_ERR_INVALID_ARG, "%s",
                       _("fd and fdset must be valid"));
        return -1;
    }

    if (!mon->hasSendFD) {
        virReportError(VIR_ERR_OPERATION_UNSUPPORTED,
                       _("qemu is not using a unix socket monitor, "
                         "cannot send fd %s"), NULLSTR(name));
        return -1;
    }

2774
    return qemuMonitorJSONAddFd(mon, fdset, fd, name);
2775 2776 2777 2778 2779 2780 2781 2782 2783 2784 2785 2786
}


/* Remove one of qemu's fds from the given FDSET, or if FD is
 * negative, remove the entire set.  Preserve any previous error on
 * entry.  Returns 0 on success, -1 on error.  */
int
qemuMonitorRemoveFd(qemuMonitorPtr mon, int fdset, int fd)
{
    int ret = -1;
    virErrorPtr error;

2787
    VIR_DEBUG("fdset=%d, fd=%d", fdset, fd);
2788 2789 2790

    error = virSaveLastError();

2791
    QEMU_CHECK_MONITOR_GOTO(mon, cleanup);
2792

2793
    ret = qemuMonitorJSONRemoveFd(mon, fdset, fd);
2794

2795
 cleanup:
2796 2797 2798 2799 2800 2801 2802 2803
    if (error) {
        virSetError(error);
        virFreeError(error);
    }
    return ret;
}


2804 2805 2806 2807 2808
int
qemuMonitorAddNetdev(qemuMonitorPtr mon,
                     const char *netdevstr,
                     int *tapfd, char **tapfdName, int tapfdSize,
                     int *vhostfd, char **vhostfdName, int vhostfdSize)
2809
{
2810
    int ret = -1;
2811
    size_t i = 0, j = 0;
2812

2813
    VIR_DEBUG("netdevstr=%s tapfd=%p tapfdName=%p tapfdSize=%d"
2814
              "vhostfd=%p vhostfdName=%p vhostfdSize=%d",
2815
              netdevstr, tapfd, tapfdName, tapfdSize,
2816
              vhostfd, vhostfdName, vhostfdSize);
2817

2818
    QEMU_CHECK_MONITOR(mon);
2819

2820 2821 2822 2823 2824 2825 2826
    for (i = 0; i < tapfdSize; i++) {
        if (qemuMonitorSendFileHandle(mon, tapfdName[i], tapfd[i]) < 0)
            goto cleanup;
    }
    for (j = 0; j < vhostfdSize; j++) {
        if (qemuMonitorSendFileHandle(mon, vhostfdName[j], vhostfd[j]) < 0)
            goto cleanup;
2827 2828
    }

2829
    ret = qemuMonitorJSONAddNetdev(mon, netdevstr);
2830

2831
 cleanup:
2832
    if (ret < 0) {
2833 2834 2835 2836 2837 2838 2839 2840
        while (i--) {
            if (qemuMonitorCloseFileHandle(mon, tapfdName[i]) < 0)
                VIR_WARN("failed to close device handle '%s'", tapfdName[i]);
        }
        while (j--) {
            if (qemuMonitorCloseFileHandle(mon, vhostfdName[j]) < 0)
                VIR_WARN("failed to close device handle '%s'", vhostfdName[j]);
        }
2841 2842
    }

2843 2844 2845
    return ret;
}

2846 2847 2848 2849

int
qemuMonitorRemoveNetdev(qemuMonitorPtr mon,
                        const char *alias)
2850
{
2851
    VIR_DEBUG("alias=%s", alias);
2852

2853
    QEMU_CHECK_MONITOR(mon);
2854

2855
    return qemuMonitorJSONRemoveNetdev(mon, alias);
2856 2857 2858
}


2859 2860 2861 2862
int
qemuMonitorQueryRxFilter(qemuMonitorPtr mon, const char *alias,
                         virNetDevRxFilterPtr *filter)
{
2863
    VIR_DEBUG("alias=%s filter=%p", alias, filter);
2864

2865
    QEMU_CHECK_MONITOR(mon);
2866 2867

    return qemuMonitorJSONQueryRxFilter(mon, alias, filter);
2868 2869 2870
}


2871
void
2872 2873 2874 2875 2876 2877 2878 2879 2880 2881
qemuMonitorChardevInfoFree(void *data,
                           const void *name ATTRIBUTE_UNUSED)
{
    qemuMonitorChardevInfoPtr info = data;

    VIR_FREE(info->ptyPath);
    VIR_FREE(info);
}


2882 2883 2884
int
qemuMonitorGetChardevInfo(qemuMonitorPtr mon,
                          virHashTablePtr *retinfo)
2885
{
2886
    int ret;
2887 2888
    virHashTablePtr info = NULL;

2889
    VIR_DEBUG("retinfo=%p", retinfo);
2890

2891
    QEMU_CHECK_MONITOR_GOTO(mon, error);
2892

2893
    if (!(info = virHashCreate(10, qemuMonitorChardevInfoFree)))
2894 2895
        goto error;

2896
    ret = qemuMonitorJSONGetChardevInfo(mon, info);
2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907

    if (ret < 0)
        goto error;

    *retinfo = info;
    return 0;

 error:
    virHashFree(info);
    *retinfo = NULL;
    return -1;
2908
}
2909 2910


2911 2912 2913 2914 2915 2916 2917
/**
 * qemuMonitorDriveDel:
 * @mon: monitor object
 * @drivestr: identifier of drive to delete.
 *
 * Attempts to remove a host drive.
 * Returns 1 if unsupported, 0 if ok, and -1 on other failure */
2918 2919 2920
int
qemuMonitorDriveDel(qemuMonitorPtr mon,
                    const char *drivestr)
2921
{
2922
    VIR_DEBUG("drivestr=%s", drivestr);
2923

2924
    QEMU_CHECK_MONITOR(mon);
2925

2926 2927
    /* there won't be a direct replacement for drive_del in QMP */
    return qemuMonitorTextDriveDel(mon, drivestr);
2928 2929
}

2930 2931 2932 2933

int
qemuMonitorDelDevice(qemuMonitorPtr mon,
                     const char *devalias)
2934
{
2935
    VIR_DEBUG("devalias=%s", devalias);
2936

2937
    QEMU_CHECK_MONITOR(mon);
2938

2939
    return qemuMonitorJSONDelDevice(mon, devalias);
2940 2941
}

2942

2943 2944 2945 2946 2947
int
qemuMonitorAddDeviceWithFd(qemuMonitorPtr mon,
                           const char *devicestr,
                           int fd,
                           const char *fdname)
2948
{
2949
    VIR_DEBUG("device=%s fd=%d fdname=%s", devicestr, fd, NULLSTR(fdname));
2950 2951
    int ret;

2952
    QEMU_CHECK_MONITOR(mon);
2953

2954 2955 2956
    if (fd >= 0 && qemuMonitorSendFileHandle(mon, fdname, fd) < 0)
        return -1;

2957
    ret = qemuMonitorJSONAddDevice(mon, devicestr);
2958 2959 2960 2961 2962 2963

    if (ret < 0 && fd >= 0) {
        if (qemuMonitorCloseFileHandle(mon, fdname) < 0)
            VIR_WARN("failed to close device handle '%s'", fdname);
    }

2964 2965 2966
    return ret;
}

2967 2968 2969 2970

int
qemuMonitorAddDevice(qemuMonitorPtr mon,
                     const char *devicestr)
2971 2972 2973 2974
{
    return qemuMonitorAddDeviceWithFd(mon, devicestr, -1, NULL);
}

2975

2976 2977 2978 2979 2980 2981 2982 2983 2984 2985 2986 2987
/**
 * qemuMonitorAddDeviceArgs:
 * @mon: monitor object
 * @args: arguments for device add, consumed on success or failure
 *
 * Adds a device described by @args. Requires JSON monitor.
 * Returns 0 on success -1 on error.
 */
int
qemuMonitorAddDeviceArgs(qemuMonitorPtr mon,
                         virJSONValuePtr args)
{
2988
    QEMU_CHECK_MONITOR(mon);
2989 2990 2991 2992 2993

    return qemuMonitorJSONAddDeviceArgs(mon, args);
}


2994
/**
2995
 * qemuMonitorAddObjectType:
2996 2997 2998 2999 3000 3001 3002 3003 3004
 * @mon: Pointer to monitor object
 * @type: Type name of object to add
 * @objalias: Alias of the new object
 * @props: Optional arguments for the given type. The object is consumed and
 *         should not be referenced by the caller after this function returns.
 *
 * Returns 0 on success -1 on error.
 */
int
3005 3006 3007 3008
qemuMonitorAddObjectType(qemuMonitorPtr mon,
                         const char *type,
                         const char *objalias,
                         virJSONValuePtr props)
3009
{
3010
    VIR_DEBUG("type=%s objalias=%s props=%p", type, objalias, props);
3011

3012
    QEMU_CHECK_MONITOR_GOTO(mon, error);
3013

3014
    return qemuMonitorJSONAddObject(mon, type, objalias, props);
3015 3016 3017 3018 3019

 error:
    virJSONValueFree(props);
    return -1;
}
3020 3021 3022 3023 3024 3025


int
qemuMonitorDelObject(qemuMonitorPtr mon,
                     const char *objalias)
{
3026
    VIR_DEBUG("objalias=%s", objalias);
3027

3028
    QEMU_CHECK_MONITOR(mon);
3029

3030
    return qemuMonitorJSONDelObject(mon, objalias);
3031 3032 3033
}


3034 3035 3036
int
qemuMonitorAddDrive(qemuMonitorPtr mon,
                    const char *drivestr)
3037
{
3038
    VIR_DEBUG("drive=%s", drivestr);
3039

3040
    QEMU_CHECK_MONITOR(mon);
3041

3042 3043
    /* there won't ever be a direct QMP replacement for this function */
    return qemuMonitorTextAddDrive(mon, drivestr);
3044
}
3045 3046


3047 3048 3049 3050
int
qemuMonitorSetDrivePassphrase(qemuMonitorPtr mon,
                              const char *alias,
                              const char *passphrase)
3051
{
3052
    VIR_DEBUG("alias=%s passphrase=%p(value hidden)", alias, passphrase);
3053

3054
    QEMU_CHECK_MONITOR(mon);
3055

3056
    return qemuMonitorJSONSetDrivePassphrase(mon, alias, passphrase);
3057
}
C
Chris Lalancette 已提交
3058

3059 3060 3061

int
qemuMonitorCreateSnapshot(qemuMonitorPtr mon, const char *name)
C
Chris Lalancette 已提交
3062
{
3063
    VIR_DEBUG("name=%s", name);
C
Chris Lalancette 已提交
3064

3065
    QEMU_CHECK_MONITOR(mon);
3066

3067 3068
    /* there won't ever be a direct QMP replacement for this function */
    return qemuMonitorTextCreateSnapshot(mon, name);
C
Chris Lalancette 已提交
3069 3070
}

3071 3072
int
qemuMonitorLoadSnapshot(qemuMonitorPtr mon, const char *name)
C
Chris Lalancette 已提交
3073
{
3074
    VIR_DEBUG("name=%s", name);
C
Chris Lalancette 已提交
3075

3076
    QEMU_CHECK_MONITOR(mon);
3077

3078 3079
    /* there won't ever be a direct QMP replacement for this function */
    return qemuMonitorTextLoadSnapshot(mon, name);
C
Chris Lalancette 已提交
3080 3081
}

3082 3083 3084

int
qemuMonitorDeleteSnapshot(qemuMonitorPtr mon, const char *name)
C
Chris Lalancette 已提交
3085
{
3086
    VIR_DEBUG("name=%s", name);
C
Chris Lalancette 已提交
3087

3088
    QEMU_CHECK_MONITOR(mon);
3089

3090 3091
    /* there won't ever be a direct QMP replacement for this function */
    return qemuMonitorTextDeleteSnapshot(mon, name);
C
Chris Lalancette 已提交
3092
}
3093

3094

3095 3096 3097 3098
/* Use the snapshot_blkdev command to convert the existing file for
 * device into a read-only backing file of a new qcow2 image located
 * at file.  */
int
3099
qemuMonitorDiskSnapshot(qemuMonitorPtr mon, virJSONValuePtr actions,
3100 3101
                        const char *device, const char *file,
                        const char *format, bool reuse)
3102
{
3103 3104
    VIR_DEBUG("actions=%p, device=%s, file=%s, format=%s, reuse=%d",
              actions, device, file, format, reuse);
3105

3106
    QEMU_CHECK_MONITOR(mon);
3107 3108

    return qemuMonitorJSONDiskSnapshot(mon, actions, device, file, format, reuse);
3109 3110
}

3111

3112
/* Start a drive-mirror block job.  bandwidth is in bytes/sec.  */
3113 3114 3115
int
qemuMonitorDriveMirror(qemuMonitorPtr mon,
                       const char *device, const char *file,
3116
                       const char *format, unsigned long long bandwidth,
3117
                       unsigned int granularity, unsigned long long buf_size,
3118 3119
                       unsigned int flags)
{
3120
    VIR_DEBUG("device=%s, file=%s, format=%s, bandwidth=%lld, "
3121
              "granularity=%#x, buf_size=%lld, flags=0x%x",
3122
              device, file, NULLSTR(format), bandwidth, granularity,
3123
              buf_size, flags);
3124

3125
    QEMU_CHECK_MONITOR(mon);
3126 3127 3128

    return qemuMonitorJSONDriveMirror(mon, device, file, format, bandwidth,
                                      granularity, buf_size, flags);
3129 3130
}

3131

3132 3133 3134 3135 3136 3137 3138 3139 3140 3141 3142 3143 3144 3145 3146
int
qemuMonitorBlockdevMirror(qemuMonitorPtr mon,
                          const char *jobname,
                          const char *device,
                          const char *target,
                          unsigned long long bandwidth,
                          unsigned int granularity,
                          unsigned long long buf_size,
                          unsigned int flags)
{
    VIR_DEBUG("jobname=%s, device=%s, target=%s, bandwidth=%lld, "
              "granularity=%#x, buf_size=%lld, flags=0x%x",
              NULLSTR(jobname), device, target, bandwidth, granularity,
              buf_size, flags);

3147
    QEMU_CHECK_MONITOR(mon);
3148 3149 3150 3151 3152 3153

    return qemuMonitorJSONBlockdevMirror(mon, jobname, device, target, bandwidth,
                                         granularity, buf_size, flags);
}


3154 3155
/* Use the transaction QMP command to run atomic snapshot commands.  */
int
3156
qemuMonitorTransaction(qemuMonitorPtr mon, virJSONValuePtr *actions)
3157
{
3158
    VIR_DEBUG("actions=%p", *actions);
3159

3160
    QEMU_CHECK_MONITOR(mon);
3161 3162

    return qemuMonitorJSONTransaction(mon, actions);
3163 3164
}

3165

3166
/* Start a block-commit block job.  bandwidth is in bytes/sec.  */
3167 3168 3169
int
qemuMonitorBlockCommit(qemuMonitorPtr mon, const char *device,
                       const char *top, const char *base,
3170
                       const char *backingName,
3171
                       unsigned long long bandwidth)
3172
{
3173 3174
    VIR_DEBUG("device=%s, top=%s, base=%s, backingName=%s, bandwidth=%llu",
              device, top, base, NULLSTR(backingName), bandwidth);
3175

3176
    QEMU_CHECK_MONITOR(mon);
3177 3178 3179

    return qemuMonitorJSONBlockCommit(mon, device, top, base,
                                         backingName, bandwidth);
3180 3181
}

3182 3183 3184 3185 3186

/* Probe whether active commits are supported by a given qemu binary. */
bool
qemuMonitorSupportsActiveCommit(qemuMonitorPtr mon)
{
3187
    if (!mon || !mon->json)
3188 3189
        return false;

3190
    return qemuMonitorJSONBlockCommit(mon, "bogus", NULL, NULL, NULL, 0) == -2;
3191 3192 3193
}


3194 3195 3196 3197 3198 3199 3200 3201
/* Determine the name that qemu is using for tracking the backing
 * element TARGET within the chain starting at TOP.  */
char *
qemuMonitorDiskNameLookup(qemuMonitorPtr mon,
                          const char *device,
                          virStorageSourcePtr top,
                          virStorageSourcePtr target)
{
3202
    QEMU_CHECK_MONITOR_NULL(mon);
3203 3204 3205 3206 3207

    return qemuMonitorJSONDiskNameLookup(mon, device, top, target);
}


3208
/* Use the block-job-complete monitor command to pivot a block copy job.  */
3209
int
3210 3211
qemuMonitorDrivePivot(qemuMonitorPtr mon,
                      const char *device)
3212
{
3213
    VIR_DEBUG("device=%s", device);
3214

3215
    QEMU_CHECK_MONITOR(mon);
3216 3217

    return qemuMonitorJSONDrivePivot(mon, device);
3218 3219
}

3220 3221 3222 3223 3224 3225

int
qemuMonitorArbitraryCommand(qemuMonitorPtr mon,
                            const char *cmd,
                            char **reply,
                            bool hmp)
3226
{
3227 3228 3229
    VIR_DEBUG("cmd=%s, reply=%p, hmp=%d", cmd, reply, hmp);

    QEMU_CHECK_MONITOR(mon);
3230

3231
    return qemuMonitorJSONArbitraryCommand(mon, cmd, reply, hmp);
3232
}
3233 3234


3235 3236
int
qemuMonitorInjectNMI(qemuMonitorPtr mon)
3237
{
3238
    QEMU_CHECK_MONITOR(mon);
3239

3240
    return qemuMonitorJSONInjectNMI(mon);
3241
}
3242

3243 3244 3245 3246 3247 3248

int
qemuMonitorSendKey(qemuMonitorPtr mon,
                   unsigned int holdtime,
                   unsigned int *keycodes,
                   unsigned int nkeycodes)
3249
{
3250 3251 3252
    VIR_DEBUG("holdtime=%u, nkeycodes=%u", holdtime, nkeycodes);

    QEMU_CHECK_MONITOR(mon);
3253

3254
    return qemuMonitorJSONSendKey(mon, holdtime, keycodes, nkeycodes);
3255 3256
}

3257 3258 3259

int
qemuMonitorScreendump(qemuMonitorPtr mon,
3260 3261
                      const char *device,
                      unsigned int head,
3262
                      const char *file)
3263
{
3264
    VIR_DEBUG("file=%s", file);
3265

3266
    QEMU_CHECK_MONITOR(mon);
3267

3268
    return qemuMonitorJSONScreendump(mon, device, head, file);
3269
}
3270

3271

3272
/* bandwidth is in bytes/sec */
3273
int
3274 3275 3276 3277
qemuMonitorBlockStream(qemuMonitorPtr mon,
                       const char *device,
                       const char *base,
                       const char *backingName,
3278
                       unsigned long long bandwidth)
3279
{
3280 3281
    VIR_DEBUG("device=%s, base=%s, backingName=%s, bandwidth=%lluB",
              device, NULLSTR(base), NULLSTR(backingName), bandwidth);
3282

3283
    QEMU_CHECK_MONITOR(mon);
3284

3285
    return qemuMonitorJSONBlockStream(mon, device, base, backingName, bandwidth);
3286
}
3287

3288

3289 3290
int
qemuMonitorBlockJobCancel(qemuMonitorPtr mon,
3291
                          const char *device)
3292
{
3293
    VIR_DEBUG("device=%s", device);
3294

3295
    QEMU_CHECK_MONITOR(mon);
3296

3297
    return qemuMonitorJSONBlockJobCancel(mon, device);
3298 3299 3300
}


3301 3302 3303
int
qemuMonitorBlockJobSetSpeed(qemuMonitorPtr mon,
                            const char *device,
3304
                            unsigned long long bandwidth)
3305
{
3306
    VIR_DEBUG("device=%s, bandwidth=%lluB", device, bandwidth);
3307

3308
    QEMU_CHECK_MONITOR(mon);
3309

3310
    return qemuMonitorJSONBlockJobSetSpeed(mon, device, bandwidth);
3311 3312 3313
}


3314 3315 3316
virHashTablePtr
qemuMonitorGetAllBlockJobInfo(qemuMonitorPtr mon)
{
3317
    QEMU_CHECK_MONITOR_NULL(mon);
3318 3319 3320 3321 3322 3323 3324 3325 3326
    return qemuMonitorJSONGetAllBlockJobInfo(mon);
}


/**
 * qemuMonitorGetBlockJobInfo:
 * Parse Block Job information, and populate info for the named device.
 * Return 1 if info available, 0 if device has no block job, and -1 on error.
 */
3327
int
3328 3329 3330
qemuMonitorGetBlockJobInfo(qemuMonitorPtr mon,
                           const char *alias,
                           qemuMonitorBlockJobInfoPtr info)
3331
{
3332 3333 3334
    virHashTablePtr all;
    qemuMonitorBlockJobInfoPtr data;
    int ret = 0;
3335

3336 3337 3338 3339
    VIR_DEBUG("alias=%s, info=%p", alias, info);

    if (!(all = qemuMonitorGetAllBlockJobInfo(mon)))
        return -1;
3340

3341 3342 3343 3344 3345 3346 3347
    if ((data = virHashLookup(all, alias))) {
        *info = *data;
        ret = 1;
    }

    virHashFree(all);
    return ret;
3348 3349 3350
}


3351 3352 3353 3354
int
qemuMonitorSetBlockIoThrottle(qemuMonitorPtr mon,
                              const char *device,
                              virDomainBlockIoTuneInfoPtr info,
3355
                              bool supportMaxOptions,
3356
                              bool supportGroupNameOption,
3357
                              bool supportMaxLengthOptions)
3358
{
3359 3360 3361
    VIR_DEBUG("device=%p, info=%p", device, info);

    QEMU_CHECK_MONITOR(mon);
3362

3363 3364 3365 3366
    return qemuMonitorJSONSetBlockIoThrottle(mon, device, info,
                                             supportMaxOptions,
                                             supportGroupNameOption,
                                             supportMaxLengthOptions);
3367 3368
}

3369 3370 3371 3372

int
qemuMonitorGetBlockIoThrottle(qemuMonitorPtr mon,
                              const char *device,
3373
                              virDomainBlockIoTuneInfoPtr reply)
3374
{
3375 3376 3377
    VIR_DEBUG("device=%p, reply=%p", device, reply);

    QEMU_CHECK_MONITOR(mon);
3378

3379
    return qemuMonitorJSONGetBlockIoThrottle(mon, device, reply);
3380 3381 3382
}


3383 3384
int
qemuMonitorVMStatusToPausedReason(const char *status)
3385 3386 3387 3388 3389 3390 3391
{
    int st;

    if (!status)
        return VIR_DOMAIN_PAUSED_UNKNOWN;

    if ((st = qemuMonitorVMStatusTypeFromString(status)) < 0) {
P
Philipp Hahn 已提交
3392
        VIR_WARN("QEMU reported unknown VM status: '%s'", status);
3393 3394 3395 3396 3397 3398 3399 3400 3401 3402 3403 3404 3405 3406 3407 3408 3409 3410 3411 3412 3413 3414
        return VIR_DOMAIN_PAUSED_UNKNOWN;
    }

    switch ((qemuMonitorVMStatus) st) {
    case QEMU_MONITOR_VM_STATUS_DEBUG:
    case QEMU_MONITOR_VM_STATUS_INTERNAL_ERROR:
    case QEMU_MONITOR_VM_STATUS_RESTORE_VM:
        return VIR_DOMAIN_PAUSED_UNKNOWN;

    case QEMU_MONITOR_VM_STATUS_INMIGRATE:
    case QEMU_MONITOR_VM_STATUS_POSTMIGRATE:
    case QEMU_MONITOR_VM_STATUS_FINISH_MIGRATE:
        return VIR_DOMAIN_PAUSED_MIGRATION;

    case QEMU_MONITOR_VM_STATUS_IO_ERROR:
        return VIR_DOMAIN_PAUSED_IOERROR;

    case QEMU_MONITOR_VM_STATUS_PAUSED:
    case QEMU_MONITOR_VM_STATUS_PRELAUNCH:
        return VIR_DOMAIN_PAUSED_USER;

    case QEMU_MONITOR_VM_STATUS_RUNNING:
P
Philipp Hahn 已提交
3415
        VIR_WARN("QEMU reports the guest is paused but status is 'running'");
3416 3417 3418 3419 3420 3421 3422 3423 3424 3425 3426
        return VIR_DOMAIN_PAUSED_UNKNOWN;

    case QEMU_MONITOR_VM_STATUS_SAVE_VM:
        return VIR_DOMAIN_PAUSED_SAVE;

    case QEMU_MONITOR_VM_STATUS_SHUTDOWN:
        return VIR_DOMAIN_PAUSED_SHUTTING_DOWN;

    case QEMU_MONITOR_VM_STATUS_WATCHDOG:
        return VIR_DOMAIN_PAUSED_WATCHDOG;

3427
    case QEMU_MONITOR_VM_STATUS_GUEST_PANICKED:
3428
        return VIR_DOMAIN_PAUSED_CRASHED;
3429

3430 3431 3432 3433 3434 3435
    /* unreachable from this point on */
    case QEMU_MONITOR_VM_STATUS_LAST:
        ;
    }
    return VIR_DOMAIN_PAUSED_UNKNOWN;
}
3436 3437


3438 3439 3440 3441 3442 3443
int
qemuMonitorOpenGraphics(qemuMonitorPtr mon,
                        const char *protocol,
                        int fd,
                        const char *fdname,
                        bool skipauth)
3444
{
3445 3446
    VIR_DEBUG("protocol=%s fd=%d fdname=%s skipauth=%d",
              protocol, fd, NULLSTR(fdname), skipauth);
3447 3448
    int ret;

3449
    QEMU_CHECK_MONITOR(mon);
3450 3451 3452 3453

    if (qemuMonitorSendFileHandle(mon, fdname, fd) < 0)
        return -1;

3454
    ret = qemuMonitorJSONOpenGraphics(mon, protocol, fdname, skipauth);
3455 3456 3457 3458 3459 3460 3461 3462

    if (ret < 0) {
        if (qemuMonitorCloseFileHandle(mon, fdname) < 0)
            VIR_WARN("failed to close device handle '%s'", fdname);
    }

    return ret;
}
3463

3464 3465 3466

int
qemuMonitorSystemWakeup(qemuMonitorPtr mon)
3467
{
3468
    QEMU_CHECK_MONITOR(mon);
3469 3470 3471

    return qemuMonitorJSONSystemWakeup(mon);
}
3472

3473 3474 3475 3476 3477 3478 3479

int
qemuMonitorGetVersion(qemuMonitorPtr mon,
                      int *major,
                      int *minor,
                      int *micro,
                      char **package)
3480
{
3481 3482
    VIR_DEBUG("major=%p minor=%p micro=%p package=%p",
              major, minor, micro, package);
3483

3484
    QEMU_CHECK_MONITOR(mon);
3485 3486 3487

    return qemuMonitorJSONGetVersion(mon, major, minor, micro, package);
}
3488

3489 3490 3491 3492

int
qemuMonitorGetMachines(qemuMonitorPtr mon,
                       qemuMonitorMachineInfoPtr **machines)
3493
{
3494
    VIR_DEBUG("machines=%p", machines);
3495

3496
    QEMU_CHECK_MONITOR(mon);
3497 3498 3499 3500

    return qemuMonitorJSONGetMachines(mon, machines);
}

3501 3502 3503

void
qemuMonitorMachineInfoFree(qemuMonitorMachineInfoPtr machine)
3504 3505 3506 3507 3508 3509 3510
{
    if (!machine)
        return;
    VIR_FREE(machine->name);
    VIR_FREE(machine->alias);
    VIR_FREE(machine);
}
3511

3512 3513 3514

int
qemuMonitorGetCPUDefinitions(qemuMonitorPtr mon,
3515
                             qemuMonitorCPUDefInfoPtr **cpus)
3516
{
3517
    VIR_DEBUG("cpus=%p", cpus);
3518

3519
    QEMU_CHECK_MONITOR(mon);
3520 3521 3522

    return qemuMonitorJSONGetCPUDefinitions(mon, cpus);
}
3523 3524


3525 3526 3527 3528 3529
void
qemuMonitorCPUDefInfoFree(qemuMonitorCPUDefInfoPtr cpu)
{
    if (!cpu)
        return;
3530 3531

    virStringListFree(cpu->blockers);
3532 3533 3534 3535 3536
    VIR_FREE(cpu->name);
    VIR_FREE(cpu);
}


3537 3538
int
qemuMonitorGetCPUModelExpansion(qemuMonitorPtr mon,
3539
                                qemuMonitorCPUModelExpansionType type,
3540
                                const char *model_name,
3541
                                bool migratable,
3542 3543
                                qemuMonitorCPUModelInfoPtr *model_info)
{
3544 3545
    VIR_DEBUG("type=%d model_name=%s migratable=%d",
              type, model_name, migratable);
3546

3547
    QEMU_CHECK_MONITOR(mon);
3548

3549 3550
    return qemuMonitorJSONGetCPUModelExpansion(mon, type, model_name,
                                               migratable, model_info);
3551 3552 3553 3554 3555 3556 3557 3558 3559 3560 3561
}


void
qemuMonitorCPUModelInfoFree(qemuMonitorCPUModelInfoPtr model_info)
{
    size_t i;

    if (!model_info)
        return;

3562
    for (i = 0; i < model_info->nprops; i++) {
3563
        VIR_FREE(model_info->props[i].name);
3564 3565 3566
        if (model_info->props[i].type == QEMU_MONITOR_CPU_PROPERTY_STRING)
            VIR_FREE(model_info->props[i].value.string);
    }
3567

3568
    VIR_FREE(model_info->props);
3569 3570 3571 3572 3573 3574 3575 3576 3577 3578 3579 3580 3581 3582 3583 3584 3585 3586 3587 3588
    VIR_FREE(model_info->name);
    VIR_FREE(model_info);
}


qemuMonitorCPUModelInfoPtr
qemuMonitorCPUModelInfoCopy(const qemuMonitorCPUModelInfo *orig)
{
    qemuMonitorCPUModelInfoPtr copy;
    size_t i;

    if (VIR_ALLOC(copy) < 0)
        goto error;

    if (VIR_ALLOC_N(copy->props, orig->nprops) < 0)
        goto error;

    if (VIR_STRDUP(copy->name, orig->name) < 0)
        goto error;

3589
    copy->migratability = orig->migratability;
3590 3591 3592 3593 3594 3595
    copy->nprops = orig->nprops;

    for (i = 0; i < orig->nprops; i++) {
        if (VIR_STRDUP(copy->props[i].name, orig->props[i].name) < 0)
            goto error;

3596
        copy->props[i].migratable = orig->props[i].migratable;
3597 3598 3599 3600 3601 3602 3603 3604 3605 3606 3607 3608 3609 3610 3611 3612 3613 3614 3615
        copy->props[i].type = orig->props[i].type;
        switch (orig->props[i].type) {
        case QEMU_MONITOR_CPU_PROPERTY_BOOLEAN:
            copy->props[i].value.boolean = orig->props[i].value.boolean;
            break;

        case QEMU_MONITOR_CPU_PROPERTY_STRING:
            if (VIR_STRDUP(copy->props[i].value.string,
                           orig->props[i].value.string) < 0)
                goto error;
            break;

        case QEMU_MONITOR_CPU_PROPERTY_NUMBER:
            copy->props[i].value.number = orig->props[i].value.number;
            break;

        case QEMU_MONITOR_CPU_PROPERTY_LAST:
            break;
        }
3616 3617 3618 3619 3620 3621 3622 3623 3624 3625
    }

    return copy;

 error:
    qemuMonitorCPUModelInfoFree(copy);
    return NULL;
}


3626 3627 3628
int
qemuMonitorGetCommands(qemuMonitorPtr mon,
                       char ***commands)
3629
{
3630
    VIR_DEBUG("commands=%p", commands);
3631

3632
    QEMU_CHECK_MONITOR(mon);
3633 3634 3635

    return qemuMonitorJSONGetCommands(mon, commands);
}
3636 3637


3638 3639 3640
int
qemuMonitorGetEvents(qemuMonitorPtr mon,
                     char ***events)
3641
{
3642
    VIR_DEBUG("events=%p", events);
3643

3644
    QEMU_CHECK_MONITOR(mon);
3645 3646 3647

    return qemuMonitorJSONGetEvents(mon, events);
}
3648 3649


3650 3651 3652 3653 3654
/* Collect the parameters associated with a given command line option.
 * Return count of known parameters or -1 on error.  */
int
qemuMonitorGetCommandLineOptionParameters(qemuMonitorPtr mon,
                                          const char *option,
3655 3656
                                          char ***params,
                                          bool *found)
3657
{
3658
    VIR_DEBUG("option=%s params=%p", option, params);
3659

3660
    QEMU_CHECK_MONITOR(mon);
3661

3662 3663
    return qemuMonitorJSONGetCommandLineOptionParameters(mon, option,
                                                         params, found);
3664 3665 3666
}


3667 3668 3669 3670
int
qemuMonitorGetKVMState(qemuMonitorPtr mon,
                       bool *enabled,
                       bool *present)
3671
{
3672
    VIR_DEBUG("enabled=%p present=%p", enabled, present);
3673

3674
    QEMU_CHECK_MONITOR(mon);
3675 3676 3677 3678 3679

    return qemuMonitorJSONGetKVMState(mon, enabled, present);
}


3680 3681 3682
int
qemuMonitorGetObjectTypes(qemuMonitorPtr mon,
                          char ***types)
3683
{
3684
    VIR_DEBUG("types=%p", types);
3685

3686
    QEMU_CHECK_MONITOR(mon);
3687 3688 3689

    return qemuMonitorJSONGetObjectTypes(mon, types);
}
3690 3691


3692
int
3693 3694
qemuMonitorGetDeviceProps(qemuMonitorPtr mon,
                          const char *device,
3695
                          char ***props)
3696
{
3697
    VIR_DEBUG("device=%s props=%p", device, props);
3698

3699
    QEMU_CHECK_MONITOR(mon);
3700

3701
    return qemuMonitorJSONGetDeviceProps(mon, device, props);
3702
}
3703 3704


3705 3706 3707 3708 3709 3710 3711
int
qemuMonitorGetObjectProps(qemuMonitorPtr mon,
                          const char *object,
                          char ***props)
{
    VIR_DEBUG("object=%s props=%p", object, props);

3712
    QEMU_CHECK_MONITOR(mon);
3713 3714 3715 3716 3717

    return qemuMonitorJSONGetObjectProps(mon, object, props);
}


3718 3719
char *
qemuMonitorGetTargetArch(qemuMonitorPtr mon)
3720
{
3721
    QEMU_CHECK_MONITOR_NULL(mon);
3722 3723 3724

    return qemuMonitorJSONGetTargetArch(mon);
}
3725 3726


3727 3728 3729 3730
int
qemuMonitorGetMigrationCapabilities(qemuMonitorPtr mon,
                                    char ***capabilities)
{
3731
    QEMU_CHECK_MONITOR(mon);
3732 3733 3734 3735 3736 3737 3738 3739 3740

    /* No capability is supported without JSON monitor */
    if (!mon->json)
        return 0;

    return qemuMonitorJSONGetMigrationCapabilities(mon, capabilities);
}


3741 3742 3743 3744 3745 3746 3747 3748 3749 3750
/**
 * qemuMonitorSetMigrationCapabilities:
 * @mon: Pointer to the monitor object.
 * @caps: Migration capabilities.
 *
 * The @caps object is consumed and should not be referenced by the caller
 * after this function returns.
 *
 * Returns 0 on success, -1 on error.
 */
3751 3752
int
qemuMonitorSetMigrationCapabilities(qemuMonitorPtr mon,
3753
                                    virJSONValuePtr caps)
3754
{
3755
    QEMU_CHECK_MONITOR_GOTO(mon, error);
3756

3757
    return qemuMonitorJSONSetMigrationCapabilities(mon, caps);
3758

3759 3760 3761
 error:
    virJSONValueFree(caps);
    return -1;
3762 3763 3764
}


A
Andrea Bolognani 已提交
3765 3766 3767 3768 3769 3770 3771 3772 3773 3774 3775
/**
 * qemuMonitorGetGICCapabilities:
 * @mon: QEMU monitor
 * @capabilities: where to store the GIC capabilities
 *
 * See qemuMonitorJSONGetGICCapabilities().
 */
int
qemuMonitorGetGICCapabilities(qemuMonitorPtr mon,
                              virGICCapability **capabilities)
{
3776
    QEMU_CHECK_MONITOR(mon);
A
Andrea Bolognani 已提交
3777 3778 3779 3780 3781

    return qemuMonitorJSONGetGICCapabilities(mon, capabilities);
}


3782 3783 3784
int
qemuMonitorNBDServerStart(qemuMonitorPtr mon,
                          const char *host,
3785 3786
                          unsigned int port,
                          const char *tls_alias)
3787
{
3788
    VIR_DEBUG("host=%s port=%u tls_alias=%s", host, port, NULLSTR(tls_alias));
3789

3790
    QEMU_CHECK_MONITOR(mon);
3791

3792
    return qemuMonitorJSONNBDServerStart(mon, host, port, tls_alias);
3793
}
3794

3795 3796 3797 3798 3799

int
qemuMonitorNBDServerAdd(qemuMonitorPtr mon,
                        const char *deviceID,
                        bool writable)
3800
{
3801
    VIR_DEBUG("deviceID=%s", deviceID);
3802

3803
    QEMU_CHECK_MONITOR(mon);
3804 3805 3806

    return qemuMonitorJSONNBDServerAdd(mon, deviceID, writable);
}
3807

3808 3809 3810

int
qemuMonitorNBDServerStop(qemuMonitorPtr mon)
3811
{
3812
    QEMU_CHECK_MONITOR(mon);
3813 3814 3815

    return qemuMonitorJSONNBDServerStop(mon);
}
S
Stefan Berger 已提交
3816 3817


3818 3819
int
qemuMonitorGetTPMModels(qemuMonitorPtr mon,
S
Stefan Berger 已提交
3820 3821
                            char ***tpmmodels)
{
3822
    VIR_DEBUG("tpmmodels=%p", tpmmodels);
S
Stefan Berger 已提交
3823

3824
    QEMU_CHECK_MONITOR(mon);
S
Stefan Berger 已提交
3825 3826 3827 3828 3829

    return qemuMonitorJSONGetTPMModels(mon, tpmmodels);
}


3830 3831 3832
int
qemuMonitorGetTPMTypes(qemuMonitorPtr mon,
                       char ***tpmtypes)
S
Stefan Berger 已提交
3833
{
3834
    VIR_DEBUG("tpmtypes=%p", tpmtypes);
S
Stefan Berger 已提交
3835

3836
    QEMU_CHECK_MONITOR(mon);
S
Stefan Berger 已提交
3837 3838 3839

    return qemuMonitorJSONGetTPMTypes(mon, tpmtypes);
}
3840

3841 3842 3843 3844 3845

int
qemuMonitorAttachCharDev(qemuMonitorPtr mon,
                         const char *chrID,
                         virDomainChrSourceDefPtr chr)
3846
{
3847
    VIR_DEBUG("chrID=%s chr=%p", chrID, chr);
3848

3849
    QEMU_CHECK_MONITOR(mon);
3850 3851 3852

    return qemuMonitorJSONAttachCharDev(mon, chrID, chr);
}
3853

3854 3855 3856 3857

int
qemuMonitorDetachCharDev(qemuMonitorPtr mon,
                         const char *chrID)
3858
{
3859
    VIR_DEBUG("chrID=%s", chrID);
3860

3861
    QEMU_CHECK_MONITOR(mon);
3862 3863 3864

    return qemuMonitorJSONDetachCharDev(mon, chrID);
}
3865

3866

3867 3868 3869 3870
int
qemuMonitorGetDeviceAliases(qemuMonitorPtr mon,
                            char ***aliases)
{
3871
    VIR_DEBUG("aliases=%p", aliases);
3872

3873
    QEMU_CHECK_MONITOR(mon);
3874 3875 3876

    return qemuMonitorJSONGetDeviceAliases(mon, aliases);
}
3877 3878 3879


/**
3880 3881
 * qemuMonitorSetDomainLogLocked:
 * @mon: Locked monitor object to set the log file reading on
3882 3883 3884
 * @func: the callback to report errors
 * @opaque: data to pass to @func
 * @destroy: optional callback to free @opaque
3885 3886 3887 3888
 *
 * Set the file descriptor of the open VM log file to report potential
 * early startup errors of qemu. This function requires @mon to be
 * locked already!
3889
 */
3890
void
3891 3892 3893 3894
qemuMonitorSetDomainLogLocked(qemuMonitorPtr mon,
                              qemuMonitorReportDomainLogError func,
                              void *opaque,
                              virFreeCallback destroy)
3895
{
3896 3897
    if (mon->logDestroy && mon->logOpaque)
        mon->logDestroy(mon->logOpaque);
3898

3899 3900 3901
    mon->logFunc = func;
    mon->logOpaque = opaque;
    mon->logDestroy = destroy;
3902
}
3903 3904


3905 3906 3907 3908 3909 3910 3911 3912 3913 3914 3915 3916 3917 3918 3919 3920 3921 3922 3923 3924 3925 3926 3927
/**
 * qemuMonitorSetDomainLog:
 * @mon: Unlocked monitor object to set the log file reading on
 * @func: the callback to report errors
 * @opaque: data to pass to @func
 * @destroy: optional callback to free @opaque
 *
 * Set the file descriptor of the open VM log file to report potential
 * early startup errors of qemu. This functions requires @mon to be
 * unlocked.
 */
void
qemuMonitorSetDomainLog(qemuMonitorPtr mon,
                        qemuMonitorReportDomainLogError func,
                        void *opaque,
                        virFreeCallback destroy)
{
    virObjectLock(mon);
    qemuMonitorSetDomainLogLocked(mon, func, opaque, destroy);
    virObjectUnlock(mon);
}


3928 3929 3930 3931
/**
 * qemuMonitorJSONGetGuestCPU:
 * @mon: Pointer to the monitor
 * @arch: arch of the guest
3932
 * @data: returns the cpu data
3933
 * @disabled: returns the CPU data for features which were disabled by QEMU
3934 3935 3936
 *
 * Retrieve the definition of the guest CPU from a running qemu instance.
 *
3937 3938
 * Returns 0 on success, -2 if the operation is not supported by the guest,
 * -1 on other errors.
3939
 */
3940
int
3941
qemuMonitorGetGuestCPU(qemuMonitorPtr mon,
3942
                       virArch arch,
3943 3944
                       virCPUDataPtr *data,
                       virCPUDataPtr *disabled)
3945
{
3946 3947
    VIR_DEBUG("arch=%s data=%p disabled=%p",
              virArchToString(arch), data, disabled);
3948

3949
    QEMU_CHECK_MONITOR(mon);
3950

3951
    *data = NULL;
3952 3953
    if (disabled)
        *disabled = NULL;
3954

3955
    return qemuMonitorJSONGetGuestCPU(mon, arch, data, disabled);
3956
}
3957

3958

3959 3960 3961 3962 3963 3964 3965 3966 3967 3968 3969 3970 3971 3972 3973
/**
 * qemuMonitorRTCResetReinjection:
 * @mon: Pointer to the monitor
 *
 * Issue rtc-reset-reinjection command.
 * This should be used in cases where guest time is restored via
 * guest agent, so RTC injection is not needed (in fact it would
 * confuse guest's RTC).
 *
 * Returns 0 on success
 *        -1 on error.
 */
int
qemuMonitorRTCResetReinjection(qemuMonitorPtr mon)
{
3974
    QEMU_CHECK_MONITOR(mon);
3975 3976 3977

    return qemuMonitorJSONRTCResetReinjection(mon);
}
3978

3979

3980 3981 3982 3983 3984 3985 3986 3987 3988 3989 3990 3991 3992
/**
 * qemuMonitorGetIOThreads:
 * @mon: Pointer to the monitor
 * @iothreads: Location to return array of IOThreadInfo data
 *
 * Issue query-iothreads command.
 * Retrieve the list of iothreads defined/running for the machine
 *
 * Returns count of IOThreadInfo structures on success
 *        -1 on error.
 */
int
qemuMonitorGetIOThreads(qemuMonitorPtr mon,
3993
                        qemuMonitorIOThreadInfoPtr **iothreads)
3994 3995
{

3996
    VIR_DEBUG("iothreads=%p", iothreads);
3997

3998
    QEMU_CHECK_MONITOR(mon);
3999

4000
    /* Requires JSON to make the query */
4001
    if (!mon->json) {
4002 4003
        *iothreads = NULL;
        return 0;
4004 4005 4006 4007 4008
    }

    return qemuMonitorJSONGetIOThreads(mon, iothreads);
}

4009

4010 4011 4012 4013 4014 4015 4016 4017 4018 4019 4020 4021 4022 4023 4024 4025
/**
 * qemuMonitorGetMemoryDeviceInfo:
 * @mon: pointer to the monitor
 * @info: Location to return the hash of qemuMonitorMemoryDeviceInfo
 *
 * Retrieve state and addresses of frontend memory devices present in
 * the guest.
 *
 * Returns 0 on success and fills @info with a newly allocated struct; if the
 * data can't be retrieved due to lack of support in qemu, returns -2. On
 * other errors returns -1.
 */
int
qemuMonitorGetMemoryDeviceInfo(qemuMonitorPtr mon,
                               virHashTablePtr *info)
{
4026
    VIR_DEBUG("info=%p", info);
4027 4028 4029 4030
    int ret;

    *info = NULL;

4031
    QEMU_CHECK_MONITOR(mon);
4032 4033 4034 4035 4036 4037 4038 4039 4040 4041 4042 4043 4044 4045

    if (!mon->json)
        return -2;

    if (!(*info = virHashCreate(10, virHashValueFree)))
        return -1;

    if ((ret = qemuMonitorJSONGetMemoryDeviceInfo(mon, *info)) < 0) {
        virHashFree(*info);
        *info = NULL;
    }

    return ret;
}
4046 4047 4048 4049 4050 4051 4052 4053


int
qemuMonitorMigrateIncoming(qemuMonitorPtr mon,
                           const char *uri)
{
    VIR_DEBUG("uri=%s", uri);

4054
    QEMU_CHECK_MONITOR(mon);
4055 4056 4057

    return qemuMonitorJSONMigrateIncoming(mon, uri);
}
4058 4059 4060 4061 4062


int
qemuMonitorMigrateStartPostCopy(qemuMonitorPtr mon)
{
4063
    QEMU_CHECK_MONITOR(mon);
4064 4065 4066

    return qemuMonitorJSONMigrateStartPostCopy(mon);
}
4067

4068 4069 4070 4071 4072 4073 4074

int
qemuMonitorMigrateContinue(qemuMonitorPtr mon,
                           qemuMonitorMigrationStatus status)
{
    VIR_DEBUG("status=%s", qemuMonitorMigrationStatusTypeToString(status));

4075
    QEMU_CHECK_MONITOR(mon);
4076 4077 4078 4079 4080

    return qemuMonitorJSONMigrateContinue(mon, status);
}


4081 4082 4083 4084
int
qemuMonitorGetRTCTime(qemuMonitorPtr mon,
                      struct tm *tm)
{
4085
    QEMU_CHECK_MONITOR(mon);
4086 4087 4088

    return qemuMonitorJSONGetRTCTime(mon, tm);
}
4089 4090


4091
virJSONValuePtr
4092 4093
qemuMonitorQueryQMPSchema(qemuMonitorPtr mon)
{
4094
    QEMU_CHECK_MONITOR_NULL(mon);
4095 4096 4097

    return qemuMonitorJSONQueryQMPSchema(mon);
}
4098 4099 4100 4101 4102 4103 4104


int
qemuMonitorSetBlockThreshold(qemuMonitorPtr mon,
                             const char *nodename,
                             unsigned long long threshold)
{
4105
    VIR_DEBUG("node='%s', threshold=%llu", nodename, threshold);
4106

4107
    QEMU_CHECK_MONITOR(mon);
4108 4109 4110

    return qemuMonitorJSONSetBlockThreshold(mon, nodename, threshold);
}
4111 4112 4113 4114 4115


virJSONValuePtr
qemuMonitorQueryNamedBlockNodes(qemuMonitorPtr mon)
{
4116
    QEMU_CHECK_MONITOR_NULL(mon);
4117 4118 4119

    return qemuMonitorJSONQueryNamedBlockNodes(mon);
}
4120 4121


4122 4123 4124 4125 4126 4127 4128 4129 4130 4131 4132 4133 4134 4135
char *
qemuMonitorGuestPanicEventInfoFormatMsg(qemuMonitorEventPanicInfoPtr info)
{
    char *ret = NULL;

    switch (info->type) {
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_HYPERV:
        ignore_value(virAsprintf(&ret,
                                 "hyper-v: arg1='0x%llx', arg2='0x%llx', "
                                 "arg3='0x%llx', arg4='0x%llx', arg5='0x%llx'",
                                 info->data.hyperv.arg1, info->data.hyperv.arg2,
                                 info->data.hyperv.arg3, info->data.hyperv.arg4,
                                 info->data.hyperv.arg5));
        break;
4136 4137 4138 4139 4140 4141 4142 4143
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_S390:
        ignore_value(virAsprintf(&ret, "s390: core='%d' psw-mask='0x%016llx' "
                                 "psw-addr='0x%016llx' reason='%s'",
                                 info->data.s390.core,
                                 info->data.s390.psw_mask,
                                 info->data.s390.psw_addr,
                                 info->data.s390.reason));
        break;
4144 4145 4146 4147 4148 4149 4150 4151 4152
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_NONE:
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_LAST:
        break;
    }

    return ret;
}


4153 4154 4155 4156 4157 4158
void
qemuMonitorEventPanicInfoFree(qemuMonitorEventPanicInfoPtr info)
{
    if (!info)
        return;

4159 4160 4161 4162 4163 4164 4165 4166 4167 4168
    switch (info->type) {
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_S390:
        VIR_FREE(info->data.s390.reason);
        break;
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_NONE:
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_HYPERV:
    case QEMU_MONITOR_EVENT_PANIC_INFO_TYPE_LAST:
        break;
    }

4169 4170
    VIR_FREE(info);
}
M
Michal Privoznik 已提交
4171 4172 4173 4174 4175 4176 4177 4178


int
qemuMonitorSetWatchdogAction(qemuMonitorPtr mon,
                             const char *action)
{
    VIR_DEBUG("watchdogAction=%s", action);

4179
    QEMU_CHECK_MONITOR(mon);
M
Michal Privoznik 已提交
4180 4181 4182

    return qemuMonitorJSONSetWatchdogAction(mon, action);
}
4183 4184 4185 4186 4187 4188 4189 4190 4191 4192 4193 4194 4195 4196 4197 4198 4199


/**
 * qemuMonitorBlockdevAdd:
 * @mon: monitor object
 * @props: JSON object describing the blockdev to add
 *
 * Adds a new block device (BDS) to qemu. Note that @props is always consumed
 * by this function and should not be accessed after calling this function.
 */
int
qemuMonitorBlockdevAdd(qemuMonitorPtr mon,
                       virJSONValuePtr props)
{
    VIR_DEBUG("props=%p (node-name=%s)", props,
              NULLSTR(virJSONValueObjectGetString(props, "node-name")));

4200
    QEMU_CHECK_MONITOR_GOTO(mon, error);
4201 4202 4203 4204 4205 4206 4207 4208 4209 4210 4211 4212 4213 4214 4215

    return qemuMonitorJSONBlockdevAdd(mon, props);

 error:
    virJSONValueFree(props);
    return -1;
}


int
qemuMonitorBlockdevDel(qemuMonitorPtr mon,
                       const char *nodename)
{
    VIR_DEBUG("nodename=%s", nodename);

4216
    QEMU_CHECK_MONITOR(mon);
4217 4218 4219

    return qemuMonitorJSONBlockdevDel(mon, nodename);
}