vmware_conf.c 14.4 KB
Newer Older
1
/*---------------------------------------------------------------------------*/
2
/*
3
 * Copyright (C) 2011-2014 Red Hat, Inc.
4
 * Copyright (C) 2010-2014, diateam (www.diateam.net)
5 6 7 8 9 10 11 12 13 14 15 16
 *
 * 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
17
 * License along with this library.  If not, see
O
Osier Yang 已提交
18
 * <http://www.gnu.org/licenses/>.
19 20 21 22 23 24 25
 */
/*---------------------------------------------------------------------------*/

#include <config.h>

#include <string.h>

26
#include "vircommand.h"
27 28
#include "cpu/cpu.h"
#include "dirname.h"
29
#include "viralloc.h"
30
#include "nodeinfo.h"
E
Eric Blake 已提交
31
#include "virfile.h"
32
#include "viruuid.h"
33
#include "virerror.h"
34
#include "vmx.h"
35
#include "vmware_conf.h"
36
#include "virstring.h"
37

38 39
VIR_ENUM_IMPL(vmwareDriver, VMWARE_DRIVER_LAST,
              "player",
40 41
              "ws",
              "fusion");
42

43 44 45 46 47 48 49 50
/* Free all memory associated with a vmware_driver structure */
void
vmwareFreeDriver(struct vmware_driver *driver)
{
    if (!driver)
        return;

    virMutexDestroy(&driver->lock);
51
    virObjectUnref(driver->domains);
52
    virObjectUnref(driver->caps);
53
    virObjectUnref(driver->xmlopt);
54
    VIR_FREE(driver->vmrun);
55 56 57
    VIR_FREE(driver);
}

58

59 60 61 62 63 64
virCapsPtr
vmwareCapsInit(void)
{
    virCapsPtr caps = NULL;
    virCapsGuestPtr guest = NULL;
    virCPUDefPtr cpu = NULL;
65
    virCPUDataPtr data = NULL;
66

67
    if ((caps = virCapabilitiesNew(virArchFromHost(),
68
                                   false, false)) == NULL)
69 70
        goto error;

71
    if (nodeCapsInitNUMA(NULL, caps) < 0)
72 73 74 75
        goto error;

    /* i686 guests are always supported */
    if ((guest = virCapabilitiesAddGuest(caps,
76
                                         VIR_DOMAIN_OSTYPE_HVM,
77
                                         VIR_ARCH_I686,
78 79 80 81
                                         NULL, NULL, 0, NULL)) == NULL)
        goto error;

    if (virCapabilitiesAddGuestDomain(guest,
82
                                      VIR_DOMAIN_VIRT_VMWARE,
83 84 85
                                      NULL, NULL, 0, NULL) == NULL)
        goto error;

86
    if (VIR_ALLOC(cpu) < 0)
87 88
        goto error;

J
Jiri Denemark 已提交
89
    cpu->arch = caps->host.arch;
90 91 92 93 94 95 96 97 98 99 100 101
    cpu->type = VIR_CPU_TYPE_HOST;

    if (!(data = cpuNodeData(cpu->arch))
        || cpuDecode(cpu, data, NULL, 0, NULL) < 0) {
        goto error;
    }

    /* x86_64 guests are supported if
     *  - Host arch is x86_64
     * Or
     *  - Host CPU is x86_64 with virtualization extensions
     */
102
    if (caps->host.arch == VIR_ARCH_X86_64 ||
J
Jiri Denemark 已提交
103 104 105
        (cpuHasFeature(data, "lm") &&
         (cpuHasFeature(data, "vmx") ||
          cpuHasFeature(data, "svm")))) {
106 107

        if ((guest = virCapabilitiesAddGuest(caps,
108
                                             VIR_DOMAIN_OSTYPE_HVM,
109
                                             VIR_ARCH_X86_64,
110 111 112 113
                                             NULL, NULL, 0, NULL)) == NULL)
            goto error;

        if (virCapabilitiesAddGuestDomain(guest,
114
                                          VIR_DOMAIN_VIRT_VMWARE,
115 116 117 118
                                          NULL, NULL, 0, NULL) == NULL)
            goto error;
    }

119
 cleanup:
120
    virCPUDefFree(cpu);
J
Jiri Denemark 已提交
121
    cpuDataFree(data);
122 123 124

    return caps;

125
 error:
126
    virObjectUnref(caps);
127 128 129 130 131 132 133 134 135 136 137 138 139 140
    goto cleanup;
}

