qemu_driver.c 106.0 KB
Newer Older
D
Daniel P. Berrange 已提交
1 2 3
/*
 * driver.c: core driver methods for managing qemu guests
 *
4
 * Copyright (C) 2006, 2007, 2008 Red Hat, Inc.
D
Daniel P. Berrange 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 * 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
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307  USA
 *
 * Author: Daniel P. Berrange <berrange@redhat.com>
 */

24
#include <config.h>
25

D
Daniel P. Berrange 已提交
26 27 28 29 30 31 32 33 34 35 36
#include <sys/types.h>
#include <sys/poll.h>
#include <dirent.h>
#include <limits.h>
#include <string.h>
#include <stdio.h>
#include <strings.h>
#include <stdarg.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
37
#include <sys/utsname.h>
38 39 40 41
#include <sys/stat.h>
#include <fcntl.h>
#include <signal.h>
#include <paths.h>
42 43
#include <pwd.h>
#include <stdio.h>
44
#include <sys/wait.h>
45
#include <sys/ioctl.h>
D
Daniel P. Berrange 已提交
46

47 48 49 50
#if HAVE_NUMACTL
#include <numa.h>
#endif

51 52 53 54
#if HAVE_SCHED_H
#include <sched.h>
#endif

55 56
#include "qemu_driver.h"
#include "qemu_conf.h"
J
Jim Meyering 已提交
57
#include "c-ctype.h"
58
#include "event.h"
59
#include "buf.h"
60
#include "util.h"
61
#include "nodeinfo.h"
62
#include "stats_linux.h"
63
#include "capabilities.h"
64
#include "memory.h"
65
#include "uuid.h"
66
#include "domain_conf.h"
67

R
Richard W.M. Jones 已提交
68 69 70
/* For storing short-lived temporary files. */
#define TEMPDIR LOCAL_STATE_DIR "/cache/libvirt"

71 72
static int qemudShutdown(void);

73 74 75
/* qemudDebug statements should be changed to use this macro instead. */
#define DEBUG(fmt,...) VIR_DEBUG(__FILE__, fmt, __VA_ARGS__)
#define DEBUG0(msg) VIR_DEBUG(__FILE__, "%s", msg)
76

77 78
#define qemudLog(level, msg...) fprintf(stderr, msg)

79 80 81 82 83 84 85 86 87
static int qemudSetCloseExec(int fd) {
    int flags;
    if ((flags = fcntl(fd, F_GETFD)) < 0)
        goto error;
    flags |= FD_CLOEXEC;
    if ((fcntl(fd, F_SETFD, flags)) < 0)
        goto error;
    return 0;
 error:
J
Jim Meyering 已提交
88
    qemudLog(QEMUD_ERR,
89
             "%s", _("Failed to set close-on-exec file descriptor flag\n"));
90 91 92 93 94 95 96 97 98 99 100 101 102
    return -1;
}


static int qemudSetNonBlock(int fd) {
    int flags;
    if ((flags = fcntl(fd, F_GETFL)) < 0)
        goto error;
    flags |= O_NONBLOCK;
    if ((fcntl(fd, F_SETFL, flags)) < 0)
        goto error;
    return 0;
 error:
J
Jim Meyering 已提交
103
    qemudLog(QEMUD_ERR,
104
             "%s", _("Failed to set non-blocking file descriptor flag\n"));
105 106 107 108
    return -1;
}


109 110 111 112 113 114 115 116
static void qemudDomainEventDispatch (struct qemud_driver *driver,
                                      virDomainObjPtr vm,
                                      virDomainEventType evt);

static void qemudDispatchVMEvent(int fd,
                                 int events,
                                 void *opaque);

117 118
static int qemudStartVMDaemon(virConnectPtr conn,
                              struct qemud_driver *driver,
119 120
                              virDomainObjPtr vm,
                              const char *migrateFrom);
121

122 123
static void qemudShutdownVMDaemon(virConnectPtr conn,
                                  struct qemud_driver *driver,
124
                                  virDomainObjPtr vm);
125

126
static int qemudDomainGetMaxVcpus(virDomainPtr dom);
127

128
static int qemudMonitorCommand (const struct qemud_driver *driver,
129
                                const virDomainObjPtr vm,
130 131
                                const char *cmd,
                                char **reply);
132

J
Jim Meyering 已提交
133
static struct qemud_driver *qemu_driver = NULL;
134 135


136 137 138 139 140 141 142 143
static void
qemudAutostartConfigs(struct qemud_driver *driver) {
    unsigned int i;

    for (i = 0 ; i < driver->domains.count ; i++) {
        if (driver->domains.objs[i]->autostart &&
            !virDomainIsActive(driver->domains.objs[i]) &&
            qemudStartVMDaemon(NULL, driver, driver->domains.objs[i], NULL) < 0) {
144
            virErrorPtr err = virGetLastError();
145
            qemudLog(QEMUD_ERR, _("Failed to autostart VM '%s': %s\n"),
146 147
                     driver->domains.objs[i]->def->name,
                     err ? err->message : NULL);
148 149
        }
    }
150 151
}

152 153 154 155 156 157 158
/**
 * qemudStartup:
 *
 * Initialization function for the QEmu daemon
 */
static int
qemudStartup(void) {
159 160 161
    uid_t uid = geteuid();
    struct passwd *pw;
    char *base = NULL;
D
Daniel P. Berrange 已提交
162
    char driverConf[PATH_MAX];
163

164
    if (VIR_ALLOC(qemu_driver) < 0)
165 166 167 168 169
        return -1;

    /* Don't have a dom0 so start from 1 */
    qemu_driver->nextvmid = 1;

170 171 172 173
    /* Init callback list */
    if(VIR_ALLOC(qemu_driver->domainEventCallbacks) < 0)
        return -1;

174
    if (!uid) {
175 176 177
        if (asprintf(&qemu_driver->logDir,
                     "%s/log/libvirt/qemu", LOCAL_STATE_DIR) == -1)
            goto out_of_memory;
178

D
Daniel P. Berrange 已提交
179
        if ((base = strdup (SYSCONF_DIR "/libvirt")) == NULL)
180 181 182
            goto out_of_memory;
    } else {
        if (!(pw = getpwuid(uid))) {
183
            qemudLog(QEMUD_ERR, _("Failed to find user record for uid '%d': %s\n"),
184
                     uid, strerror(errno));
185
            goto out_nouid;
186 187
        }

188 189 190
        if (asprintf(&qemu_driver->logDir,
                     "%s/.libvirt/qemu/log", pw->pw_dir) == -1)
            goto out_of_memory;
191

192
        if (asprintf (&base, "%s/.libvirt", pw->pw_dir) == -1)
193 194 195 196 197 198
            goto out_of_memory;
    }

    /* Configuration paths are either ~/.libvirt/qemu/... (session) or
     * /etc/libvirt/qemu/... (system).
     */
D
Daniel P. Berrange 已提交
199
    if (snprintf (driverConf, sizeof(driverConf), "%s/qemu.conf", base) == -1)
200
        goto out_of_memory;
D
Daniel P. Berrange 已提交
201
    driverConf[sizeof(driverConf)-1] = '\0';
202

D
Daniel P. Berrange 已提交
203
    if (asprintf (&qemu_driver->configDir, "%s/qemu", base) == -1)
204 205
        goto out_of_memory;

D
Daniel P. Berrange 已提交
206
    if (asprintf (&qemu_driver->autostartDir, "%s/qemu/autostart", base) == -1)
207 208
        goto out_of_memory;

209
    VIR_FREE(base);
210 211 212

    if ((qemu_driver->caps = qemudCapsInit()) == NULL)
        goto out_of_memory;
D
Daniel P. Berrange 已提交
213 214

    if (qemudLoadDriverConfig(qemu_driver, driverConf) < 0) {
215
        qemudShutdown();
D
Daniel P. Berrange 已提交
216 217 218
        return -1;
    }

219 220 221 222 223
    if (virDomainLoadAllConfigs(NULL,
                                qemu_driver->caps,
                                &qemu_driver->domains,
                                qemu_driver->configDir,
                                qemu_driver->autostartDir) < 0) {
D
Daniel P. Berrange 已提交
224 225 226
        qemudShutdown();
        return -1;
    }
227 228 229 230 231
    qemudAutostartConfigs(qemu_driver);

    return 0;

 out_of_memory:
J
Jim Meyering 已提交
232
    qemudLog (QEMUD_ERR,
233
              "%s", _("qemudStartup: out of memory\n"));
234
 out_nouid:
235 236
    VIR_FREE(base);
    VIR_FREE(qemu_driver);
237 238 239
    return -1;
}

240 241 242 243 244 245 246 247
/**
 * qemudReload:
 *
 * Function to restart the QEmu daemon, it will recheck the configuration
 * files and update its state and the networking
 */
static int
qemudReload(void) {
248 249 250
    if (!qemu_driver)
        return 0;

251 252 253 254 255 256
    virDomainLoadAllConfigs(NULL,
                            qemu_driver->caps,
                            &qemu_driver->domains,
                            qemu_driver->configDir,
                            qemu_driver->autostartDir);

257
    qemudAutostartConfigs(qemu_driver);
258 259

    return 0;
260 261
}

262 263 264 265 266 267 268 269 270 271
/**
 * qemudActive:
 *
 * Checks if the QEmu daemon is active, i.e. has an active domain or
 * an active network
 *
 * Returns 1 if active, 0 otherwise
 */
static int
qemudActive(void) {
272
    unsigned int i;
273

274 275 276 277 278
    if (!qemu_driver)
        return 0;

    for (i = 0 ; i < qemu_driver->domains.count ; i++)
        if (virDomainIsActive(qemu_driver->domains.objs[i]))
279 280
            return 1;

281 282 283 284
    /* Otherwise we're happy to deal with a shutdown */
    return 0;
}

285 286 287 288 289 290 291
/**
 * qemudShutdown:
 *
 * Shutdown the QEmu daemon, it will stop all active domains and networks
 */
static int
qemudShutdown(void) {
292
    unsigned int i;
293

294
    if (!qemu_driver)
295
        return -1;
296

297 298
    virCapabilitiesFree(qemu_driver->caps);

299
    /* shutdown active VMs */
300 301 302 303 304
    for (i = 0 ; i < qemu_driver->domains.count ; i++) {
        virDomainObjPtr dom = qemu_driver->domains.objs[i];
        if (virDomainIsActive(dom))
            qemudShutdownVMDaemon(NULL, qemu_driver, dom);
        if (!dom->persistent)
305
            virDomainRemoveInactive(&qemu_driver->domains,
306
                                    dom);
307
    }
308

309
    virDomainObjListFree(&qemu_driver->domains);
310

311
    VIR_FREE(qemu_driver->logDir);
312 313 314
    VIR_FREE(qemu_driver->configDir);
    VIR_FREE(qemu_driver->autostartDir);
    VIR_FREE(qemu_driver->vncTLSx509certdir);
D
Daniel P. Berrange 已提交
315

316 317 318
    /* Free domain callback list */
    virDomainEventCallbackListFree(qemu_driver->domainEventCallbacks);

319 320 321
    if (qemu_driver->brctl)
        brShutdown(qemu_driver->brctl);

322
    VIR_FREE(qemu_driver);
323 324

    return 0;
325 326 327
}

/* Return -1 for error, 1 to continue reading and 0 for success */
328 329
typedef int qemudHandlerMonitorOutput(virConnectPtr conn,
                                      struct qemud_driver *driver,
330
                                      virDomainObjPtr vm,
331 332 333 334
                                      const char *output,
                                      int fd);

static int
335 336
qemudReadMonitorOutput(virConnectPtr conn,
                       struct qemud_driver *driver,
337
                       virDomainObjPtr vm,
338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353
                       int fd,
                       char *buf,
                       int buflen,
                       qemudHandlerMonitorOutput func,
                       const char *what)
{
#define MONITOR_TIMEOUT 3000
    int got = 0;
    buf[0] = '\0';

   /* Consume & discard the initial greeting */
    while (got < (buflen-1)) {
        int ret;

        ret = read(fd, buf+got, buflen-got-1);
        if (ret == 0) {
354
            qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
355
                             _("QEMU quit during %s startup\n%s"), what, buf);
356 357 358 359 360 361 362 363
            return -1;
        }
        if (ret < 0) {
            struct pollfd pfd = { .fd = fd, .events = POLLIN };
            if (errno == EINTR)
                continue;

            if (errno != EAGAIN) {
364
                qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
365
                                 _("Failure while reading %s startup output: %s"),
366 367 368 369 370 371
                                 what, strerror(errno));
                return -1;
            }

            ret = poll(&pfd, 1, MONITOR_TIMEOUT);
            if (ret == 0) {
372
                qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
373
                                 _("Timed out while reading %s startup output"), what);
374 375 376
                return -1;
            } else if (ret == -1) {
                if (errno != EINTR) {
377
                    qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
378
                                     _("Failure while reading %s startup output: %s"),
379 380 381 382 383 384 385 386 387
                                     what, strerror(errno));
                    return -1;
                }
            } else {
                /* Make sure we continue loop & read any further data
                   available before dealing with EOF */
                if (pfd.revents & (POLLIN | POLLHUP))
                    continue;

388
                qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
389
                                 _("Failure while reading %s startup output"), what);
390 391 392 393 394
                return -1;
            }
        } else {
            got += ret;
            buf[got] = '\0';
395
            if ((ret = func(conn, driver, vm, buf, fd)) != 1)
396 397 398 399
                return ret;
        }
    }

400
    qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
401
                     _("Out of space while reading %s startup output"), what);
402 403 404 405 406 407
    return -1;

#undef MONITOR_TIMEOUT
}

static int
408 409
qemudCheckMonitorPrompt(virConnectPtr conn ATTRIBUTE_UNUSED,
                        struct qemud_driver *driver ATTRIBUTE_UNUSED,
410
                        virDomainObjPtr vm,
411 412 413 414 415 416 417 418 419 420 421
                        const char *output,
                        int fd)
{
    if (strstr(output, "(qemu) ") == NULL)
        return 1; /* keep reading */

    vm->monitor = fd;

    return 0;
}

422 423
static int qemudOpenMonitor(virConnectPtr conn,
                            struct qemud_driver *driver,
424
                            virDomainObjPtr vm,
425
                            const char *monitor) {
426 427 428 429
    int monfd;
    char buf[1024];
    int ret = -1;

D
Daniel Veillard 已提交
430
    if ((monfd = open(monitor, O_RDWR)) < 0) {
431
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
432
                         _("Unable to open monitor path %s"), monitor);
433 434 435
        return -1;
    }
    if (qemudSetCloseExec(monfd) < 0) {
436
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
437
                         "%s", _("Unable to set monitor close-on-exec flag"));