int
vmwareLoadDomains(struct vmware_driver *driver)
{
    virDomainDefPtr vmdef = NULL;
    virDomainObjPtr vm = NULL;
    char *vmxPath = NULL;
    char *vmx = NULL;
    vmwareDomainPtr pDomain;
    char *directoryName = NULL;
    char *fileName = NULL;
    int ret = -1;
141
    virVMXContext ctx;
142 143 144 145 146
    char *outbuf = NULL;
    char *str;
    char *saveptr = NULL;
    virCommandPtr cmd;

147
    ctx.parseFileName = vmwareCopyVMXFileName;
148 149 150
    ctx.formatFileName = NULL;
    ctx.autodetectSCSIControllerModel = NULL;
    ctx.datacenterPath = NULL;
151

152
    cmd = virCommandNewArgList(driver->vmrun, "-T",
153
                               vmwareDriverTypeToString(driver->type),
154 155 156 157 158
                               "list", NULL);
    virCommandSetOutputBuffer(cmd, &outbuf);
    if (virCommandRun(cmd, NULL) < 0)
        goto cleanup;

159
    for (str = outbuf; (vmxPath = strtok_r(str, "\n", &saveptr)) != NULL;
160 161 162 163 164 165 166 167 168
        str = NULL) {

        if (vmxPath[0] != '/')
            continue;

        if (virFileReadAll(vmxPath, 10000, &vmx) < 0)
            goto cleanup;

        if ((vmdef =
169 170
             virVMXParseConfig(&ctx, driver->xmlopt,
                               driver->caps, vmx)) == NULL) {
171 172 173
            goto cleanup;
        }

174
        if (!(vm = virDomainObjListAdd(driver->domains, vmdef,
175
                                       driver->xmlopt,
176
                                       0, NULL)))
177 178 179 180
            goto cleanup;

        pDomain = vm->privateData;

181
        if (VIR_STRDUP(pDomain->vmxPath, vmxPath) < 0)
182 183 184 185 186 187 188
            goto cleanup;

        vmwareDomainConfigDisplay(pDomain, vmdef);

        if ((vm->def->id = vmwareExtractPid(vmxPath)) < 0)
            goto cleanup;
        /* vmrun list only reports running vms */
J
Jiri Denemark 已提交
189 190
        virDomainObjSetState(vm, VIR_DOMAIN_RUNNING,
                             VIR_DOMAIN_RUNNING_UNKNOWN);
191 192
        vm->persistent = 1;

193
        virObjectUnlock(vm);
194 195 196 197 198 199 200

        vmdef = NULL;
        vm = NULL;
    }

    ret = 0;

201
 cleanup:
202 203 204 205 206 207
    virCommandFree(cmd);
    VIR_FREE(outbuf);
    virDomainDefFree(vmdef);
    VIR_FREE(directoryName);
    VIR_FREE(fileName);
    VIR_FREE(vmx);
208
    virObjectUnref(vm);
209 210 211 212 213 214 215 216 217
    return ret;
}

void
vmwareSetSentinal(const char **prog, const char *key)
{
    const char **tmp = prog;

    while (tmp && *tmp) {
E
Eric Blake 已提交
218
        if (*tmp == PROGRAM_SENTINEL) {
219 220 221 222 223 224 225
            *tmp = key;
            break;
        }
        tmp++;
    }
}

226 227 228 229 230 231 232 233 234 235 236 237 238
int
vmwareParseVersionStr(int type, const char *verbuf, unsigned long *version)
{
    const char *pattern;
    const char *tmp;

    switch (type) {
        case VMWARE_DRIVER_PLAYER:
            pattern = "VMware Player ";
            break;
        case VMWARE_DRIVER_WORKSTATION:
            pattern = "VMware Workstation ";
            break;
239 240 241
        case VMWARE_DRIVER_FUSION:
            pattern = "\nVMware Fusion Information:\nVMware Fusion ";
            break;
242 243 244 245 246 247
        default:
            virReportError(VIR_ERR_INTERNAL_ERROR,
                           _("Invalid driver type: %d"), type);
            return -1;
    }

248 249 250 251 252 253 254
    if ((tmp = strstr(verbuf, pattern)) == NULL) {
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("cannot find version pattern \"%s\""), pattern);
        return -1;
    }

    if ((tmp = STRSKIP(tmp, pattern)) == NULL) {
255 256 257 258 259 260 261 262 263 264 265 266 267 268
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("failed to parse %sversion"), pattern);
        return -1;
    }

    if (virParseVersionString(tmp, version, false) < 0) {
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("version parsing error"));
        return -1;
    }

    return 0;
}