438 439 440
        goto error;
    }
    if (qemudSetNonBlock(monfd) < 0) {
441
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
442
                         "%s", _("Unable to put monitor into non-blocking mode"));
443 444 445
        goto error;
    }

446 447
    ret = qemudReadMonitorOutput(conn,
                                 driver, vm, monfd,
448 449 450
                                 buf, sizeof(buf),
                                 qemudCheckMonitorPrompt,
                                 "monitor");
451 452 453 454 455

    /* Keep monitor open upon success */
    if (ret == 0)
        return ret;

456 457 458 459 460
 error:
    close(monfd);
    return ret;
}

461 462
static int qemudExtractMonitorPath(virConnectPtr conn,
                                   const char *haystack,
463
                                   size_t *offset,
464
                                   char **path) {
465
    static const char needle[] = "char device redirected to";
466
    char *tmp, *dev;
467

468
    VIR_FREE(*path);
469
    /* First look for our magic string */
470 471 472 473 474
    if (!(tmp = strstr(haystack + *offset, needle))) {
        return 1;
    }
    tmp += sizeof(needle);
    dev = tmp;
475

476 477 478 479 480
    /*
     * And look for first whitespace character and nul terminate
     * to mark end of the pty path
     */
    while (*tmp) {
481
        if (c_isspace(*tmp)) {
482 483 484 485 486 487 488
            if (VIR_ALLOC_N(*path, (tmp-dev)+1) < 0) {
                qemudReportError(conn, NULL, NULL,
                                 VIR_ERR_NO_MEMORY, NULL);
                return -1;
            }
            strncpy(*path, dev, (tmp-dev));
            (*path)[(tmp-dev)] = '\0';
489
            /* ... now further update offset till we get EOL */
490
            *offset = tmp - haystack;
491 492
            return 0;
        }
493
        tmp++;
494 495 496 497 498
    }

    /*
     * We found a path, but didn't find any whitespace,
     * so it must be still incomplete - we should at
499 500
     * least see a \n - indicate that we want to carry
     * on trying again
501
     */
502
    return 1;
503 504 505
}

static int
506 507
qemudFindCharDevicePTYs(virConnectPtr conn,
                        struct qemud_driver *driver,
508
                        virDomainObjPtr vm,
509 510
                        const char *output,
                        int fd ATTRIBUTE_UNUSED)
511
{
512
    char *monitor = NULL;
513
    size_t offset = 0;
514
    int ret, i;
515 516 517 518 519

    /* The order in which QEMU prints out the PTY paths is
       the order in which it procsses its monitor, serial
       and parallel device args. This code must match that
       ordering.... */
520

521
    /* So first comes the monitor device */
522 523
    if ((ret = qemudExtractMonitorPath(conn, output, &offset, &monitor)) != 0)
        goto cleanup;
524

525
    /* then the serial devices */
526 527
    for (i = 0 ; i < vm->def->nserials ; i++) {
        virDomainChrDefPtr chr = vm->def->serials[i];
528 529 530 531
        if (chr->type == VIR_DOMAIN_CHR_TYPE_PTY) {
            if ((ret = qemudExtractMonitorPath(conn, output, &offset,
                                               &chr->data.file.path)) != 0)
                goto cleanup;
532 533 534 535
        }
    }

    /* and finally the parallel devices */
536 537
    for (i = 0 ; i < vm->def->nparallels ; i++) {
        virDomainChrDefPtr chr = vm->def->parallels[i];
538 539 540 541
        if (chr->type == VIR_DOMAIN_CHR_TYPE_PTY) {
            if ((ret = qemudExtractMonitorPath(conn, output, &offset,
                                               &chr->data.file.path)) != 0)
                goto cleanup;
542 543 544 545
        }
    }

    /* Got them all, so now open the monitor console */
546 547 548 549 550
    ret = qemudOpenMonitor(conn, driver, vm, monitor);

cleanup:
    VIR_FREE(monitor);
    return ret;
551 552
}

553 554
static int qemudWaitForMonitor(virConnectPtr conn,
                               struct qemud_driver *driver,
555
                               virDomainObjPtr vm) {
556
    char buf[1024]; /* Plenty of space to get startup greeting */
557
    int ret = qemudReadMonitorOutput(conn,
558
                                     driver, vm, vm->stderr_fd,
559
                                     buf, sizeof(buf),
560
                                     qemudFindCharDevicePTYs,
561 562 563
                                     "console");

    buf[sizeof(buf)-1] = '\0';
564 565

    if (safewrite(vm->logfile, buf, strlen(buf)) < 0) {
566
        /* Log, but ignore failures to write logfile for VM */
567
        qemudLog(QEMUD_WARN, _("Unable to log VM console data: %s\n"),
568 569 570 571 572
                 strerror(errno));
    }
    return ret;
}

573 574 575
static int
qemudDetectVcpuPIDs(virConnectPtr conn,
                    struct qemud_driver *driver,
576
                    virDomainObjPtr vm) {
577 578 579 580 581 582
    char *qemucpus = NULL;
    char *line;
    int lastVcpu = -1;

    /* Only KVM has seperate threads for CPUs,
       others just use main QEMU process for CPU */
583
    if (vm->def->virtType != VIR_DOMAIN_VIRT_KVM)
584 585 586 587 588 589 590 591 592 593
        vm->nvcpupids = 1;
    else
        vm->nvcpupids = vm->def->vcpus;

    if (VIR_ALLOC_N(vm->vcpupids, vm->nvcpupids) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_MEMORY,
                         "%s", _("allocate cpumap"));
        return -1;
    }

594
    if (vm->def->virtType != VIR_DOMAIN_VIRT_KVM) {
595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657
        vm->vcpupids[0] = vm->pid;
        return 0;
    }

    if (qemudMonitorCommand(driver, vm, "info cpus", &qemucpus) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot run monitor command to fetch CPU thread info"));
        VIR_FREE(vm->vcpupids);
        vm->nvcpupids = 0;
        return -1;
    }

    /*
     * This is the gross format we're about to parse :-{
     *
     * (qemu) info cpus
     * * CPU #0: pc=0x00000000000f0c4a thread_id=30019
     *   CPU #1: pc=0x00000000fffffff0 thread_id=30020
     *   CPU #2: pc=0x00000000fffffff0 thread_id=30021
     *
     */
    line = qemucpus;
    do {
        char *offset = strchr(line, '#');
        char *end = NULL;
        int vcpu = 0, tid = 0;

        /* See if we're all done */
        if (offset == NULL)
            break;

        /* Extract VCPU number */
        if (virStrToLong_i(offset + 1, &end, 10, &vcpu) < 0)
            goto error;
        if (end == NULL || *end != ':')
            goto error;

        /* Extract host Thread ID */
        if ((offset = strstr(line, "thread_id=")) == NULL)
            goto error;
        if (virStrToLong_i(offset + strlen("thread_id="), &end, 10, &tid) < 0)
            goto error;
        if (end == NULL || !c_isspace(*end))
            goto error;

        /* Validate the VCPU is in expected range & order */
        if (vcpu > vm->nvcpupids ||
            vcpu != (lastVcpu + 1))
            goto error;

        lastVcpu = vcpu;
        vm->vcpupids[vcpu] = tid;

        /* Skip to next data line */
        line = strchr(offset, '\r');
        if (line == NULL)
            line = strchr(offset, '\n');
    } while (line != NULL);

    /* Validate we got data for all VCPUs we expected */
    if (lastVcpu != (vm->def->vcpus - 1))
        goto error;

658
    VIR_FREE(qemucpus);
659 660 661 662
    return 0;

error:
    VIR_FREE(vm->vcpupids);
663 664
    vm->nvcpupids = 0;
    VIR_FREE(qemucpus);
665 666 667 668 669 670 671 672

    /* Explicitly return success, not error. Older KVM does
       not have vCPU -> Thread mapping info and we don't
       want to break its use. This merely disables ability
       to pin vCPUS with libvirt */
    return 0;
}

673 674 675
static int
qemudInitCpus(virConnectPtr conn,
              struct qemud_driver *driver,
676
              virDomainObjPtr vm) {
677 678 679 680 681 682 683 684 685 686 687 688 689 690 691
    char *info = NULL;
#if HAVE_SCHED_GETAFFINITY
    cpu_set_t mask;
    int i, maxcpu = QEMUD_CPUMASK_LEN;
    virNodeInfo nodeinfo;

    if (virNodeInfoPopulate(conn, &nodeinfo) < 0)
        return -1;

    /* setaffinity fails if you set bits for CPUs which
     * aren't present, so we have to limit ourselves */
    if (maxcpu > nodeinfo.cpus)
        maxcpu = nodeinfo.cpus;

    CPU_ZERO(&mask);
D
Daniel P. Berrange 已提交
692 693 694 695 696 697
    if (vm->def->cpumask) {
        for (i = 0 ; i < maxcpu ; i++)
            if (vm->def->cpumask[i])
                CPU_SET(i, &mask);
    } else {
        for (i = 0 ; i < maxcpu ; i++)
698
            CPU_SET(i, &mask);
D
Daniel P. Berrange 已提交
699
    }
700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717

    for (i = 0 ; i < vm->nvcpupids ; i++) {
        if (sched_setaffinity(vm->vcpupids[i],
                              sizeof(mask), &mask) < 0) {
            qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                             _("failed to set CPU affinity %s"),
                             strerror(errno));
            return -1;
        }
    }
#endif /* HAVE_SCHED_GETAFFINITY */

    /* Allow the CPUS to start executing */
    if (qemudMonitorCommand(driver, vm, "cont", &info) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("resume operation failed"));
        return -1;
    }
718
    VIR_FREE(info);
719 720 721 722 723

    return 0;
}


724
static int qemudNextFreeVNCPort(struct qemud_driver *driver ATTRIBUTE_UNUSED) {
725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759
    int i;

    for (i = 5900 ; i < 6000 ; i++) {
        int fd;
        int reuse = 1;
        struct sockaddr_in addr;
        addr.sin_family = AF_INET;
        addr.sin_port = htons(i);
        addr.sin_addr.s_addr = htonl(INADDR_ANY);
        fd = socket(PF_INET, SOCK_STREAM, 0);
        if (fd < 0)
            return -1;

        if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (void*)&reuse, sizeof(reuse)) < 0) {
            close(fd);
            break;
        }

        if (bind(fd, (struct sockaddr*)&addr, sizeof(addr)) == 0) {
            /* Not in use, lets grab it */
            close(fd);
            return i;
        }
        close(fd);

        if (errno == EADDRINUSE) {
            /* In use, try next */
            continue;
        }
        /* Some other bad failure, get out.. */
        break;
    }
    return -1;
}

760 761 762
static virDomainPtr qemudDomainLookupByName(virConnectPtr conn,
                                            const char *name);

763 764
static int qemudStartVMDaemon(virConnectPtr conn,
                              struct qemud_driver *driver,
765 766
                              virDomainObjPtr vm,
                              const char *migrateFrom) {
767
    const char **argv = NULL, **tmp;
768
    const char **progenv = NULL;
769
    int i, ret;
770
    char logfile[PATH_MAX];
771
    struct stat sb;
772 773
    int *tapfds = NULL;
    int ntapfds = 0;
774
    unsigned int qemuCmdFlags;
775
    fd_set keepfd;
776
    const char *emulator;
777 778

    FD_ZERO(&keepfd);
779

780
    if (virDomainIsActive(vm)) {
781
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
782
                         "%s", _("VM is already active"));
783 784 785
        return -1;
    }

786 787 788
    if (vm->def->graphics &&
        vm->def->graphics->type == VIR_DOMAIN_GRAPHICS_TYPE_VNC &&
        vm->def->graphics->data.vnc.autoport) {
789
        int port = qemudNextFreeVNCPort(driver);
790
        if (port < 0) {
791
            qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
792
                             "%s", _("Unable to find an unused VNC port"));
793 794
            return -1;
        }
795 796
        vm->def->graphics->data.vnc.port = port;
    }
797

798
    if ((strlen(driver->logDir) + /* path */
799 800 801 802
         1 + /* Separator */
         strlen(vm->def->name) + /* basename */
         4 + /* suffix .log */
         1 /* NULL */) > PATH_MAX) {
803
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
804
                         _("config file path too long: %s/%s.log"),
805
                         driver->logDir, vm->def->name);
806 807
        return -1;
    }
808
    strcpy(logfile, driver->logDir);
809 810 811 812
    strcat(logfile, "/");
    strcat(logfile, vm->def->name);
    strcat(logfile, ".log");

813
    if (virFileMakePath(driver->logDir) < 0) {
814
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
815
                         _("cannot create log directory %s: %s"),
816
                         driver->logDir, strerror(errno));
817 818 819 820 821
        return -1;
    }

    if ((vm->logfile = open(logfile, O_CREAT | O_TRUNC | O_WRONLY,
                            S_IRUSR | S_IWUSR)) < 0) {
822
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
823
                         _("failed to create logfile %s: %s"),
824 825 826
                         logfile, strerror(errno));
        return -1;
    }
827 828
    if (qemudSetCloseExec(vm->logfile) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
829
                         _("Unable to set VM logfile close-on-exec flag %s"),
830 831 832 833 834
                         strerror(errno));
        close(vm->logfile);
        vm->logfile = -1;
        return -1;
    }
835

836 837 838 839 840 841
    emulator = vm->def->emulator;
    if (!emulator)
        emulator = virDomainDefDefaultEmulator(conn, vm->def, driver->caps);
    if (!emulator)
        return -1;

842 843 844 845
    /* Make sure the binary we are about to try exec'ing exists.
     * Technically we could catch the exec() failure, but that's
     * in a sub-process so its hard to feed back a useful error
     */
846
    if (stat(emulator, &sb) < 0) {
847 848
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("Cannot find QEMU binary %s: %s"),
849
                         emulator,
850 851 852 853
                         strerror(errno));
        return -1;
    }

854
    if (qemudExtractVersionInfo(emulator,
855 856 857 858
                                NULL,
                                &qemuCmdFlags) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("Cannot determine QEMU argv syntax %s"),
859
                         emulator);
860 861
        return -1;
    }
862

863
    if (qemudBuildCommandLine(conn, driver, vm,
864
                              qemuCmdFlags, &argv, &progenv,
865
                              &tapfds, &ntapfds, migrateFrom) < 0) {
866 867 868 869 870
        close(vm->logfile);
        vm->logfile = -1;
        return -1;
    }

871 872 873 874 875 876 877 878 879 880
    tmp = progenv;
    while (*tmp) {
        if (safewrite(vm->logfile, *tmp, strlen(*tmp)) < 0)
            qemudLog(QEMUD_WARN, _("Unable to write envv to logfile %d: %s\n"),
                     errno, strerror(errno));
        if (safewrite(vm->logfile, " ", 1) < 0)
            qemudLog(QEMUD_WARN, _("Unable to write envv to logfile %d: %s\n"),
                     errno, strerror(errno));
        tmp++;
    }
881 882
    tmp = argv;
    while (*tmp) {
883
        if (safewrite(vm->logfile, *tmp, strlen(*tmp)) < 0)
884
            qemudLog(QEMUD_WARN, _("Unable to write argv to logfile %d: %s\n"),
885
                     errno, strerror(errno));
886
        if (safewrite(vm->logfile, " ", 1) < 0)
887
            qemudLog(QEMUD_WARN, _("Unable to write argv to logfile %d: %s\n"),
888 889 890
                     errno, strerror(errno));
        tmp++;
    }
891
    if (safewrite(vm->logfile, "\n", 1) < 0)
892
        qemudLog(QEMUD_WARN, _("Unable to write argv to logfile %d: %s\n"),
893 894
                 errno, strerror(errno));

895 896 897
    vm->stdout_fd = -1;
    vm->stderr_fd = -1;

898 899 900
    for (i = 0 ; i < ntapfds ; i++)
        FD_SET(tapfds[i], &keepfd);

901
    ret = virExec(conn, argv, progenv, &keepfd, &vm->pid,
902 903
                  vm->stdin_fd, &vm->stdout_fd, &vm->stderr_fd,
                  VIR_EXEC_NONBLOCK);
904
    if (ret == 0) {
905 906
        vm->def->id = driver->nextvmid++;
        vm->state = migrateFrom ? VIR_DOMAIN_PAUSED : VIR_DOMAIN_RUNNING;
907 908
    }

909
    for (i = 0 ; argv[i] ; i++)
910 911
        VIR_FREE(argv[i]);
    VIR_FREE(argv);
912

913 914 915 916
    for (i = 0 ; progenv[i] ; i++)
        VIR_FREE(progenv[i]);
    VIR_FREE(progenv);

917 918 919
    if (tapfds) {
        for (i = 0 ; i < ntapfds ; i++) {
            close(tapfds[i]);
920
        }
921
        VIR_FREE(tapfds);
922 923
    }

924
    if (ret == 0) {
925
        if ((virEventAddHandle(vm->stdout_fd,
926 927 928
                               VIR_EVENT_HANDLE_READABLE |
                                   VIR_EVENT_HANDLE_ERROR |
                                   VIR_EVENT_HANDLE_HANGUP,
929 930
                               qemudDispatchVMEvent,
                               driver) < 0) ||
931
            (virEventAddHandle(vm->stderr_fd,
932 933 934
                               VIR_EVENT_HANDLE_READABLE |
                                   VIR_EVENT_HANDLE_ERROR |
                                   VIR_EVENT_HANDLE_HANGUP,
935 936 937 938 939
                               qemudDispatchVMEvent,
                               driver) < 0) ||
            (qemudWaitForMonitor(conn, driver, vm) < 0) ||
            (qemudDetectVcpuPIDs(conn, driver, vm) < 0) ||
            (qemudInitCpus(conn, driver, vm) < 0)) {
940 941 942
            qemudShutdownVMDaemon(conn, driver, vm);
            return -1;
        }
943
        qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_STARTED);
944 945
    }

946
    return ret;
947 948
}

949
static int qemudVMData(struct qemud_driver *driver ATTRIBUTE_UNUSED,
950
                       virDomainObjPtr vm, int fd) {
951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966
    char buf[4096];
    if (vm->pid < 0)
        return 0;

    for (;;) {
        int ret = read(fd, buf, sizeof(buf)-1);
        if (ret < 0) {
            if (errno == EAGAIN)
                return 0;
            return -1;
        }
        if (ret == 0) {
            return 0;
        }
        buf[ret] = '\0';

967
        if (safewrite(vm->logfile, buf, ret) < 0) {
968
            /* Log, but ignore failures to write logfile for VM */
969
            qemudLog(QEMUD_WARN, _("Unable to log VM console data: %s\n"),
970 971 972 973 974 975
                     strerror(errno));
        }
    }
}


976
static void qemudShutdownVMDaemon(virConnectPtr conn ATTRIBUTE_UNUSED,
977 978
                                  struct qemud_driver *driver, virDomainObjPtr vm) {
    if (!virDomainIsActive(vm))
979
        return;
980

981
    qemudLog(QEMUD_INFO, _("Shutting down VM '%s'\n"), vm->def->name);
982 983 984

    kill(vm->pid, SIGTERM);

985 986
    qemudVMData(driver, vm, vm->stdout_fd);
    qemudVMData(driver, vm, vm->stderr_fd);
987

988 989
    virEventRemoveHandle(vm->stdout_fd);
    virEventRemoveHandle(vm->stderr_fd);
990 991

    if (close(vm->logfile) < 0)
992
        qemudLog(QEMUD_WARN, _("Unable to close logfile %d: %s\n"),
993
                 errno, strerror(errno));
994 995
    close(vm->stdout_fd);
    close(vm->stderr_fd);
996 997 998
    if (vm->monitor != -1)
        close(vm->monitor);
    vm->logfile = -1;
999 1000
    vm->stdout_fd = -1;
    vm->stderr_fd = -1;
1001 1002 1003 1004 1005
    vm->monitor = -1;

    if (waitpid(vm->pid, NULL, WNOHANG) != vm->pid) {
        kill(vm->pid, SIGKILL);
        if (waitpid(vm->pid, NULL, 0) != vm->pid) {
J
Jim Meyering 已提交
1006
            qemudLog(QEMUD_WARN,
1007
                     "%s", _("Got unexpected pid, damn\n"));
1008 1009 1010 1011
        }
    }

    vm->pid = -1;
1012
    vm->def->id = -1;
1013
    vm->state = VIR_DOMAIN_SHUTOFF;
1014
    VIR_FREE(vm->vcpupids);
1015
    vm->nvcpupids = 0;
1016 1017

    if (vm->newDef) {
1018
        virDomainDefFree(vm->def);
1019
        vm->def = vm->newDef;
1020
        vm->def->id = -1;
1021 1022 1023 1024
        vm->newDef = NULL;
    }
}

1025
static int qemudDispatchVMLog(struct qemud_driver *driver, virDomainObjPtr vm, int fd) {
1026 1027
    if (qemudVMData(driver, vm, fd) < 0) {
        qemudShutdownVMDaemon(NULL, driver, vm);
1028
        if (!vm->persistent)
1029 1030
            virDomainRemoveInactive(&driver->domains,
                                    vm);
1031
    }
1032 1033 1034
    return 0;
}

1035
static int qemudDispatchVMFailure(struct qemud_driver *driver, virDomainObjPtr vm,
1036
                                  int fd ATTRIBUTE_UNUSED) {
1037
    qemudShutdownVMDaemon(NULL, driver, vm);
1038
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_STOPPED);
1039
    if (!vm->persistent)
1040 1041
        virDomainRemoveInactive(&driver->domains,
                                vm);
1042 1043 1044 1045
    return 0;
}


1046 1047
static void
qemudDispatchVMEvent(int fd, int events, void *opaque) {
1048
    struct qemud_driver *driver = (struct qemud_driver *)opaque;
1049 1050 1051 1052 1053 1054 1055 1056
    virDomainObjPtr vm = NULL;
    unsigned int i;

    for (i = 0 ; i < driver->domains.count ; i++) {
        if (virDomainIsActive(driver->domains.objs[i]) &&
            (driver->domains.objs[i]->stdout_fd == fd ||
             driver->domains.objs[i]->stderr_fd == fd)) {
            vm = driver->domains.objs[i];
1057
            break;
1058
        }
1059 1060 1061 1062 1063
    }

    if (!vm)
        return;

1064
    if (events == VIR_EVENT_HANDLE_READABLE)
1065
        qemudDispatchVMLog(driver, vm, fd);
1066
    else
1067
        qemudDispatchVMFailure(driver, vm, fd);
1068 1069
}

1070 1071
static int
qemudMonitorCommand (const struct qemud_driver *driver ATTRIBUTE_UNUSED,
1072
                     const virDomainObjPtr vm,
1073 1074
                     const char *cmd,
                     char **reply) {
D
Daniel P. Berrange 已提交
1075 1076
    int size = 0;
    char *buf = NULL;
1077
    size_t cmdlen = strlen(cmd);
D
Daniel P. Berrange 已提交
1078

1079 1080 1081
    if (safewrite(vm->monitor, cmd, cmdlen) != cmdlen)
        return -1;
    if (safewrite(vm->monitor, "\r", 1) != 1)
D
Daniel P. Berrange 已提交
1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093
        return -1;

    *reply = NULL;

    for (;;) {
        struct pollfd fd = { vm->monitor, POLLIN | POLLERR | POLLHUP, 0 };
        char *tmp;

        /* Read all the data QEMU has sent thus far */
        for (;;) {
            char data[1024];
            int got = read(vm->monitor, data, sizeof(data));
D
Daniel P. Berrange 已提交
1094

1095 1096
            if (got == 0)
                goto error;
D
Daniel P. Berrange 已提交
1097 1098 1099 1100 1101
            if (got < 0) {
                if (errno == EINTR)
                    continue;
                if (errno == EAGAIN)
                    break;
1102
                goto error;
1103
            }
1104
            if (VIR_REALLOC_N(buf, size+got+1) < 0)
1105 1106
                goto error;

D
Daniel P. Berrange 已提交
1107 1108 1109 1110
            memmove(buf+size, data, got);
            buf[size+got] = '\0';
            size += got;
        }
1111

D
Daniel P. Berrange 已提交
1112
        /* Look for QEMU prompt to indicate completion */
D
Daniel P. Berrange 已提交
1113
        if (buf && ((tmp = strstr(buf, "\n(qemu) ")) != NULL)) {
D
Daniel P. Berrange 已提交
1114 1115 1116 1117 1118 1119 1120 1121
            tmp[0] = '\0';
            break;
        }
    pollagain:
        /* Need to wait for more data */
        if (poll(&fd, 1, -1) < 0) {
            if (errno == EINTR)
                goto pollagain;
1122
            goto error;
D
Daniel P. Berrange 已提交
1123 1124 1125
        }
    }

1126 1127
    /* Log, but ignore failures to write logfile for VM */
    if (safewrite(vm->logfile, buf, strlen(buf)) < 0)
1128
        qemudLog(QEMUD_WARN, _("Unable to log VM console data: %s\n"),
1129 1130
                 strerror(errno));

D
Daniel P. Berrange 已提交
1131 1132
    *reply = buf;
    return 0;
1133 1134 1135 1136 1137

 error:
    if (buf) {
        /* Log, but ignore failures to write logfile for VM */
        if (safewrite(vm->logfile, buf, strlen(buf)) < 0)
1138
            qemudLog(QEMUD_WARN, _("Unable to log VM console data: %s\n"),
1139
                     strerror(errno));
1140
        VIR_FREE(buf);
1141 1142
    }
    return -1;
D
Daniel P. Berrange 已提交
1143 1144
}

1145 1146 1147 1148 1149 1150 1151 1152 1153 1154
/**
 * qemudProbe:
 *
 * Probe for the availability of the qemu driver, assume the
 * presence of QEmu emulation if the binaries are installed
 */
static const char *qemudProbe(void)
{
    if ((virFileExists("/usr/bin/qemu")) ||
        (virFileExists("/usr/bin/qemu-kvm")) ||
1155
        (virFileExists("/usr/bin/xenner"))) {
1156
        if (getuid() == 0) {
1157 1158 1159 1160
            return("qemu:///system");
        } else {
            return("qemu:///session");
        }
1161 1162 1163
    }
    return(NULL);
}
1164

1165
static virDrvOpenStatus qemudOpen(virConnectPtr conn,
1166
                                  xmlURIPtr uri,
1167
                                  virConnectAuthPtr auth ATTRIBUTE_UNUSED,
1168
                                  int flags ATTRIBUTE_UNUSED) {
1169 1170 1171
    uid_t uid = getuid();

    if (qemu_driver == NULL)
1172
        goto decline;
1173

1174 1175 1176 1177 1178 1179
    if (uri == NULL || uri->scheme == NULL || uri->path == NULL)
        goto decline;

    if (STRNEQ (uri->scheme, "qemu"))
        goto decline;

1180
    if (uid != 0) {
1181 1182
        if (STRNEQ (uri->path, "/session"))
            goto decline;
1183
    } else { /* root */
1184 1185
        if (STRNEQ (uri->path, "/system") &&
            STRNEQ (uri->path, "/session"))
1186
            goto decline;
1187 1188 1189 1190 1191
    }

    conn->privateData = qemu_driver;

    return VIR_DRV_OPEN_SUCCESS;
1192 1193

 decline:
1194
    return VIR_DRV_OPEN_DECLINED;
1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205
}

static int qemudClose(virConnectPtr conn) {
    /*struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;*/

    conn->privateData = NULL;

    return 0;
}

static const char *qemudGetType(virConnectPtr conn ATTRIBUTE_UNUSED) {
1206
    return "QEMU";
1207 1208
}

1209 1210 1211 1212 1213

static int kvmGetMaxVCPUs(void) {
    int maxvcpus = 1;

    int r, fd;
1214

1215 1216
    fd = open(KVM_DEVICE, O_RDONLY);
    if (fd < 0) {
1217
        qemudLog(QEMUD_WARN, _("Unable to open %s: %s\n"), KVM_DEVICE, strerror(errno));
1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229
        return maxvcpus;
    }

    r = ioctl(fd, KVM_CHECK_EXTENSION, KVM_CAP_NR_VCPUS);
    if (r > 0)
        maxvcpus = r;

    close(fd);
    return maxvcpus;
}


1230
static int qemudGetMaxVCPUs(virConnectPtr conn, const char *type) {
1231 1232 1233
    if (!type)
        return 16;

1234
    if (STRCASEEQ(type, "qemu"))
1235 1236 1237 1238
        return 16;

    /* XXX future KVM will support SMP. Need to probe
       kernel to figure out KVM module version i guess */
1239
    if (STRCASEEQ(type, "kvm"))
1240
        return kvmGetMaxVCPUs();
1241

1242
    if (STRCASEEQ(type, "kqemu"))
1243
        return 1;
1244 1245 1246

    qemudReportError(conn, NULL, NULL, VIR_ERR_INVALID_ARG,
                     _("unknown type '%s'"), type);
1247 1248 1249
    return -1;
}

1250 1251 1252
static int qemudGetNodeInfo(virConnectPtr conn,
                            virNodeInfoPtr nodeinfo) {
    return virNodeInfoPopulate(conn, nodeinfo);
1253 1254 1255
}


1256 1257 1258
static char *qemudGetCapabilities(virConnectPtr conn) {
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
    char *xml;
1259

1260
    if ((xml = virCapabilitiesFormatXML(driver->caps)) == NULL) {
1261 1262
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_MEMORY,
                 "%s", _("failed to allocate space for capabilities support"));
1263 1264 1265
        return NULL;
    }