269 270 271 272
int
vmwareExtractVersion(struct vmware_driver *driver)
{
    int ret = -1;
273
    virCommandPtr cmd = NULL;
274
    char * outbuf = NULL;
275 276 277 278 279 280 281 282
    char *bin = NULL;
    char *vmwarePath = NULL;

    if ((vmwarePath = mdir_name(driver->vmrun)) == NULL)
        goto cleanup;

    switch (driver->type) {
        case VMWARE_DRIVER_PLAYER:
283
            if (virAsprintf(&bin, "%s/%s", vmwarePath, "vmplayer") < 0)
284 285 286 287
                goto cleanup;
            break;

        case VMWARE_DRIVER_WORKSTATION:
288
            if (virAsprintf(&bin, "%s/%s", vmwarePath, "vmware") < 0)
289 290 291
                goto cleanup;
            break;

292
        case VMWARE_DRIVER_FUSION:
293
            if (virAsprintf(&bin, "%s/%s", vmwarePath, "vmware-vmx") < 0)
294 295 296
                goto cleanup;
            break;

297 298
        default:
            virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
299
                           _("invalid driver type for version detection"));
300 301
            goto cleanup;
    }
302 303 304

    cmd = virCommandNewArgList(bin, "-v", NULL);
    virCommandSetOutputBuffer(cmd, &outbuf);
305
    virCommandSetErrorBuffer(cmd, &outbuf);
306 307 308 309

    if (virCommandRun(cmd, NULL) < 0)
        goto cleanup;

310
    if (vmwareParseVersionStr(driver->type, outbuf, &driver->version) < 0)
311 312 313 314
        goto cleanup;

    ret = 0;

315
 cleanup:
316 317
    virCommandFree(cmd);
    VIR_FREE(outbuf);
318 319
    VIR_FREE(bin);
    VIR_FREE(vmwarePath);
320 321 322 323 324 325
    return ret;
}

int
vmwareDomainConfigDisplay(vmwareDomainPtr pDomain, virDomainDefPtr def)
{
326
    size_t i;
327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342

    if (def->ngraphics == 0) {
        pDomain->gui = true;
        return 0;
    } else {
        pDomain->gui = false;
        for (i = 0; i < def->ngraphics; i++) {
            if (def->graphics[i]->type == VIR_DOMAIN_GRAPHICS_TYPE_DESKTOP) {
                pDomain->gui = true;
                return 0;
            }
        }
        return 0;
    }
}

E
Eric Blake 已提交
343 344
static int
vmwareParsePath(const char *path, char **directory, char **filename)
345 346 347 348 349 350
{
    char *separator;

    separator = strrchr(path, '/');

    if (separator != NULL) {
E
Eric Blake 已提交
351
        separator++;
352 353

        if (*separator == '\0') {
354 355
            virReportError(VIR_ERR_INTERNAL_ERROR,
                           _("path '%s' doesn't reference a file"), path);
356 357 358
            return -1;
        }

E
Eric Blake 已提交
359
        if (VIR_STRNDUP(*directory, path, separator - path - 1) < 0)
360 361
            goto error;
        if (VIR_STRDUP(*filename, separator) < 0) {
362
            VIR_FREE(*directory);
363
            goto error;
364 365 366
        }

    } else {
367 368
        if (VIR_STRDUP(*filename, path) < 0)
            goto error;
369 370 371 372
    }

    return 0;

373
 error:
374 375 376 377 378 379
    return -1;
}

int
vmwareConstructVmxPath(char *directoryName, char *name, char **vmxPath)
{
380 381
    int ret;

382
    if (directoryName != NULL)
383 384 385 386 387 388 389
        ret = virAsprintf(vmxPath, "%s/%s.vmx", directoryName, name);
    else
        ret = virAsprintf(vmxPath, "%s.vmx", name);

    if (ret < 0)
        return -1;
    return 0;
390 391 392 393 394 395 396 397 398
}

int
vmwareVmxPath(virDomainDefPtr vmdef, char **vmxPath)
{
    virDomainDiskDefPtr disk = NULL;
    char *directoryName = NULL;
    char *fileName = NULL;
    int ret = -1;
399
    size_t i;
400
    const char *src;
401 402 403 404

    /*
     * Build VMX URL. Use the source of the first file-based harddisk
     * to deduce the path for the VMX file. Don't just use the
N
Nehal J Wani 已提交
405
     * first disk, because it may be CDROM disk and ISO images are normally not
406 407 408 409
     * located in the virtual machine's directory. This approach
     * isn't perfect but should work in the majority of cases.
     */
    if (vmdef->ndisks < 1) {
410 411 412
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Domain XML doesn't contain any disks, "
                         "cannot deduce datastore and path for VMX file"));