1266
    return xml;
1267 1268 1269
}


1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324
#if HAVE_NUMACTL
static int
qemudNodeGetCellsFreeMemory(virConnectPtr conn,
                            unsigned long long *freeMems,
                            int startCell,
                            int maxCells)
{
    int n, lastCell, numCells;

    if (numa_available() < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_SUPPORT,
                         "%s", _("NUMA not supported on this host"));
        return -1;
    }
    lastCell = startCell + maxCells - 1;
    if (lastCell > numa_max_node())
        lastCell = numa_max_node();

    for (numCells = 0, n = startCell ; n <= lastCell ; n++) {
        long long mem;
        if (numa_node_size64(n, &mem) < 0) {
            qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                             "%s", _("Failed to query NUMA free memory"));
            return -1;
        }
        freeMems[numCells++] = mem;
    }
    return numCells;
}

static unsigned long long
qemudNodeGetFreeMemory (virConnectPtr conn)
{
    unsigned long long freeMem = 0;
    int n;
    if (numa_available() < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_SUPPORT,
                         "%s", _("NUMA not supported on this host"));
        return -1;
    }

    for (n = 0 ; n <= numa_max_node() ; n++) {
        long long mem;
        if (numa_node_size64(n, &mem) < 0) {
            qemudReportError(conn, NULL, NULL, VIR_ERR_INTERNAL_ERROR,
                             "%s", _("Failed to query NUMA free memory"));
            return -1;
        }
        freeMem += mem;
    }

    return freeMem;
}

#endif
1325

D
Daniel P. Berrange 已提交
1326 1327 1328
static int qemudGetProcessInfo(unsigned long long *cpuTime, int pid) {
    char proc[PATH_MAX];
    FILE *pidinfo;
1329
    unsigned long long usertime, systime;
D
Daniel P. Berrange 已提交
1330 1331 1332 1333 1334 1335

    if (snprintf(proc, sizeof(proc), "/proc/%d/stat", pid) >= (int)sizeof(proc)) {
        return -1;
    }

    if (!(pidinfo = fopen(proc, "r"))) {
1336
        /*printf("cannot read pid info");*/
D
Daniel P. Berrange 已提交
1337 1338 1339 1340 1341
        /* VM probably shut down, so fake 0 */
        *cpuTime = 0;
        return 0;
    }

1342
    if (fscanf(pidinfo, "%*d %*s %*c %*d %*d %*d %*d %*d %*u %*u %*u %*u %*u %llu %llu", &usertime, &systime) != 2) {
1343
        qemudDebug("not enough arg");
D
Daniel P. Berrange 已提交
1344 1345 1346 1347 1348 1349 1350 1351
        return -1;
    }

    /* We got jiffies
     * We want nanoseconds
     * _SC_CLK_TCK is jiffies per second
     * So calulate thus....
     */
1352
    *cpuTime = 1000ull * 1000ull * 1000ull * (usertime + systime) / (unsigned long long)sysconf(_SC_CLK_TCK);
D
Daniel P. Berrange 已提交
1353

1354
    qemudDebug("Got %llu %llu %llu", usertime, systime, *cpuTime);
D
Daniel P. Berrange 已提交
1355 1356 1357 1358 1359 1360 1361

    fclose(pidinfo);

    return 0;
}


1362
static virDomainPtr qemudDomainLookupByID(virConnectPtr conn,
1363
                                          int id) {
1364
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1365
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, id);
1366 1367 1368
    virDomainPtr dom;

    if (!vm) {
1369
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_DOMAIN, NULL);
1370 1371 1372
        return NULL;
    }

1373
    dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
1374
    if (dom) dom->id = vm->def->id;
1375 1376
    return dom;
}
1377
static virDomainPtr qemudDomainLookupByUUID(virConnectPtr conn,
1378
                                            const unsigned char *uuid) {
1379
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1380
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, uuid);
1381 1382 1383
    virDomainPtr dom;

    if (!vm) {
1384
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_DOMAIN, NULL);
1385 1386 1387
        return NULL;
    }

1388
    dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
1389
    if (dom) dom->id = vm->def->id;
1390 1391
    return dom;
}
1392
static virDomainPtr qemudDomainLookupByName(virConnectPtr conn,
1393
                                            const char *name) {
1394
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1395
    virDomainObjPtr vm = virDomainFindByName(&driver->domains, name);
1396 1397 1398
    virDomainPtr dom;

    if (!vm) {
1399
        qemudReportError(conn, NULL, NULL, VIR_ERR_NO_DOMAIN, NULL);
1400 1401 1402
        return NULL;
    }

1403
    dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
1404
    if (dom) dom->id = vm->def->id;
1405 1406 1407
    return dom;
}

1408
static int qemudGetVersion(virConnectPtr conn, unsigned long *version) {
1409
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1410
    if (qemudExtractVersion(conn, driver) < 0)
1411 1412
        return -1;

1413 1414
    *version = qemu_driver->qemuVersion;
    return 0;
D
Daniel P. Berrange 已提交
1415 1416
}

1417 1418 1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438
static char *
qemudGetHostname (virConnectPtr conn)
{
    int r;
    char hostname[HOST_NAME_MAX+1], *str;

    r = gethostname (hostname, HOST_NAME_MAX+1);
    if (r == -1) {
        qemudReportError (conn, NULL, NULL, VIR_ERR_SYSTEM_ERROR,
                          "%s", strerror (errno));
        return NULL;
    }
    /* Caller frees this string. */
    str = strdup (hostname);
    if (str == NULL) {
        qemudReportError (conn, NULL, NULL, VIR_ERR_SYSTEM_ERROR,
                         "%s", strerror (errno));
        return NULL;
    }
    return str;
}

1439
static int qemudListDomains(virConnectPtr conn, int *ids, int nids) {
1440
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1441 1442 1443 1444 1445 1446
    int got = 0, i;

    for (i = 0 ; i < driver->domains.count && got < nids ; i++)
        if (virDomainIsActive(driver->domains.objs[i]))
            ids[got++] = driver->domains.objs[i]->def->id;

D
Daniel P. Berrange 已提交
1447 1448
    return got;
}
1449
static int qemudNumDomains(virConnectPtr conn) {
1450
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
1451 1452 1453 1454
    int n = 0, i;

    for (i = 0 ; i < driver->domains.count ; i++)
        if (virDomainIsActive(driver->domains.objs[i]))
1455
            n++;
1456

1457
    return n;
D
Daniel P. Berrange 已提交
1458
}
1459
static virDomainPtr qemudDomainCreate(virConnectPtr conn, const char *xml,
1460
                                      unsigned int flags ATTRIBUTE_UNUSED) {
1461 1462
    virDomainDefPtr def;
    virDomainObjPtr vm;
1463 1464
    virDomainPtr dom;
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
D
Daniel P. Berrange 已提交
1465

1466
    if (!(def = virDomainDefParseString(conn, driver->caps, xml)))
1467 1468
        return NULL;

1469
    vm = virDomainFindByName(&driver->domains, def->name);
1470
    if (vm) {
1471
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
1472
                         _("domain '%s' is already defined"),
1473 1474 1475 1476
                         def->name);
        virDomainDefFree(def);
        return NULL;
    }
1477
    vm = virDomainFindByUUID(&driver->domains, def->uuid);
1478 1479 1480 1481 1482
    if (vm) {
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(def->uuid, uuidstr);
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
1483
                         _("domain with uuid '%s' is already defined"),
1484 1485 1486 1487
                         uuidstr);
        virDomainDefFree(def);
        return NULL;
    }
1488

1489 1490 1491 1492
    if (!(vm = virDomainAssignDef(conn,
                                  &driver->domains,
                                  def))) {
        virDomainDefFree(def);
D
Daniel P. Berrange 已提交
1493 1494 1495
        return NULL;
    }

1496 1497 1498
    if (qemudStartVMDaemon(conn, driver, vm, NULL) < 0) {
        virDomainRemoveInactive(&driver->domains,
                                vm);
D
Daniel P. Berrange 已提交
1499 1500 1501
        return NULL;
    }

1502
    dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
1503
    if (dom) dom->id = vm->def->id;
1504
    return dom;
D
Daniel P. Berrange 已提交
1505 1506 1507
}


1508
static int qemudDomainSuspend(virDomainPtr dom) {
1509
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
D
Daniel P. Berrange 已提交
1510
    char *info;
1511
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
D
Daniel P. Berrange 已提交
1512
    if (!vm) {
1513
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN, _("no domain with matching id %d"), dom->id);
D
Daniel P. Berrange 已提交
1514 1515
        return -1;
    }
1516
    if (!virDomainIsActive(vm)) {
1517 1518
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("domain is not running"));
D
Daniel P. Berrange 已提交
1519 1520
        return -1;
    }
1521
    if (vm->state == VIR_DOMAIN_PAUSED)
D
Daniel P. Berrange 已提交
1522
        return 0;
D
Daniel P. Berrange 已提交
1523

1524
    if (qemudMonitorCommand(driver, vm, "stop", &info) < 0) {
1525 1526
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("suspend operation failed"));
D
Daniel P. Berrange 已提交
1527 1528
        return -1;
    }
1529
    vm->state = VIR_DOMAIN_PAUSED;
1530
    qemudDebug("Reply %s", info);
1531
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_SUSPENDED);
1532
    VIR_FREE(info);
D
Daniel P. Berrange 已提交
1533 1534 1535 1536
    return 0;
}


1537
static int qemudDomainResume(virDomainPtr dom) {
1538
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
D
Daniel P. Berrange 已提交
1539
    char *info;
1540
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
D
Daniel P. Berrange 已提交
1541
    if (!vm) {
1542 1543
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         _("no domain with matching id %d"), dom->id);
D
Daniel P. Berrange 已提交
1544 1545
        return -1;
    }
1546
    if (!virDomainIsActive(vm)) {
1547 1548
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("domain is not running"));
D
Daniel P. Berrange 已提交
1549 1550
        return -1;
    }
1551
    if (vm->state == VIR_DOMAIN_RUNNING)
D
Daniel P. Berrange 已提交
1552
        return 0;
1553
    if (qemudMonitorCommand(driver, vm, "cont", &info) < 0) {
1554 1555
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("resume operation failed"));
D
Daniel P. Berrange 已提交
1556 1557
        return -1;
    }
1558
    vm->state = VIR_DOMAIN_RUNNING;
1559
    qemudDebug("Reply %s", info);
1560
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_RESUMED);
1561
    VIR_FREE(info);
D
Daniel P. Berrange 已提交
1562
    return 0;
D
Daniel P. Berrange 已提交
1563 1564 1565
}


1566 1567
static int qemudDomainShutdown(virDomainPtr dom) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1568
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
1569 1570 1571 1572
    char* info;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1573
                         _("no domain with matching id %d"), dom->id);
1574 1575 1576 1577 1578
        return -1;
    }

    if (qemudMonitorCommand(driver, vm, "system_powerdown", &info) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1579
                         "%s", _("shutdown operation failed"));
1580 1581
        return -1;
    }
D
Daniel Veillard 已提交
1582
    VIR_FREE(info);
1583 1584 1585 1586 1587
    return 0;

}


1588
static int qemudDomainDestroy(virDomainPtr dom) {
1589
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1590
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
1591

D
Daniel P. Berrange 已提交
1592
    if (!vm) {
1593
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1594
                         _("no domain with matching id %d"), dom->id);
D
Daniel P. Berrange 已提交
1595 1596
        return -1;
    }
1597

1598
    qemudShutdownVMDaemon(dom->conn, driver, vm);
1599
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_STOPPED);
1600
    if (!vm->persistent)
1601 1602
        virDomainRemoveInactive(&driver->domains,
                                vm);
1603
    return 0;
D
Daniel P. Berrange 已提交
1604 1605 1606
}


1607 1608
static char *qemudDomainGetOSType(virDomainPtr dom) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1609
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1610 1611 1612 1613
    char *type;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1614
                         "%s", _("no domain with matching uuid"));
1615 1616 1617 1618
        return NULL;
    }

    if (!(type = strdup(vm->def->os.type))) {
1619 1620
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_MEMORY,
                         "%s", _("failed to allocate space for ostype"));
1621 1622 1623 1624 1625
        return NULL;
    }
    return type;
}

1626 1627 1628
/* Returns max memory in kb, 0 if error */
static unsigned long qemudDomainGetMaxMemory(virDomainPtr dom) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1629
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1630 1631

    if (!vm) {
1632 1633 1634
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(dom->uuid, uuidstr);
1635
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1636
                         _("no domain with matching uuid '%s'"), uuidstr);
1637 1638 1639 1640 1641 1642 1643 1644
        return 0;
    }

    return vm->def->maxmem;
}

static int qemudDomainSetMaxMemory(virDomainPtr dom, unsigned long newmax) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1645
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1646 1647

    if (!vm) {
1648 1649 1650
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(dom->uuid, uuidstr);
1651
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1652
                         _("no domain with matching uuid '%s'"), uuidstr);
1653 1654 1655 1656 1657
        return -1;
    }

    if (newmax < vm->def->memory) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
1658
                         "%s", _("cannot set max memory lower than current memory"));
1659 1660 1661 1662 1663 1664 1665 1666 1667
        return -1;
    }

    vm->def->maxmem = newmax;
    return 0;
}

static int qemudDomainSetMemory(virDomainPtr dom, unsigned long newmem) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1668
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1669 1670

    if (!vm) {
1671 1672 1673
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(dom->uuid, uuidstr);
1674
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1675
                         _("no domain with matching uuid '%s'"), uuidstr);
1676 1677 1678
        return -1;
    }

1679
    if (virDomainIsActive(vm)) {
1680
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
1681
                         "%s", _("cannot set memory of an active domain"));
1682 1683 1684 1685 1686
        return -1;
    }

    if (newmem > vm->def->maxmem) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
1687
                         "%s", _("cannot set memory higher than max memory"));
1688 1689 1690 1691 1692 1693 1694
        return -1;
    }

    vm->def->memory = newmem;
    return 0;
}

1695
static int qemudDomainGetInfo(virDomainPtr dom,
1696
                              virDomainInfoPtr info) {
1697
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1698
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
D
Daniel P. Berrange 已提交
1699
    if (!vm) {
1700 1701
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
D
Daniel P. Berrange 已提交
1702 1703 1704
        return -1;
    }

1705
    info->state = vm->state;
D
Daniel P. Berrange 已提交
1706

1707
    if (!virDomainIsActive(vm)) {
1708
        info->cpuTime = 0;
D
Daniel P. Berrange 已提交
1709
    } else {
1710
        if (qemudGetProcessInfo(&(info->cpuTime), vm->pid) < 0) {
1711
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED, ("cannot read cputime for domain"));
D
Daniel P. Berrange 已提交
1712 1713 1714 1715
            return -1;
        }
    }

1716 1717 1718
    info->maxMem = vm->def->maxmem;
    info->memory = vm->def->memory;
    info->nrVirtCpu = vm->def->vcpus;
D
Daniel P. Berrange 已提交
1719 1720 1721 1722
    return 0;
}


D
Daniel P. Berrange 已提交
1723
static char *qemudEscape(const char *in, int shell)
1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744
{
    int len = 0;
    int i, j;
    char *out;

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

       To pass through both QEMU + the shell, we need to escape
       the single character ' as the five characters '\\''
    */

    for (i = 0; in[i] != '\0'; i++) {
        switch(in[i]) {
        case '\r':
        case '\n':
        case '"':
        case '\\':
            len += 2;
            break;
        case '\'':
D
Daniel P. Berrange 已提交
1745 1746 1747 1748
            if (shell)
                len += 5;
            else
                len += 1;
1749 1750 1751 1752 1753 1754 1755
            break;
        default:
            len += 1;
            break;
        }
    }

1756
    if (VIR_ALLOC_N(out, len + 1) < 0)
1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774
        return NULL;

    for (i = j = 0; in[i] != '\0'; i++) {
        switch(in[i]) {
        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;
        case '\'':
D
Daniel P. Berrange 已提交
1775 1776 1777 1778 1779 1780 1781 1782 1783
            if (shell) {
                out[j++] = '\'';
                out[j++] = '\\';
                out[j++] = '\\';
                out[j++] = '\'';
                out[j++] = '\'';
            } else {
                out[j++] = in[i];
            }
1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794
            break;
        default:
            out[j++] = in[i];
            break;
        }
    }
    out[j] = '\0';

    return out;
}

1795 1796 1797 1798 1799
static char *qemudEscapeMonitorArg(const char *in)
{
    return qemudEscape(in, 0);
}

D
Daniel P. Berrange 已提交
1800 1801 1802 1803
static char *qemudEscapeShellArg(const char *in)
{
    return qemudEscape(in, 1);
}
1804

1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815
#define QEMUD_SAVE_MAGIC "LibvirtQemudSave"
#define QEMUD_SAVE_VERSION 1

struct qemud_save_header {
    char magic[sizeof(QEMUD_SAVE_MAGIC)-1];
    int version;
    int xml_len;
    int was_running;
    int unused[16];
};

1816
static int qemudDomainSave(virDomainPtr dom,
1817
                           const char *path) {
1818
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1819
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
1820 1821 1822 1823 1824 1825 1826 1827 1828 1829
    char *command, *info;
    int fd;
    char *safe_path;
    char *xml;
    struct qemud_save_header header;

    memset(&header, 0, sizeof(header));
    memcpy(header.magic, QEMUD_SAVE_MAGIC, sizeof(header.magic));
    header.version = QEMUD_SAVE_VERSION;

D
Daniel P. Berrange 已提交
1830
    if (!vm) {
1831
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1832
                         _("no domain with matching id %d"), dom->id);
D
Daniel P. Berrange 已提交
1833 1834
        return -1;
    }
1835

1836
    if (!virDomainIsActive(vm)) {
1837
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1838
                         "%s", _("domain is not running"));
D
Daniel P. Berrange 已提交
1839 1840
        return -1;
    }
1841 1842 1843 1844 1845 1846

    /* Pause */
    if (vm->state == VIR_DOMAIN_RUNNING) {
        header.was_running = 1;
        if (qemudDomainSuspend(dom) != 0) {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1847
                             "%s", _("failed to pause domain"));
1848 1849 1850 1851 1852
            return -1;
        }
    }

    /* Get XML for the domain */
1853
    xml = virDomainDefFormat(dom->conn, vm->def, VIR_DOMAIN_XML_SECURE);
1854 1855
    if (!xml) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1856
                         "%s", _("failed to get domain xml"));
1857 1858 1859 1860 1861 1862 1863
        return -1;
    }
    header.xml_len = strlen(xml) + 1;

    /* Write header to file, followed by XML */
    if ((fd = open(path, O_CREAT|O_TRUNC|O_WRONLY, S_IRUSR|S_IWUSR)) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1864
                         _("failed to create '%s'"), path);
1865
        VIR_FREE(xml);
1866 1867 1868 1869 1870
        return -1;
    }

    if (safewrite(fd, &header, sizeof(header)) != sizeof(header)) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1871
                         "%s", _("failed to write save header"));
1872
        close(fd);
1873
        VIR_FREE(xml);
1874 1875 1876 1877 1878
        return -1;
    }

    if (safewrite(fd, xml, header.xml_len) != header.xml_len) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1879
                         "%s", _("failed to write xml"));
1880
        close(fd);
1881
        VIR_FREE(xml);
1882 1883 1884 1885
        return -1;
    }

    close(fd);
1886
    VIR_FREE(xml);
1887 1888 1889 1890 1891

    /* Migrate to file */
    safe_path = qemudEscapeShellArg(path);
    if (!safe_path) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1892
                         "%s", _("out of memory"));
1893 1894 1895 1896
        return -1;
    }
    if (asprintf (&command, "migrate \"exec:"
                  "dd of='%s' oflag=append conv=notrunc 2>/dev/null"
1897
                  "\"", safe_path) == -1) {
1898
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1899
                         "%s", _("out of memory"));
1900
        VIR_FREE(safe_path);
1901 1902 1903 1904 1905 1906
        return -1;
    }
    free(safe_path);

    if (qemudMonitorCommand(driver, vm, command, &info) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
1907
                         "%s", _("migrate operation failed"));
1908
        VIR_FREE(command);
1909 1910 1911
        return -1;
    }

1912 1913 1914 1915 1916 1917 1918 1919 1920 1921 1922 1923 1924
    DEBUG ("migrate reply: %s", info);

    /* If the command isn't supported then qemu prints:
     * unknown command: migrate" */
    if (strstr(info, "unknown command:")) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                          "%s",
                          _("'migrate' not supported by this qemu"));
        VIR_FREE(info);
        VIR_FREE(command);
        return -1;
    }

1925 1926
    VIR_FREE(info);
    VIR_FREE(command);
1927 1928 1929

    /* Shut it down */
    qemudShutdownVMDaemon(dom->conn, driver, vm);
1930
    if (!vm->persistent)
1931 1932
        virDomainRemoveInactive(&driver->domains,
                                vm);
1933
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_SAVED);
1934
    return 0;
D
Daniel P. Berrange 已提交
1935 1936 1937
}


1938
static int qemudDomainSetVcpus(virDomainPtr dom, unsigned int nvcpus) {
1939 1940
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1941 1942 1943
    int max;

    if (!vm) {
1944 1945 1946
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(dom->uuid, uuidstr);
1947
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
1948
                         _("no domain with matching uuid '%s'"), uuidstr);
1949 1950 1951
        return -1;
    }

1952
    if (virDomainIsActive(vm)) {
1953
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT, "%s",
1954 1955 1956 1957 1958 1959 1960 1961 1962 1963 1964 1965 1966 1967 1968 1969 1970 1971 1972 1973 1974
                         _("cannot change vcpu count of an active domain"));
        return -1;
    }

    if ((max = qemudDomainGetMaxVcpus(dom)) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR, "%s",
                         _("could not determine max vcpus for the domain"));
        return -1;
    }

    if (nvcpus > max) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                         _("requested vcpus is greater than max allowable"
                           " vcpus for the domain: %d > %d"), nvcpus, max);
        return -1;
    }

    vm->def->vcpus = nvcpus;
    return 0;
}

1975 1976 1977 1978 1979 1980 1981 1982

#if HAVE_SCHED_GETAFFINITY
static int
qemudDomainPinVcpu(virDomainPtr dom,
                   unsigned int vcpu,
                   unsigned char *cpumap,
                   int maplen) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
1983
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
1984 1985 1986 1987
    cpu_set_t mask;
    int i, maxcpu;
    virNodeInfo nodeinfo;

1988
    if (!virDomainIsActive(vm)) {
1989 1990 1991 1992 1993 1994 1995 1996 1997 1998 1999 2000 2001 2002 2003 2004 2005 2006 2007 2008 2009 2010 2011 2012 2013 2014 2015 2016 2017 2018 2019 2020 2021 2022 2023 2024 2025 2026 2027 2028 2029 2030 2031 2032 2033 2034 2035
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                         "%s",_("cannot pin vcpus on an inactive domain"));
        return -1;
    }

    if (vcpu > (vm->nvcpupids-1)) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                         _("vcpu number out of range %d > %d"),
                         vcpu, vm->nvcpupids);
        return -1;
    }

    if (virNodeInfoPopulate(dom->conn, &nodeinfo) < 0)
        return -1;

    maxcpu = maplen * 8;
    if (maxcpu > nodeinfo.cpus)
        maxcpu = nodeinfo.cpus;

    CPU_ZERO(&mask);
    for (i = 0 ; i < maxcpu ; i++) {
        if ((cpumap[i/8] >> (i % 8)) & 1)
            CPU_SET(i, &mask);
    }

    if (vm->vcpupids != NULL) {
        if (sched_setaffinity(vm->vcpupids[vcpu], sizeof(mask), &mask) < 0) {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                             _("cannot set affinity: %s"), strerror(errno));
            return -1;
        }
    } else {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                         "%s", _("cpu affinity is not supported"));
        return -1;
    }

    return 0;
}

static int
qemudDomainGetVcpus(virDomainPtr dom,
                    virVcpuInfoPtr info,
                    int maxinfo,
                    unsigned char *cpumaps,
                    int maplen) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2036
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2037 2038 2039
    virNodeInfo nodeinfo;
    int i, v, maxcpu;

2040
    if (!virDomainIsActive(vm)) {
2041 2042 2043 2044 2045 2046 2047 2048 2049 2050 2051 2052 2053 2054 2055 2056 2057 2058 2059 2060 2061 2062 2063 2064 2065 2066 2067 2068 2069 2070 2071 2072 2073 2074 2075 2076 2077 2078 2079 2080 2081 2082 2083 2084 2085 2086 2087 2088 2089 2090 2091 2092 2093 2094 2095 2096 2097 2098
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                         "%s",_("cannot pin vcpus on an inactive domain"));
        return -1;
    }

    if (virNodeInfoPopulate(dom->conn, &nodeinfo) < 0)
        return -1;

    maxcpu = maplen * 8;
    if (maxcpu > nodeinfo.cpus)
        maxcpu = nodeinfo.cpus;

    /* Clamp to actual number of vcpus */
    if (maxinfo > vm->nvcpupids)
        maxinfo = vm->nvcpupids;

    if (maxinfo < 1)
        return 0;

    if (info != NULL) {
        memset(info, 0, sizeof(*info) * maxinfo);
        for (i = 0 ; i < maxinfo ; i++) {
            info[i].number = i;
            info[i].state = VIR_VCPU_RUNNING;
            /* XXX cpu time, current pCPU mapping */
        }
    }

    if (cpumaps != NULL) {
        memset(cpumaps, 0, maplen * maxinfo);
        if (vm->vcpupids != NULL) {
            for (v = 0 ; v < maxinfo ; v++) {
                cpu_set_t mask;
                unsigned char *cpumap = VIR_GET_CPUMAP(cpumaps, maplen, v);
                CPU_ZERO(&mask);

                if (sched_getaffinity(vm->vcpupids[v], sizeof(mask), &mask) < 0) {
                    qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                                     _("cannot get affinity: %s"), strerror(errno));
                    return -1;
                }

                for (i = 0 ; i < maxcpu ; i++)
                    if (CPU_ISSET(i, &mask))
                        VIR_USE_CPU(cpumap, i);
            }
        } else {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                             "%s", _("cpu affinity is not available"));
            return -1;
        }
    }

    return maxinfo;
}
#endif /* HAVE_SCHED_GETAFFINITY */


2099 2100
static int qemudDomainGetMaxVcpus(virDomainPtr dom) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2101
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2102 2103 2104 2105
    const char *type;
    int ret;

    if (!vm) {
2106 2107 2108
        char uuidstr[VIR_UUID_STRING_BUFLEN];

        virUUIDFormat(dom->uuid, uuidstr);
2109
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
2110
                         _("no domain with matching uuid '%s'"), uuidstr);
2111 2112 2113
        return -1;
    }

2114
    if (!(type = virDomainVirtTypeToString(vm->def->virtType))) {
2115 2116 2117 2118 2119 2120 2121 2122 2123 2124 2125 2126 2127 2128
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("unknown virt type in domain definition '%d'"),
                         vm->def->virtType);
        return -1;
    }

    if ((ret = qemudGetMaxVCPUs(dom->conn, type)) < 0) {
        return -1;
    }

    return ret;
}


2129
static int qemudDomainRestore(virConnectPtr conn,
2130 2131
                       const char *path) {
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
2132 2133
    virDomainDefPtr def;
    virDomainObjPtr vm;
2134
    int fd;
2135
    int ret;
2136 2137 2138 2139 2140 2141
    char *xml;
    struct qemud_save_header header;

    /* Verify the header and read the XML */
    if ((fd = open(path, O_RDONLY)) < 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2142
                         "%s", _("cannot read domain image"));
2143 2144 2145 2146 2147
        return -1;
    }

    if (saferead(fd, &header, sizeof(header)) != sizeof(header)) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2148
                         "%s", _("failed to read qemu header"));
2149 2150 2151 2152 2153 2154
        close(fd);
        return -1;
    }

    if (memcmp(header.magic, QEMUD_SAVE_MAGIC, sizeof(header.magic)) != 0) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2155
                         "%s", _("image magic is incorrect"));
2156 2157 2158 2159 2160 2161
        close(fd);
        return -1;
    }

    if (header.version > QEMUD_SAVE_VERSION) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2162
                         _("image version is not supported (%d > %d)"),
2163 2164 2165 2166 2167
                         header.version, QEMUD_SAVE_VERSION);
        close(fd);
        return -1;
    }

2168
    if (VIR_ALLOC_N(xml, header.xml_len) < 0) {
2169
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2170
                         "%s", _("out of memory"));
2171 2172 2173 2174 2175 2176
        close(fd);
        return -1;
    }

    if (saferead(fd, xml, header.xml_len) != header.xml_len) {
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2177
                         "%s", _("failed to read XML"));
2178
        close(fd);
2179
        VIR_FREE(xml);
2180 2181 2182 2183
        return -1;
    }

    /* Create a domain from this XML */
2184
    if (!(def = virDomainDefParseString(conn, driver->caps, xml))) {
2185
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2186
                         "%s", _("failed to parse XML"));
2187
        close(fd);
2188
        VIR_FREE(xml);
2189 2190
        return -1;
    }
2191
    VIR_FREE(xml);
2192 2193

    /* Ensure the name and UUID don't already exist in an active VM */
2194
    vm = virDomainFindByUUID(&driver->domains, def->uuid);
2195
    if (!vm)