413 414 415 416 417
        goto cleanup;
    }

    for (i = 0; i < vmdef->ndisks; ++i) {
        if (vmdef->disks[i]->device == VIR_DOMAIN_DISK_DEVICE_DISK &&
E
Eric Blake 已提交
418
            virDomainDiskGetType(vmdef->disks[i]) == VIR_STORAGE_TYPE_FILE) {
419 420 421 422 423 424
            disk = vmdef->disks[i];
            break;
        }
    }

    if (disk == NULL) {
425 426 427
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("Domain XML doesn't contain any file-based harddisks, "
                         "cannot deduce datastore and path for VMX file"));
428 429 430
        goto cleanup;
    }

431 432
    src = virDomainDiskGetSource(disk);
    if (!src) {
433 434 435
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("First file-based harddisk has no source, cannot "
                         "deduce datastore and path for VMX file"));
436 437 438
        goto cleanup;
    }

439
    if (vmwareParsePath(src, &directoryName, &fileName) < 0)
440 441 442
        goto cleanup;

    if (!virFileHasSuffix(fileName, ".vmdk")) {
443 444
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Expecting source '%s' of first file-based harddisk "
445
                         "to be a VMDK image"), src);
446 447 448
        goto cleanup;
    }

449
    if (vmwareConstructVmxPath(directoryName, vmdef->name, vmxPath) < 0)
450 451 452 453
        goto cleanup;

    ret = 0;

454
 cleanup:
455 456 457 458 459 460 461 462 463
    VIR_FREE(directoryName);
    VIR_FREE(fileName);
    return ret;
}

int
vmwareMoveFile(char *srcFile, char *dstFile)
{
    const char *cmdmv[] =
E
Eric Blake 已提交
464
        { "mv", PROGRAM_SENTINEL, PROGRAM_SENTINEL, NULL };
465 466

    if (!virFileExists(srcFile)) {
467 468
        virReportError(VIR_ERR_INTERNAL_ERROR, _("file %s does not exist"),
                       srcFile);
469 470 471 472 473 474 475 476 477
        return -1;
    }

    if (STREQ(srcFile, dstFile))
        return 0;

    vmwareSetSentinal(cmdmv, srcFile);
    vmwareSetSentinal(cmdmv, dstFile);
    if (virRun(cmdmv, NULL) < 0) {
478 479
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("failed to move file to %s "), dstFile);
480 481 482 483 484 485 486 487 488
        return -1;
    }

    return 0;
}

int
vmwareMakePath(char *srcDir, char *srcName, char *srcExt, char **outpath)
{
489 490 491
    if (virAsprintf(outpath, "%s/%s.%s", srcDir, srcName, srcExt) < 0)
        return -1;
    return 0;
492 493 494 495 496 497 498 499 500 501
}

int
vmwareExtractPid(const char * vmxPath)
{
    char *vmxDir = NULL;
    char *logFilePath = NULL;
    FILE *logFile = NULL;
    char line[1024];
    char *tmp = NULL;
502
    int pid_value = -1;
503 504 505 506 507

    if ((vmxDir = mdir_name(vmxPath)) == NULL)
        goto cleanup;

    if (virAsprintf(&logFilePath, "%s/vmware.log",
508
                    vmxDir) < 0)
509 510 511 512 513 514
        goto cleanup;

    if ((logFile = fopen(logFilePath, "r")) == NULL)
        goto cleanup;

    if (!fgets(line, sizeof(line), logFile)) {
515 516
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("unable to read vmware log file"));
517 518 519 520
        goto cleanup;
    }

    if ((tmp = strstr(line, " pid=")) == NULL) {
521 522
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("cannot find pid in vmware log file"));
523 524 525 526 527
        goto cleanup;
    }

    tmp += strlen(" pid=");

528 529 530
    /* Although 64-bit windows allows 64-bit pid_t, a domain id has to be
     * 32 bits.  For now, we just reject pid values that overflow int.  */
    if (virStrToLong_i(tmp, &tmp, 10, &pid_value) < 0 || *tmp != ' ') {
531 532
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("cannot parse pid in vmware log file"));
533 534 535
        goto cleanup;
    }

536
 cleanup:
537 538 539
    VIR_FREE(vmxDir);
    VIR_FREE(logFilePath);
    VIR_FORCE_FCLOSE(logFile);
540
    return pid_value;
541 542 543
}

char *
544
vmwareCopyVMXFileName(const char *datastorePath, void *opaque ATTRIBUTE_UNUSED)
545
{
546
    char *path;
547

548
    ignore_value(VIR_STRDUP(path, datastorePath));
549 550
    return path;
}