2196
        vm = virDomainFindByName(&driver->domains, def->name);
2197
    if (vm && virDomainIsActive(vm)) {
2198
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2199
                         _("domain is already active as '%s'"), vm->def->name);
2200 2201 2202 2203
        close(fd);
        return -1;
    }

2204 2205 2206
    if (!(vm = virDomainAssignDef(conn,
                                  &driver->domains,
                                  def))) {
2207
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2208
                         "%s", _("failed to assign new VM"));
2209
        virDomainDefFree(def);
2210 2211 2212 2213 2214
        close(fd);
        return -1;
    }

    /* Set the migration source and start it up. */
2215
    vm->stdin_fd = fd;
2216
    ret = qemudStartVMDaemon(conn, driver, vm, "stdio");
2217
    close(fd);
2218
    vm->stdin_fd = -1;
2219
    if (ret < 0) {
2220
        qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2221
                         "%s", _("failed to start VM"));
2222
        if (!vm->persistent)
2223 2224
            virDomainRemoveInactive(&driver->domains,
                                    vm);
2225 2226 2227 2228 2229 2230
        return -1;
    }

    /* If it was running before, resume it now. */
    if (header.was_running) {
        char *info;
2231
        if (qemudMonitorCommand(driver, vm, "cont", &info) < 0) {
2232
            qemudReportError(conn, NULL, NULL, VIR_ERR_OPERATION_FAILED,
2233
                             "%s", _("failed to resume domain"));
2234 2235
            return -1;
        }
2236
        VIR_FREE(info);
2237 2238 2239
        vm->state = VIR_DOMAIN_RUNNING;
    }

2240
    qemudDomainEventDispatch(driver, vm, VIR_DOMAIN_EVENT_RESTORED);
2241
    return 0;
D
Daniel P. Berrange 已提交
2242 2243 2244
}


2245
static char *qemudDomainDumpXML(virDomainPtr dom,
2246
                                int flags ATTRIBUTE_UNUSED) {
2247
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2248
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
D
Daniel P. Berrange 已提交
2249
    if (!vm) {
2250 2251
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
2252
        return NULL;
D
Daniel P. Berrange 已提交
2253 2254
    }

2255 2256 2257 2258
    return virDomainDefFormat(dom->conn,
                              (flags & VIR_DOMAIN_XML_INACTIVE) && vm->newDef ?
                              vm->newDef : vm->def,
                              flags);
D
Daniel P. Berrange 已提交
2259 2260 2261
}


2262
static int qemudListDefinedDomains(virConnectPtr conn,
2263 2264 2265
                            char **const names, int nnames) {
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
    int got = 0, i;
2266 2267 2268 2269

    for (i = 0 ; i < driver->domains.count && got < nnames ; i++) {
        if (!virDomainIsActive(driver->domains.objs[i])) {
            if (!(names[got++] = strdup(driver->domains.objs[i]->def->name))) {
2270
                qemudReportError(conn, NULL, NULL, VIR_ERR_NO_MEMORY,
2271
                                 "%s", _("failed to allocate space for VM name string"));
2272 2273
                goto cleanup;
            }
2274
        }
D
Daniel P. Berrange 已提交
2275
    }
2276

D
Daniel P. Berrange 已提交
2277
    return got;
2278 2279 2280

 cleanup:
    for (i = 0 ; i < got ; i++)
2281
        VIR_FREE(names[i]);
2282
    return -1;
D
Daniel P. Berrange 已提交
2283 2284
}

2285
static int qemudNumDefinedDomains(virConnectPtr conn) {
2286
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
2287 2288 2289 2290
    int n = 0, i;

    for (i = 0 ; i < driver->domains.count ; i++)
        if (!virDomainIsActive(driver->domains.objs[i]))
2291
            n++;
2292

2293
    return n;
D
Daniel P. Berrange 已提交
2294 2295 2296
}


2297
static int qemudDomainStart(virDomainPtr dom) {
2298
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2299
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2300 2301

    if (!vm) {
2302
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
2303
                         "%s", _("no domain with matching uuid"));
2304
        return -1;
2305 2306
    }

2307
    return qemudStartVMDaemon(dom->conn, driver, vm, NULL);
D
Daniel P. Berrange 已提交
2308 2309 2310
}


2311
static virDomainPtr qemudDomainDefine(virConnectPtr conn, const char *xml) {
2312
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;
2313 2314
    virDomainDefPtr def;
    virDomainObjPtr vm;
2315
    virDomainPtr dom;
2316

2317
    if (!(def = virDomainDefParseString(conn, driver->caps, xml)))
2318 2319
        return NULL;

2320 2321 2322 2323
    if (!(vm = virDomainAssignDef(conn,
                                  &driver->domains,
                                  def))) {
        virDomainDefFree(def);
2324 2325
        return NULL;
    }
2326
    vm->persistent = 1;
2327

2328 2329
    if (virDomainSaveConfig(conn,
                            driver->configDir,
2330
                            vm->newDef ? vm->newDef : vm->def) < 0) {
2331 2332
        virDomainRemoveInactive(&driver->domains,
                                vm);
2333 2334 2335
        return NULL;
    }

2336
    dom = virGetDomain(conn, vm->def->name, vm->def->uuid);
2337
    if (dom) dom->id = vm->def->id;
2338
    return dom;
D
Daniel P. Berrange 已提交
2339 2340
}

2341
static int qemudDomainUndefine(virDomainPtr dom) {
2342
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2343
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
D
Daniel P. Berrange 已提交
2344 2345

    if (!vm) {
2346 2347
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
D
Daniel P. Berrange 已提交
2348 2349 2350
        return -1;
    }

2351
    if (virDomainIsActive(vm)) {
2352 2353
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot delete active domain"));
D
Daniel P. Berrange 已提交
2354 2355 2356
        return -1;
    }

2357 2358 2359 2360 2361 2362 2363
    if (!vm->persistent) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot undefine transient domain"));
        return -1;
    }

    if (virDomainDeleteConfig(dom->conn, driver->configDir, driver->autostartDir, vm) < 0)
D
Daniel P. Berrange 已提交
2364 2365
        return -1;

2366 2367
    virDomainRemoveInactive(&driver->domains,
                            vm);
D
Daniel P. Berrange 已提交
2368 2369 2370 2371

    return 0;
}

2372
/* Return the disks name for use in monitor commands */
2373 2374
static char *qemudDiskDeviceName(const virDomainPtr dom,
                                 const virDomainDiskDefPtr disk) {
2375 2376 2377 2378 2379 2380 2381 2382 2383 2384 2385 2386 2387 2388

    int busid, devid;
    int ret;
    char *devname;

    if (virDiskNameToBusDeviceIndex(disk, &busid, &devid) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("cannot convert disk '%s' to bus/device index"),
                         disk->dst);
        return NULL;
    }

    switch (disk->bus) {
        case VIR_DOMAIN_DISK_BUS_IDE:
2389 2390 2391 2392
            if (disk->device== VIR_DOMAIN_DISK_DEVICE_DISK)
                ret = asprintf(&devname, "ide%d-hd%d", busid, devid);
            else
                ret = asprintf(&devname, "ide%d-cd%d", busid, devid);
2393 2394
            break;
        case VIR_DOMAIN_DISK_BUS_SCSI:
2395 2396 2397 2398
            if (disk->device == VIR_DOMAIN_DISK_DEVICE_DISK)
                ret = asprintf(&devname, "scsi%d-hd%d", busid, devid);
            else
                ret = asprintf(&devname, "scsi%d-cd%d", busid, devid);
2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420 2421 2422 2423
            break;
        case VIR_DOMAIN_DISK_BUS_FDC:
            ret = asprintf(&devname, "floppy%d", devid);
            break;
        case VIR_DOMAIN_DISK_BUS_VIRTIO:
            ret = asprintf(&devname, "virtio%d", devid);
            break;
        default:
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                             _("Unsupported disk name mapping for bus '%s'"),
                             virDomainDiskBusTypeToString(disk->bus));
            return NULL;
    }

    if (ret == -1) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return NULL;
    }

    return devname;
}

static int qemudDomainChangeEjectableMedia(virDomainPtr dom,
                                           virDomainDeviceDefPtr dev)
{
2424
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2425
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2426
    virDomainDiskDefPtr origdisk = NULL, newdisk;
2427
    char *cmd, *reply, *safe_path;
2428
    char *devname = NULL;
2429
    unsigned int qemuCmdFlags;
2430
    int i;
2431 2432 2433 2434 2435 2436 2437

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

2438
    origdisk = NULL;
2439
    newdisk = dev->data.disk;
2440 2441 2442 2443
    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (vm->def->disks[i]->bus == newdisk->bus &&
            STREQ(vm->def->disks[i]->dst, newdisk->dst)) {
            origdisk = vm->def->disks[i];
2444
            break;
2445
        }
2446 2447 2448 2449 2450 2451 2452 2453 2454 2455 2456 2457 2458 2459 2460 2461 2462 2463 2464 2465 2466 2467 2468 2469 2470 2471 2472 2473 2474 2475 2476 2477 2478 2479 2480 2481 2482 2483 2484 2485 2486 2487 2488
    }

    if (!origdisk) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("No device with bus '%s' and target '%s'"),
                         virDomainDiskBusTypeToString(newdisk->bus),
                         newdisk->dst);
        return -1;
    }

    if (qemudExtractVersionInfo(vm->def->emulator,
                                NULL,
                                &qemuCmdFlags) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         _("Cannot determine QEMU argv syntax %s"),
                         vm->def->emulator);
        return -1;
    }

    if (qemuCmdFlags & QEMUD_CMD_FLAG_DRIVE) {
        if (!(devname = qemudDiskDeviceName(dom, newdisk)))
            return -1;
    } else {
        /* Back compat for no -drive option */
        if (newdisk->device == VIR_DOMAIN_DISK_DEVICE_FLOPPY)
            devname = strdup(newdisk->dst);
        else if (newdisk->device == VIR_DOMAIN_DISK_DEVICE_CDROM &&
                 STREQ(newdisk->dst, "hdc"))
            devname = strdup("cdrom");
        else {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                             _("Emulator version does not support removable "
                               "media for device '%s' and target '%s'"),
                               virDomainDiskDeviceTypeToString(newdisk->device),
                               newdisk->dst);
            return -1;
        }

        if (!devname) {
            qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
            return -1;
        }
    }
2489

2490
    if (newdisk->src) {
2491 2492
        safe_path = qemudEscapeMonitorArg(newdisk->src);
        if (!safe_path) {
2493 2494
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_MEMORY, NULL);
            VIR_FREE(devname);
2495 2496
            return -1;
        }
2497 2498
        if (asprintf (&cmd, "change %s \"%s\"", devname, safe_path) == -1) {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_MEMORY, NULL);
2499
            VIR_FREE(safe_path);
2500
            VIR_FREE(devname);
2501 2502
            return -1;
        }
2503
        VIR_FREE(safe_path);
2504

2505 2506 2507
    } else if (asprintf(&cmd, "eject %s", devname) == -1) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_MEMORY, NULL);
        VIR_FREE(devname);
2508 2509
        return -1;
    }
2510
    VIR_FREE(devname);
2511 2512

    if (qemudMonitorCommand(driver, vm, cmd, &reply) < 0) {
2513 2514
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("cannot change cdrom media"));
2515
        VIR_FREE(cmd);
2516 2517
        return -1;
    }
2518 2519 2520 2521

    /* If the command failed qemu prints:
     * device not found, device is locked ...
     * No message is printed on success it seems */
2522
    DEBUG ("ejectable media change reply: %s", reply);
2523 2524 2525 2526 2527 2528 2529
    if (strstr(reply, "\ndevice ")) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          "%s", _("changing cdrom media failed"));
        VIR_FREE(reply);
        VIR_FREE(cmd);
        return -1;
    }
2530 2531
    VIR_FREE(reply);
    VIR_FREE(cmd);
2532

2533 2534
    VIR_FREE(origdisk->src);
    origdisk->src = newdisk->src;
2535
    newdisk->src = NULL;
2536
    origdisk->type = newdisk->type;
2537 2538 2539
    return 0;
}

2540
static int qemudDomainAttachPciDiskDevice(virDomainPtr dom, virDomainDeviceDefPtr dev)
2541 2542 2543 2544
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
    int ret, i;
2545
    char *cmd, *reply, *s;
2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581 2582 2583 2584 2585 2586 2587 2588 2589 2590 2591 2592
    char *safe_path;
    const char* type = virDomainDiskBusTypeToString(dev->data.disk->bus);

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (STREQ(vm->def->disks[i]->dst, dev->data.disk->dst)) {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                           _("target %s already exists"), dev->data.disk->dst);
            return -1;
        }
    }

    if (VIR_REALLOC_N(vm->def->disks, vm->def->ndisks+1) < 0) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return -1;
    }

    safe_path = qemudEscapeMonitorArg(dev->data.disk->src);
    if (!safe_path) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("out of memory"));
        return -1;
    }

    ret = asprintf(&cmd, "pci_add 0 storage file=%s,if=%s",
                         safe_path, type);
    VIR_FREE(safe_path);
    if (ret == -1) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return ret;
    }

    if (qemudMonitorCommand(driver, vm, cmd, &reply) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         _("cannot attach %s disk"), type);
        VIR_FREE(cmd);
        return -1;
    }

    DEBUG ("pci_add reply: %s", reply);
    /* If the command succeeds qemu prints:
     * OK bus 0... */
2593 2594 2595 2596 2597 2598 2599 2600
#define PCI_ATTACH_OK_MSG "OK bus 0, slot "
    if ((s=strstr(reply, PCI_ATTACH_OK_MSG))) {
        char* dummy = s;
        s += strlen(PCI_ATTACH_OK_MSG);

        if (virStrToLong_i ((const char*)s, &dummy, 10, &dev->data.disk->slotnum) == -1)
            qemudLog(QEMUD_WARN, _("Unable to parse slot number\n"));
    } else {
2601 2602 2603 2604 2605 2606 2607 2608 2609 2610 2611 2612 2613 2614 2615
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          _("adding %s disk failed"), type);
        VIR_FREE(reply);
        VIR_FREE(cmd);
        return -1;
    }

    vm->def->disks[vm->def->ndisks++] = dev->data.disk;
    qsort(vm->def->disks, vm->def->ndisks, sizeof(*vm->def->disks),
          virDomainDiskQSort);

    VIR_FREE(reply);
    VIR_FREE(cmd);
    return 0;
}
2616

2617 2618 2619
static int qemudDomainAttachUsbMassstorageDevice(virDomainPtr dom, virDomainDeviceDefPtr dev)
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2620
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2621 2622
    int ret, i;
    char *safe_path;
2623 2624 2625 2626 2627 2628 2629 2630
    char *cmd, *reply;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

2631 2632 2633 2634 2635 2636 2637 2638
    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (STREQ(vm->def->disks[i]->dst, dev->data.disk->dst)) {
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                           _("target %s already exists"), dev->data.disk->dst);
            return -1;
        }
    }

2639 2640 2641
    if (VIR_REALLOC_N(vm->def->disks, vm->def->ndisks+1) < 0) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return -1;
2642 2643
    }

2644 2645 2646 2647 2648 2649 2650 2651 2652
    safe_path = qemudEscapeMonitorArg(dev->data.disk->src);
    if (!safe_path) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("out of memory"));
        return -1;
    }

    ret = asprintf(&cmd, "usb_add disk:%s", safe_path);
    VIR_FREE(safe_path);
2653 2654 2655 2656 2657 2658 2659
    if (ret == -1) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return ret;
    }

    if (qemudMonitorCommand(driver, vm, cmd, &reply) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
2660
                         "%s", _("cannot attach usb disk"));
2661 2662 2663 2664 2665 2666 2667 2668 2669 2670
        VIR_FREE(cmd);
        return -1;
    }

    DEBUG ("attach_usb reply: %s", reply);
    /* If the command failed qemu prints:
     * Could not add ... */
    if (strstr(reply, "Could not add ")) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          "%s",
2671
                          _("adding usb disk failed"));
2672 2673 2674 2675
        VIR_FREE(reply);
        VIR_FREE(cmd);
        return -1;
    }
2676

2677 2678 2679
    vm->def->disks[vm->def->ndisks++] = dev->data.disk;
    qsort(vm->def->disks, vm->def->ndisks, sizeof(*vm->def->disks),
          virDomainDiskQSort);
2680

2681 2682 2683 2684 2685
    VIR_FREE(reply);
    VIR_FREE(cmd);
    return 0;
}

2686 2687 2688
static int qemudDomainAttachHostDevice(virDomainPtr dom, virDomainDeviceDefPtr dev)
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2689
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2690 2691 2692 2693 2694 2695 2696 2697
    int ret;
    char *cmd, *reply;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }
2698 2699 2700 2701
    if (VIR_REALLOC_N(vm->def->hostdevs, vm->def->nhostdevs+1) < 0) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return -1;
    }
2702 2703 2704 2705 2706 2707 2708 2709 2710 2711 2712 2713 2714 2715 2716 2717 2718 2719 2720 2721 2722 2723 2724 2725 2726 2727 2728 2729 2730 2731 2732 2733 2734

    if (dev->data.hostdev->source.subsys.usb.vendor) {
        ret = asprintf(&cmd, "usb_add host:%.4x:%.4x",
                       dev->data.hostdev->source.subsys.usb.vendor,
                       dev->data.hostdev->source.subsys.usb.product);
    } else {
        ret = asprintf(&cmd, "usb_add host:%.3d.%.3d",
                       dev->data.hostdev->source.subsys.usb.bus,
                       dev->data.hostdev->source.subsys.usb.device);
    }
    if (ret == -1) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return -1;
    }

    if (qemudMonitorCommand(driver, vm, cmd, &reply) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("cannot attach usb device"));
        VIR_FREE(cmd);
        return -1;
    }

    DEBUG ("attach_usb reply: %s", reply);
    /* If the command failed qemu prints:
     * Could not add ... */
    if (strstr(reply, "Could not add ")) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          "%s",
                          _("adding usb device failed"));
        VIR_FREE(reply);
        VIR_FREE(cmd);
        return -1;
    }
2735

2736
    vm->def->hostdevs[vm->def->nhostdevs++] = dev->data.hostdev;
2737

2738 2739
    VIR_FREE(reply);
    VIR_FREE(cmd);
2740 2741 2742
    return 0;
}

2743 2744 2745
static int qemudDomainAttachDevice(virDomainPtr dom,
                                   const char *xml) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2746
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2747
    virDomainDeviceDefPtr dev;
2748
    int ret = 0, supported = 0;
2749 2750 2751 2752 2753 2754 2755 2756 2757 2758 2759 2760 2761

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

    if (!virDomainIsActive(vm)) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot attach device on inactive domain"));
        return -1;
    }

2762 2763 2764
    dev = virDomainDeviceDefParse(dom->conn,
                                  driver->caps,
                                  vm->def, xml);
2765 2766 2767 2768
    if (dev == NULL) {
        return -1;
    }

2769 2770 2771 2772 2773
    if (dev->type == VIR_DOMAIN_DEVICE_DISK) {
        switch (dev->data.disk->device) {
            case VIR_DOMAIN_DISK_DEVICE_CDROM:
            case VIR_DOMAIN_DISK_DEVICE_FLOPPY:
                supported = 1;
2774
                ret = qemudDomainChangeEjectableMedia(dom, dev);
2775 2776 2777 2778 2779 2780 2781 2782
                break;
            case VIR_DOMAIN_DISK_DEVICE_DISK:
                if (dev->data.disk->bus == VIR_DOMAIN_DISK_BUS_USB) {
                    supported = 1;
                    ret = qemudDomainAttachUsbMassstorageDevice(dom, dev);
                } else if (dev->data.disk->bus == VIR_DOMAIN_DISK_BUS_SCSI ||
                           dev->data.disk->bus == VIR_DOMAIN_DISK_BUS_VIRTIO) {
                    supported = 1;
2783
                    ret = qemudDomainAttachPciDiskDevice(dom, dev);
2784 2785 2786
                }
                break;
        }
2787 2788 2789
    } else if (dev->type == VIR_DOMAIN_DEVICE_HOSTDEV &&
        dev->data.hostdev->mode == VIR_DOMAIN_HOSTDEV_MODE_SUBSYS &&
        dev->data.hostdev->source.subsys.type == VIR_DOMAIN_HOSTDEV_SUBSYS_TYPE_USB) {
2790
                supported = 1;
2791
                ret = qemudDomainAttachHostDevice(dom, dev);
2792 2793 2794
    }

    if (!supported) {
2795
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
2796
                         "%s", _("this device type cannot be attached"));
2797 2798 2799 2800 2801 2802 2803
        ret = -1;
    }

    VIR_FREE(dev);
    return ret;
}

2804 2805 2806 2807 2808 2809 2810 2811 2812 2813 2814 2815 2816 2817 2818 2819 2820 2821 2822 2823 2824 2825 2826 2827 2828 2829 2830 2831 2832 2833 2834 2835 2836 2837 2838 2839 2840 2841 2842 2843 2844 2845 2846 2847 2848 2849 2850 2851 2852 2853 2854 2855 2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894 2895 2896 2897 2898 2899 2900 2901 2902 2903 2904 2905 2906 2907 2908 2909 2910 2911 2912 2913 2914 2915 2916 2917 2918 2919 2920
static int qemudDomainDetachPciDiskDevice(virDomainPtr dom, virDomainDeviceDefPtr dev)
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
    int i, ret = -1;
    char *cmd, *reply;
    virDomainDiskDefPtr detach = NULL;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (STREQ(vm->def->disks[i]->dst, dev->data.disk->dst)) {
            detach = vm->def->disks[i];
            break;
        }
    }

    if (!detach) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         _("disk %s not found"), dev->data.disk->dst);
        return -1;
    }

    if (detach->slotnum < 1) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         _("disk %s cannot be detached - invalid slot number %d"), 
                           detach->dst, detach->slotnum);
        return -1;
    }

    ret = asprintf(&cmd, "pci_del 0 %d", detach->slotnum);
    if (ret == -1) {
        qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
        return ret;
    }

    if (qemudMonitorCommand(driver, vm, cmd, &reply) < 0) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          _("failed to execute detach disk %s command"), detach->dst);
        VIR_FREE(cmd);
        return -1;
    }

    DEBUG ("pci_del reply: %s", reply);
    /* If the command fails due to a wrong slot qemu prints: invalid slot,
     * nothing is printed on success */
    if (strstr(reply, "invalid slot")) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          _("failed to detach disk %s: invalid slot %d"), 
                            detach->dst, detach->slotnum);
        ret = -1;
        goto out;
    }

    if (vm->def->ndisks > 1) {
        vm->def->disks[i] = vm->def->disks[--vm->def->ndisks];
        if (VIR_REALLOC_N(vm->def->disks, vm->def->ndisks) < 0) {
            qemudReportError(dom->conn, NULL, NULL, VIR_ERR_NO_MEMORY, NULL);
            ret = -1;
            goto out;
        }
        qsort(vm->def->disks, vm->def->ndisks, sizeof(*vm->def->disks),
              virDomainDiskQSort);
    } else {
        VIR_FREE(vm->def->disks[0]);
        vm->def->ndisks = 0;
    }
    ret = 0;
out:
    VIR_FREE(reply);
    VIR_FREE(cmd);
    return ret;
}

static int qemudDomainDetachDevice(virDomainPtr dom,
                                   const char *xml) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
    virDomainDeviceDefPtr dev;
    int ret = 0;

    if (!vm) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
        return -1;
    }

    if (!virDomainIsActive(vm)) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot attach device on inactive domain"));
        return -1;
    }

    dev = virDomainDeviceDefParse(dom->conn, driver->caps, vm->def, xml);
    if (dev == NULL) {
        return -1;
    }

    if (dev->type == VIR_DOMAIN_DEVICE_DISK &&
        dev->data.disk->device == VIR_DOMAIN_DISK_DEVICE_DISK &&
        (dev->data.disk->bus == VIR_DOMAIN_DISK_BUS_SCSI ||
         dev->data.disk->bus == VIR_DOMAIN_DISK_BUS_VIRTIO))
                    ret = qemudDomainDetachPciDiskDevice(dom, dev);
    else {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                         "%s", _("only SCSI or virtio disk device can be detached dynamically"));
        ret = -1;
    }

    VIR_FREE(dev);
    return ret;
}

2921
static int qemudDomainGetAutostart(virDomainPtr dom,
2922 2923
                            int *autostart) {
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2924
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2925 2926

    if (!vm) {
2927 2928
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
2929 2930 2931 2932 2933 2934 2935 2936
        return -1;
    }

    *autostart = vm->autostart;

    return 0;
}

2937
static int qemudDomainSetAutostart(virDomainPtr dom,
2938
                                   int autostart) {
2939
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
2940
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
2941 2942
    char *configFile = NULL, *autostartLink = NULL;
    int ret = -1;
2943 2944

    if (!vm) {
2945 2946
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                         "%s", _("no domain with matching uuid"));
2947 2948 2949
        return -1;
    }

2950 2951 2952 2953 2954 2955
    if (!vm->persistent) {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
                         "%s", _("cannot set autostart for transient domain"));
        return -1;
    }

2956 2957 2958 2959 2960
    autostart = (autostart != 0);

    if (vm->autostart == autostart)
        return 0;

2961 2962 2963 2964 2965
    if ((configFile = virDomainConfigFile(dom->conn, driver->configDir, vm->def->name)) == NULL)
        goto cleanup;
    if ((autostartLink = virDomainConfigFile(dom->conn, driver->autostartDir, vm->def->name)) == NULL)
        goto cleanup;

2966 2967 2968
    if (autostart) {
        int err;

2969
        if ((err = virFileMakePath(driver->autostartDir))) {
2970
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
2971
                             _("cannot create autostart directory %s: %s"),
2972
                             driver->autostartDir, strerror(err));
2973
            goto cleanup;
2974 2975
        }

2976
        if (symlink(configFile, autostartLink) < 0) {
2977
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
2978 2979 2980
                             _("Failed to create symlink '%s to '%s': %s"),
                             autostartLink, configFile, strerror(errno));
            goto cleanup;
2981 2982
        }
    } else {
2983
        if (unlink(autostartLink) < 0 && errno != ENOENT && errno != ENOTDIR) {
2984
            qemudReportError(dom->conn, dom, NULL, VIR_ERR_INTERNAL_ERROR,
2985
                             _("Failed to delete symlink '%s': %s"),
2986 2987
                             autostartLink, strerror(errno));
            goto cleanup;
2988 2989 2990
        }
    }

2991
    vm->autostart = autostart;
2992
    ret = 0;
2993

2994 2995 2996 2997 2998
cleanup:
    VIR_FREE(configFile);
    VIR_FREE(autostartLink);

    return ret;
2999 3000
}

3001 3002 3003 3004 3005 3006 3007 3008 3009
/* This uses the 'info blockstats' monitor command which was
 * integrated into both qemu & kvm in late 2007.  If the command is
 * not supported we detect this and return the appropriate error.
 */
static int
qemudDomainBlockStats (virDomainPtr dom,
                       const char *path,
                       struct _virDomainBlockStats *stats)
{
3010
    struct qemud_driver *driver =
3011
        (struct qemud_driver *)dom->conn->privateData;
3012
    char *dummy, *info = NULL;
3013
    const char *p, *eol;
3014
    const char *qemu_dev_name = NULL;
3015
    size_t len;
3016
    int i, ret = -1;
3017
    const virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
3018
    virDomainDiskDefPtr disk = NULL;
3019 3020 3021 3022 3023 3024

    if (!vm) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                          _("no domain with matching id %d"), dom->id);
        return -1;
    }
3025
    if (!virDomainIsActive (vm)) {
3026 3027 3028 3029 3030
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          "%s", _("domain is not running"));
        return -1;
    }

3031 3032 3033 3034 3035 3036 3037 3038
    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (STREQ(path, vm->def->disks[i]->dst)) {
            disk = vm->def->disks[i];
            break;
        }
    }

    if (!disk) {
3039 3040 3041 3042 3043
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                          _("invalid path: %s"), path);
        return -1;
    }

3044 3045 3046
    qemu_dev_name = qemudDiskDeviceName(dom, disk);
    if (!qemu_dev_name)
        return -1;
3047 3048 3049 3050 3051
    len = strlen (qemu_dev_name);

    if (qemudMonitorCommand (driver, vm, "info blockstats", &info) < 0) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                          "%s", _("'info blockstats' command failed"));
3052
        goto out;
3053 3054 3055 3056 3057 3058 3059 3060
    }
    DEBUG ("info blockstats reply: %s", info);

    /* If the command isn't supported then qemu prints the supported
     * info commands, so the output starts "info ".  Since this is
     * unlikely to be the name of a block device, we can use this
     * to detect if qemu supports the command.
     */
3061
    if (STRPREFIX (info, "info ")) {
3062 3063 3064
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                          "%s",
                          _("'info blockstats' not supported by this qemu"));
3065
        goto out;
3066 3067 3068 3069 3070 3071 3072 3073 3074 3075 3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091
    }

    stats->rd_req = -1;
    stats->rd_bytes = -1;
    stats->wr_req = -1;
    stats->wr_bytes = -1;
    stats->errs = -1;

    /* The output format for both qemu & KVM is:
     *   blockdevice: rd_bytes=% wr_bytes=% rd_operations=% wr_operations=%
     *   (repeated for each block device)
     * where '%' is a 64 bit number.
     */
    p = info;

    while (*p) {
        if (STREQLEN (p, qemu_dev_name, len)
            && p[len] == ':' && p[len+1] == ' ') {

            eol = strchr (p, '\n');
            if (!eol)
                eol = p + strlen (p);

            p += len+2;         /* Skip to first label. */

            while (*p) {
3092
                if (STRPREFIX (p, "rd_bytes=")) {
3093 3094 3095
                    p += 9;
                    if (virStrToLong_ll (p, &dummy, 10, &stats->rd_bytes) == -1)
                        DEBUG ("error reading rd_bytes: %s", p);
3096
                } else if (STRPREFIX (p, "wr_bytes=")) {
3097 3098 3099
                    p += 9;
                    if (virStrToLong_ll (p, &dummy, 10, &stats->wr_bytes) == -1)
                        DEBUG ("error reading wr_bytes: %s", p);
3100
                } else if (STRPREFIX (p, "rd_operations=")) {
3101 3102 3103
                    p += 14;
                    if (virStrToLong_ll (p, &dummy, 10, &stats->rd_req) == -1)
                        DEBUG ("error reading rd_req: %s", p);
3104
                } else if (STRPREFIX (p, "wr_operations=")) {
3105 3106 3107 3108 3109 3110 3111 3112 3113 3114 3115
                    p += 14;
                    if (virStrToLong_ll (p, &dummy, 10, &stats->wr_req) == -1)
                        DEBUG ("error reading wr_req: %s", p);
                } else
                    DEBUG ("unknown block stat near %s", p);

                /* Skip to next label. */
                p = strchr (p, ' ');
                if (!p || p >= eol) break;
                p++;
            }
3116 3117
            ret = 0;
            goto out;
3118 3119 3120 3121 3122 3123 3124 3125 3126 3127 3128
        }

        /* Skip to next line. */
        p = strchr (p, '\n');
        if (!p) break;
        p++;
    }

    /* If we reach here then the device was not found. */
    qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
                      _("device not found: %s (%s)"), path, qemu_dev_name);
3129 3130 3131 3132
 out:
    VIR_FREE(qemu_dev_name);
    VIR_FREE(info);
    return ret;
3133 3134
}

3135 3136 3137 3138 3139 3140 3141
static int
qemudDomainInterfaceStats (virDomainPtr dom,
                           const char *path,
                           struct _virDomainInterfaceStats *stats)
{
#ifdef __linux__
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
3142
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
3143
    int i;
3144 3145 3146

    if (!vm) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
3147
                          _("no domain with matching id %d"), dom->id);
3148 3149 3150
        return -1;
    }

3151
    if (!virDomainIsActive(vm)) {
3152
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
3153
                         "%s", _("domain is not running"));
3154 3155 3156 3157 3158
        return -1;
    }

    if (!path || path[0] == '\0') {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
3159
                         "%s", _("NULL or empty path"));
3160 3161 3162 3163
        return -1;
    }

    /* Check the path is one of the domain's network interfaces. */
3164 3165 3166
    for (i = 0 ; i < vm->def->nnets ; i++) {
        if (vm->def->nets[i]->ifname &&
            STREQ (vm->def->nets[i]->ifname, path))
3167
            goto ok;
3168 3169 3170
    }

    qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
3171
                      _("invalid path, '%s' is not a known interface"), path);
3172 3173 3174 3175 3176 3177 3178 3179 3180 3181 3182
    return -1;
 ok:

    return linuxDomainInterfaceStats (dom->conn, path, stats);
#else
    qemudReportError (dom->conn, dom, NULL, VIR_ERR_NO_SUPPORT,
                      "%s", __FUNCTION__);
    return -1;
#endif
}

3183 3184 3185 3186 3187 3188 3189 3190
static int
qemudDomainBlockPeek (virDomainPtr dom,
                      const char *path,
                      unsigned long long offset, size_t size,
                      void *buffer,
                      unsigned int flags ATTRIBUTE_UNUSED)
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
3191
    virDomainObjPtr vm = virDomainFindByUUID(&driver->domains, dom->uuid);
3192
    int fd, ret = -1, i;
3193 3194 3195

    if (!vm) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
J
Jim Meyering 已提交
3196
                          "%s", _("no domain with matching uuid"));
3197 3198 3199 3200 3201
        return -1;
    }

    if (!path || path[0] == '\0') {
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
J
Jim Meyering 已提交
3202
                         "%s", _("NULL or empty path"));
3203 3204 3205 3206
        return -1;
    }

    /* Check the path belongs to this domain. */
3207 3208 3209 3210
    for (i = 0 ; i < vm->def->ndisks ; i++) {
        if (vm->def->disks[i]->src != NULL &&
            STREQ (vm->def->disks[i]->src, path))
            goto found;
3211 3212
    }
    qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
J
Jim Meyering 已提交
3213
                      "%s", _("invalid path"));
3214 3215 3216 3217 3218 3219 3220 3221 3222 3223 3224 3225 3226 3227 3228 3229 3230 3231 3232 3233 3234 3235 3236 3237 3238 3239 3240 3241
    return -1;

found:
    /* The path is correct, now try to open it and get its size. */
    fd = open (path, O_RDONLY);
    if (fd == -1) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_SYSTEM_ERROR,
                          "%s", strerror (errno));
        goto done;
    }

    /* Seek and read. */
    /* NB. Because we configure with AC_SYS_LARGEFILE, off_t should
     * be 64 bits on all platforms.
     */
    if (lseek (fd, offset, SEEK_SET) == (off_t) -1 ||
        saferead (fd, buffer, size) == (ssize_t) -1) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_SYSTEM_ERROR,
                          "%s", strerror (errno));
        goto done;
    }

    ret = 0;
 done:
    if (fd >= 0) close (fd);
    return ret;
}

R
Richard W.M. Jones 已提交
3242 3243 3244 3245 3246 3247 3248
static int
qemudDomainMemoryPeek (virDomainPtr dom,
                       unsigned long long offset, size_t size,
                       void *buffer,
                       unsigned int flags)
{
    struct qemud_driver *driver = (struct qemud_driver *)dom->conn->privateData;
3249
    virDomainObjPtr vm = virDomainFindByID(&driver->domains, dom->id);
R
Richard W.M. Jones 已提交
3250 3251 3252 3253 3254 3255
    char cmd[256], *info;
    char tmp[] = TEMPDIR "/qemu.mem.XXXXXX";
    int fd = -1, ret = -1;

    if (flags != VIR_MEMORY_VIRTUAL) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_ARG,
J
Jim Meyering 已提交
3256
                          "%s", _("QEMU driver only supports virtual memory addrs"));
R
Richard W.M. Jones 已提交
3257 3258 3259 3260 3261 3262 3263 3264 3265
        return -1;
    }

    if (!vm) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_INVALID_DOMAIN,
                          _("no domain with matching id %d"), dom->id);
        return -1;
    }

3266
    if (!virDomainIsActive(vm)) {
R
Richard W.M. Jones 已提交
3267 3268 3269 3270 3271 3272 3273 3274 3275 3276 3277 3278 3279 3280 3281 3282
        qemudReportError(dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
                         "%s", _("domain is not running"));
        return -1;
    }

    /* Create a temporary filename. */
    if ((fd = mkstemp (tmp)) == -1) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_SYSTEM_ERROR,
                          "%s", strerror (errno));
        return -1;
    }

    /* Issue the memsave command. */
    snprintf (cmd, sizeof cmd, "memsave %llu %zi \"%s\"", offset, size, tmp);
    if (qemudMonitorCommand (driver, vm, cmd, &info) < 0) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_OPERATION_FAILED,
3283
                          "%s", _("'memsave' command failed"));
R
Richard W.M. Jones 已提交
3284 3285 3286 3287 3288 3289 3290 3291 3292 3293 3294 3295 3296 3297 3298 3299 3300 3301 3302 3303
        goto done;
    }

    DEBUG ("memsave reply: %s", info);
    free (info);

    /* Read the memory file into buffer. */
    if (saferead (fd, buffer, size) == (ssize_t) -1) {
        qemudReportError (dom->conn, dom, NULL, VIR_ERR_SYSTEM_ERROR,
                          "%s", strerror (errno));
        goto done;
    }

    ret = 0;
done:
    if (fd >= 0) close (fd);
    unlink (tmp);
    return ret;
}

3304

3305 3306 3307 3308 3309 3310 3311 3312 3313 3314 3315 3316 3317 3318 3319 3320 3321 3322 3323 3324 3325 3326 3327 3328 3329 3330 3331 3332 3333 3334 3335 3336 3337 3338 3339 3340 3341 3342 3343 3344 3345 3346 3347 3348 3349 3350 3351 3352 3353 3354
static int
qemudDomainEventRegister (virConnectPtr conn,
                          void *callback,
                          void *opaque)
{
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;

    return virDomainEventCallbackListAdd(conn, driver->domainEventCallbacks,
                                         callback, opaque);
}

static int
qemudDomainEventDeregister (virConnectPtr conn,
                            void *callback)
{
    struct qemud_driver *driver = (struct qemud_driver *)conn->privateData;

    return virDomainEventCallbackListRemove(conn, driver->domainEventCallbacks,
                                            callback);
}

static void qemudDomainEventDispatch (struct qemud_driver *driver,
                                      virDomainObjPtr vm,
                                      virDomainEventType evt)
{
    int i;
    virDomainEventCallbackListPtr cbList;

    cbList = driver->domainEventCallbacks;

    for(i=0 ; i < cbList->count ; i++) {
        if(cbList->callbacks[i] && cbList->callbacks[i]->cb) {
            virConnectPtr conn = cbList->callbacks[i]->conn;
            virDomainPtr dom = virGetDomain(conn, vm->def->name,
                                            vm->def->uuid);
            if (dom) {
                dom->id = virDomainIsActive(vm) ? vm->def->id : -1;
                DEBUG("Dispatching callback %p %p event %d",
                        cbList->callbacks[i],
                        cbList->callbacks[i]->cb, evt);
                cbList->callbacks[i]->cb(cbList->callbacks[i]->conn,
                                         dom, evt,
                                         cbList->callbacks[i]->opaque);
                virDomainFree(dom);
            }
        }
    }

}

3355 3356 3357 3358
static virDriver qemuDriver = {
    VIR_DRV_QEMU,
    "QEMU",
    LIBVIR_VERSION_NUMBER,
3359
    qemudProbe, /* probe */
3360 3361
    qemudOpen, /* open */
    qemudClose, /* close */
3362
    NULL, /* supports_feature */
3363 3364
    qemudGetType, /* type */
    qemudGetVersion, /* version */
3365
    qemudGetHostname, /* hostname */
3366
    NULL, /* URI  */
3367 3368 3369 3370 3371
    qemudGetMaxVCPUs, /* getMaxVcpus */
    qemudGetNodeInfo, /* nodeGetInfo */
    qemudGetCapabilities, /* getCapabilities */
    qemudListDomains, /* listDomains */
    qemudNumDomains, /* numOfDomains */
3372
    qemudDomainCreate, /* domainCreateXML */
3373 3374 3375 3376 3377
    qemudDomainLookupByID, /* domainLookupByID */
    qemudDomainLookupByUUID, /* domainLookupByUUID */
    qemudDomainLookupByName, /* domainLookupByName */
    qemudDomainSuspend, /* domainSuspend */
    qemudDomainResume, /* domainResume */
3378
    qemudDomainShutdown, /* domainShutdown */
3379 3380 3381
    NULL, /* domainReboot */
    qemudDomainDestroy, /* domainDestroy */
    qemudDomainGetOSType, /* domainGetOSType */
3382 3383 3384
    qemudDomainGetMaxMemory, /* domainGetMaxMemory */
    qemudDomainSetMaxMemory, /* domainSetMaxMemory */
    qemudDomainSetMemory, /* domainSetMemory */
3385 3386 3387 3388
    qemudDomainGetInfo, /* domainGetInfo */
    qemudDomainSave, /* domainSave */
    qemudDomainRestore, /* domainRestore */
    NULL, /* domainCoreDump */
3389
    qemudDomainSetVcpus, /* domainSetVcpus */
3390 3391 3392 3393
#if HAVE_SCHED_GETAFFINITY
    qemudDomainPinVcpu, /* domainPinVcpu */
    qemudDomainGetVcpus, /* domainGetVcpus */
#else
3394 3395
    NULL, /* domainPinVcpu */
    NULL, /* domainGetVcpus */
3396
#endif
3397
    qemudDomainGetMaxVcpus, /* domainGetMaxVcpus */
3398 3399 3400 3401 3402 3403
    qemudDomainDumpXML, /* domainDumpXML */
    qemudListDefinedDomains, /* listDomains */
    qemudNumDefinedDomains, /* numOfDomains */
    qemudDomainStart, /* domainCreate */
    qemudDomainDefine, /* domainDefineXML */
    qemudDomainUndefine, /* domainUndefine */
3404
    qemudDomainAttachDevice, /* domainAttachDevice */
3405
    qemudDomainDetachDevice, /* domainDetachDevice */
3406 3407 3408 3409 3410
    qemudDomainGetAutostart, /* domainGetAutostart */
    qemudDomainSetAutostart, /* domainSetAutostart */
    NULL, /* domainGetSchedulerType */
    NULL, /* domainGetSchedulerParameters */
    NULL, /* domainSetSchedulerParameters */
3411 3412 3413
    NULL, /* domainMigratePrepare */
    NULL, /* domainMigratePerform */
    NULL, /* domainMigrateFinish */
3414
    qemudDomainBlockStats, /* domainBlockStats */
3415
    qemudDomainInterfaceStats, /* domainInterfaceStats */
3416
    qemudDomainBlockPeek, /* domainBlockPeek */
R
Richard W.M. Jones 已提交
3417
    qemudDomainMemoryPeek, /* domainMemoryPeek */
3418 3419 3420 3421
#if HAVE_NUMACTL
    qemudNodeGetCellsFreeMemory, /* nodeGetCellsFreeMemory */
    qemudNodeGetFreeMemory,  /* getFreeMemory */
#else
3422
    NULL, /* nodeGetCellsFreeMemory */
3423
    NULL, /* getFreeMemory */
3424
#endif
3425 3426
    qemudDomainEventRegister, /* domainEventRegister */
    qemudDomainEventDeregister, /* domainEventDeregister */
3427 3428 3429
};


3430
static virStateDriver qemuStateDriver = {
3431 3432 3433 3434
    .initialize = qemudStartup,
    .cleanup = qemudShutdown,
    .reload = qemudReload,
    .active = qemudActive,
3435
};
3436

3437 3438 3439 3440 3441 3442
int qemudRegister(void) {
    virRegisterDriver(&qemuDriver);
    virRegisterStateDriver(&qemuStateDriver);
    return 0;
}