domaincapstest.c 17.5 KB
Newer Older
M
Michal Privoznik 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
/*
 * Copyright (C) Red Hat, Inc. 2014
 *
 * 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, see
 * <http://www.gnu.org/licenses/>.
 *
 * Authors:
 *      Michal Privoznik <mprivozn@redhat.com>
 */

#include <config.h>
#include <stdlib.h>

#include "testutils.h"
#include "domain_capabilities.h"


#define VIR_FROM_THIS VIR_FROM_NONE

31 32
typedef int (*virDomainCapsFill)(virDomainCapsPtr domCaps,
                                 void *opaque);
M
Michal Privoznik 已提交
33 34 35 36

#define SET_ALL_BITS(x) \
    memset(&(x.values), 0xff, sizeof(x.values))

37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
static int ATTRIBUTE_SENTINEL
fillStringValues(virDomainCapsStringValuesPtr values, ...)
{
    int ret = 0;
    va_list list;
    const char *str;

    va_start(list, values);
    while ((str = va_arg(list, const char *))) {
        if (VIR_REALLOC_N(values->values, values->nvalues + 1) < 0 ||
            VIR_STRDUP(values->values[values->nvalues], str) < 0) {
            ret = -1;
            break;
        }
        values->nvalues++;
    }
    va_end(list);

    return ret;
}

58
static int
J
Jiri Denemark 已提交
59
fillAllCaps(virDomainCapsPtr domCaps)
M
Michal Privoznik 已提交
60
{
61 62
    virDomainCapsOSPtr os = &domCaps->os;
    virDomainCapsLoaderPtr loader = &os->loader;
63
    virDomainCapsCPUPtr cpu = &domCaps->cpu;
M
Michal Privoznik 已提交
64
    virDomainCapsDeviceDiskPtr disk = &domCaps->disk;
65
    virDomainCapsDeviceGraphicsPtr graphics = &domCaps->graphics;
66
    virDomainCapsDeviceVideoPtr video = &domCaps->video;
M
Michal Privoznik 已提交
67
    virDomainCapsDeviceHostdevPtr hostdev = &domCaps->hostdev;
68
    virCPUDef host = {
69 70 71 72
        .type = VIR_CPU_TYPE_HOST,
        .arch = VIR_ARCH_X86_64,
        .model = (char *) "host",
        .vendor = (char *) "CPU Vendorrr",
73
    };
M
Michal Privoznik 已提交
74

75
    domCaps->maxvcpus = 255;
76
    os->supported = true;
77

78
    loader->supported = true;
79 80
    SET_ALL_BITS(loader->type);
    SET_ALL_BITS(loader->readonly);
81 82 83 84 85
    if (fillStringValues(&loader->values,
                         "/foo/bar",
                         "/tmp/my_path",
                         NULL) < 0)
        return -1;
86

87
    cpu->hostPassthrough = true;
88
    cpu->hostModel = virCPUDefCopy(&host);
89
    if (!(cpu->custom = virDomainCapsCPUModelsNew(3)) ||
J
Jiri Denemark 已提交
90
        virDomainCapsCPUModelsAdd(cpu->custom, "Model1", -1,
91
                                  VIR_DOMCAPS_CPU_USABLE_UNKNOWN, NULL) < 0 ||
J
Jiri Denemark 已提交
92
        virDomainCapsCPUModelsAdd(cpu->custom, "Model2", -1,
93
                                  VIR_DOMCAPS_CPU_USABLE_NO, NULL) < 0 ||
J
Jiri Denemark 已提交
94
        virDomainCapsCPUModelsAdd(cpu->custom, "Model3", -1,
95
                                  VIR_DOMCAPS_CPU_USABLE_YES, NULL) < 0)
96 97
        return -1;

98
    disk->supported = true;
M
Michal Privoznik 已提交
99 100 101
    SET_ALL_BITS(disk->diskDevice);
    SET_ALL_BITS(disk->bus);

102 103 104
    graphics->supported = true;
    SET_ALL_BITS(graphics->type);

105 106 107
    video->supported = true;
    SET_ALL_BITS(video->modelType);

108
    hostdev->supported = true;
M
Michal Privoznik 已提交
109 110 111 112 113
    SET_ALL_BITS(hostdev->mode);
    SET_ALL_BITS(hostdev->startupPolicy);
    SET_ALL_BITS(hostdev->subsysType);
    SET_ALL_BITS(hostdev->capsType);
    SET_ALL_BITS(hostdev->pciBackend);
114
    return 0;
M
Michal Privoznik 已提交
115 116
}

117

J
Jiri Denemark 已提交
118
#if WITH_QEMU
119
# include "testutilsqemu.h"
P
Pavel Hrdina 已提交
120
# include "testutilshostcpus.h"
121

122 123 124 125 126 127
static int
fakeHostCPU(virCapsPtr caps,
            virArch arch)
{
    virCPUDefPtr cpu;

P
Pavel Hrdina 已提交
128
    if (!(cpu = testUtilsHostCpusGetDefForArch(arch))) {
129 130 131 132 133 134
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "cannot fake host CPU for arch %s",
                       virArchToString(arch));
        return -1;
    }

P
Pavel Hrdina 已提交
135
    qemuTestSetHostCPU(caps, cpu);
136 137 138 139

    return 0;
}

140
static int
141
fillQemuCaps(virDomainCapsPtr domCaps,
J
Jiri Denemark 已提交
142
             const char *name,
J
Jiri Denemark 已提交
143
             const char *arch,
144
             const char *machine,
J
Jiri Denemark 已提交
145
             virQEMUDriverConfigPtr cfg)
146
{
J
Jiri Denemark 已提交
147 148
    int ret = -1;
    char *path = NULL;
149
    virCapsPtr caps = NULL;
J
Jiri Denemark 已提交
150
    virQEMUCapsPtr qemuCaps = NULL;
151
    virDomainCapsLoaderPtr loader = &domCaps->os.loader;
152

153 154 155 156
    if (!(caps = virCapabilitiesNew(domCaps->arch, false, false)) ||
        fakeHostCPU(caps, domCaps->arch) < 0)
        goto cleanup;

157
    if (virAsprintf(&path, "%s/qemucapabilitiesdata/%s.%s.xml",
J
Jiri Denemark 已提交
158
                    abs_srcdir, name, arch) < 0 ||
159
        !(qemuCaps = qemuTestParseCapabilities(caps, path)))
J
Jiri Denemark 已提交
160 161
        goto cleanup;

162 163 164 165 166 167 168 169 170 171
    if (machine &&
        VIR_STRDUP(domCaps->machine,
                   virQEMUCapsGetCanonicalMachine(qemuCaps, machine)) < 0)
        goto cleanup;

    if (!domCaps->machine &&
        VIR_STRDUP(domCaps->machine,
                   virQEMUCapsGetDefaultMachine(qemuCaps)) < 0)
        goto cleanup;

172
    if (virQEMUCapsFillDomainCaps(caps, domCaps, qemuCaps,
173
                                  cfg->firmwares,
174
                                  cfg->nfirmwares) < 0)
J
Jiri Denemark 已提交
175
        goto cleanup;
176 177 178 179 180 181 182 183 184 185

    /* The function above tries to query host's KVM & VFIO capabilities by
     * calling qemuHostdevHostSupportsPassthroughLegacy() and
     * qemuHostdevHostSupportsPassthroughVFIO() which, however, can't be
     * successfully mocked as they are not exposed as internal APIs. Therefore,
     * instead of mocking set the expected values here by hand. */
    VIR_DOMAIN_CAPS_ENUM_SET(domCaps->hostdev.pciBackend,
                             VIR_DOMAIN_HOSTDEV_PCI_BACKEND_DEFAULT,
                             VIR_DOMAIN_HOSTDEV_PCI_BACKEND_KVM,
                             VIR_DOMAIN_HOSTDEV_PCI_BACKEND_VFIO);
186

J
Jiri Denemark 已提交
187 188
    /* As of f05b6a918e28 we are expecting to see OVMF_CODE.fd file which
     * may not exists everywhere. */
189 190 191 192
    while (loader->values.nvalues)
        VIR_FREE(loader->values.values[--loader->values.nvalues]);

    if (fillStringValues(&loader->values,
193
                         "/usr/share/AAVMF/AAVMF_CODE.fd",
194
                         "/usr/share/AAVMF/AAVMF32_CODE.fd",
195 196
                         "/usr/share/OVMF/OVMF_CODE.fd",
                         NULL) < 0)
M
Michal Privoznik 已提交
197 198
        goto cleanup;

J
Jiri Denemark 已提交
199
    ret = 0;
M
Michal Privoznik 已提交
200
 cleanup:
201
    virObjectUnref(caps);
J
Jiri Denemark 已提交
202 203
    virObjectUnref(qemuCaps);
    VIR_FREE(path);
204
    return ret;
M
Michal Privoznik 已提交
205
}
J
Jiri Denemark 已提交
206 207 208
#endif /* WITH_QEMU */


209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
#ifdef WITH_LIBXL
# include "testutilsxen.h"

static int
fillXenCaps(virDomainCapsPtr domCaps)
{
    virFirmwarePtr *firmwares;
    int ret = -1;

    if (VIR_ALLOC_N(firmwares, 2) < 0)
        return ret;

    if (VIR_ALLOC(firmwares[0]) < 0 || VIR_ALLOC(firmwares[1]) < 0)
        goto cleanup;
    if (VIR_STRDUP(firmwares[0]->name, "/usr/lib/xen/boot/hvmloader") < 0 ||
        VIR_STRDUP(firmwares[1]->name, "/usr/lib/xen/boot/ovmf.bin") < 0)
        goto cleanup;

    if (libxlMakeDomainCapabilities(domCaps, firmwares, 2) < 0)
        goto cleanup;

    ret = 0;

 cleanup:
    virFirmwareFreeList(firmwares, 2);
    return ret;
}
#endif /* WITH_LIBXL */

238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261
#ifdef WITH_BHYVE
# include "bhyve/bhyve_capabilities.h"

static int
fillBhyveCaps(virDomainCapsPtr domCaps, unsigned int *bhyve_caps)
{
    virDomainCapsStringValuesPtr firmwares = NULL;
    int ret = -1;

    if (VIR_ALLOC(firmwares) < 0)
        return -1;

    if (fillStringValues(firmwares, "/foo/bar", "/foo/baz", NULL) < 0)
        goto cleanup;

    if (virBhyveDomainCapsFill(domCaps, *bhyve_caps, firmwares) < 0)
        goto cleanup;

    ret = 0;
 cleanup:
    VIR_FREE(firmwares);
    return ret;
}
#endif /* WITH_BHYVE */
262

J
Jiri Denemark 已提交
263 264 265 266
enum testCapsType {
    CAPS_NONE,
    CAPS_ALL,
    CAPS_QEMU,
267
    CAPS_LIBXL,
268
    CAPS_BHYVE,
J
Jiri Denemark 已提交
269
};
M
Michal Privoznik 已提交
270

J
Jiri Denemark 已提交
271 272 273
struct testData {
    const char *name;
    const char *emulator;
M
Michal Privoznik 已提交
274
    const char *machine;
J
Jiri Denemark 已提交
275
    const char *arch;
M
Michal Privoznik 已提交
276
    virDomainVirtType type;
J
Jiri Denemark 已提交
277 278 279
    enum testCapsType capsType;
    const char *capsName;
    void *capsOpaque;
M
Michal Privoznik 已提交
280 281 282 283 284
};

static int
test_virDomainCapsFormat(const void *opaque)
{
J
Jiri Denemark 已提交
285
    const struct testData *data = opaque;
M
Michal Privoznik 已提交
286 287 288 289 290
    virDomainCapsPtr domCaps = NULL;
    char *path = NULL;
    char *domCapsXML = NULL;
    int ret = -1;

291
    if (virAsprintf(&path, "%s/domaincapsschemadata/%s.xml",
J
Jiri Denemark 已提交
292
                    abs_srcdir, data->name) < 0)
M
Michal Privoznik 已提交
293 294
        goto cleanup;

J
Jiri Denemark 已提交
295 296
    if (!(domCaps = virDomainCapsNew(data->emulator, data->machine,
                                     virArchFromString(data->arch),
J
Jiri Denemark 已提交
297
                                     data->type)))
M
Michal Privoznik 已提交
298 299
        goto cleanup;

J
Jiri Denemark 已提交
300 301 302 303 304 305 306 307 308 309 310
    switch (data->capsType) {
    case CAPS_NONE:
        break;

    case CAPS_ALL:
        if (fillAllCaps(domCaps) < 0)
            goto cleanup;
        break;

    case CAPS_QEMU:
#if WITH_QEMU
311
        if (fillQemuCaps(domCaps, data->capsName, data->arch, data->machine,
312
                         data->capsOpaque) < 0)
J
Jiri Denemark 已提交
313
            goto cleanup;
314 315 316 317 318 319 320
#endif
        break;

    case CAPS_LIBXL:
#if WITH_LIBXL
        if (fillXenCaps(domCaps) < 0)
            goto cleanup;
321 322 323 324 325 326
#endif
        break;
    case CAPS_BHYVE:
#if WITH_BHYVE
        if (fillBhyveCaps(domCaps, data->capsOpaque) < 0)
            goto cleanup;
J
Jiri Denemark 已提交
327 328 329 330
#endif
        break;
    }

M
Michal Privoznik 已提交
331 332 333
    if (!(domCapsXML = virDomainCapsFormat(domCaps)))
        goto cleanup;

334
    if (virTestCompareToFile(domCapsXML, path) < 0)
M
Michal Privoznik 已提交
335 336 337 338 339 340 341 342 343 344 345 346 347 348 349
        goto cleanup;

    ret = 0;
 cleanup:
    VIR_FREE(domCapsXML);
    VIR_FREE(path);
    virObjectUnref(domCaps);
    return ret;
}

static int
mymain(void)
{
    int ret = 0;

350 351 352 353
#if WITH_BHYVE
    unsigned int bhyve_caps = 0;
#endif

J
Jiri Denemark 已提交
354
#if WITH_QEMU
355 356
    virQEMUDriverConfigPtr cfg = virQEMUDriverConfigNew(false);

357 358
    if (!cfg)
        return EXIT_FAILURE;
J
Jiri Denemark 已提交
359 360 361 362 363 364 365 366 367 368 369 370
#endif

#define DO_TEST(Name, Emulator, Machine, Arch, Type, CapsType)          \
    do {                                                                \
        struct testData data = {                                        \
            .name = Name,                                               \
            .emulator = Emulator,                                       \
            .machine = Machine,                                         \
            .arch = Arch,                                               \
            .type = Type,                                               \
            .capsType = CapsType,                                       \
        };                                                              \
371
        if (virTestRun(Name, test_virDomainCapsFormat, &data) < 0)      \
J
Jiri Denemark 已提交
372 373
            ret = -1;                                                   \
    } while (0)
374

J
Jiri Denemark 已提交
375 376
#define DO_TEST_QEMU(Name, CapsName, Emulator, Machine, Arch, Type)     \
    do {                                                                \
377 378 379 380 381 382 383 384
        char *name = NULL;                                              \
        if (virAsprintf(&name, "qemu_%s%s%s.%s",                        \
                        Name,                                           \
                        Machine ? "-" : "", Machine ? Machine : "",     \
                        Arch) < 0) {                                    \
            ret = -1;                                                   \
            break;                                                      \
        }                                                               \
J
Jiri Denemark 已提交
385
        struct testData data = {                                        \
386
            .name = name,                                               \
J
Jiri Denemark 已提交
387 388 389 390 391 392 393 394
            .emulator = Emulator,                                       \
            .machine = Machine,                                         \
            .arch = Arch,                                               \
            .type = Type,                                               \
            .capsType = CAPS_QEMU,                                      \
            .capsName = CapsName,                                       \
            .capsOpaque = cfg,                                          \
        };                                                              \
395
        if (virTestRun(name, test_virDomainCapsFormat, &data) < 0)      \
J
Jiri Denemark 已提交
396
            ret = -1;                                                   \
397
        VIR_FREE(name);                                                 \
398 399
    } while (0)

400 401 402 403 404 405 406 407 408 409 410 411 412 413
#define DO_TEST_LIBXL(Name, Emulator, Machine, Arch, Type)              \
    do {                                                                \
        struct testData data = {                                        \
            .name = Name,                                               \
            .emulator = Emulator,                                       \
            .machine = Machine,                                         \
            .arch = Arch,                                               \
            .type = Type,                                               \
            .capsType = CAPS_LIBXL,                                     \
        };                                                              \
        if (virTestRun(Name, test_virDomainCapsFormat, &data) < 0)     \
            ret = -1;                                                   \
    } while (0)

J
Jiri Denemark 已提交
414
    DO_TEST("basic", "/bin/emulatorbin", "my-machine-type",
J
Jiri Denemark 已提交
415
            "x86_64", VIR_DOMAIN_VIRT_UML, CAPS_NONE);
J
Jiri Denemark 已提交
416
    DO_TEST("full", "/bin/emulatorbin", "my-machine-type",
J
Jiri Denemark 已提交
417
            "x86_64", VIR_DOMAIN_VIRT_KVM, CAPS_ALL);
J
Jiri Denemark 已提交
418

419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438
#define DO_TEST_BHYVE(Name, Emulator, BhyveCaps, Type) \
    do {                                                                \
        char *name = NULL;                                              \
        if (virAsprintf(&name, "bhyve_%s.x86_64", Name) < 0) {          \
             ret = -1;                                                  \
             break;                                                     \
        }                                                               \
        struct testData data = {                                        \
            .name = name,                                               \
            .emulator = Emulator,                                       \
            .arch = "x86_64",                                           \
            .type = Type,                                               \
            .capsType = CAPS_BHYVE,                                     \
            .capsOpaque = BhyveCaps,                                    \
        };                                                              \
        if (virTestRun(name, test_virDomainCapsFormat, &data) < 0)      \
            ret = -1;                                                   \
        VIR_FREE(name);                                                 \
    } while (0)

J
Jiri Denemark 已提交
439 440
#if WITH_QEMU

441
    DO_TEST_QEMU("1.7.0", "caps_1.7.0",
442
                 "/usr/bin/qemu-system-x86_64", NULL,
443
                 "x86_64", VIR_DOMAIN_VIRT_KVM);
444

445
    DO_TEST_QEMU("2.6.0", "caps_2.6.0",
446
                 "/usr/bin/qemu-system-x86_64", NULL,
447
                 "x86_64", VIR_DOMAIN_VIRT_KVM);
448

449
    DO_TEST_QEMU("2.6.0", "caps_2.6.0-gicv2",
450
                 "/usr/bin/qemu-system-aarch64", NULL,
451
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
452

453
    DO_TEST_QEMU("2.6.0-gicv2", "caps_2.6.0-gicv2",
454
                 "/usr/bin/qemu-system-aarch64", "virt",
455
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
456

457
    DO_TEST_QEMU("2.6.0-gicv3", "caps_2.6.0-gicv3",
458
                 "/usr/bin/qemu-system-aarch64", "virt",
459
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
460

461
    DO_TEST_QEMU("2.6.0", "caps_2.6.0",
462
                 "/usr/bin/qemu-system-ppc64", NULL,
463
                 "ppc64le", VIR_DOMAIN_VIRT_KVM);
464

465 466 467 468 469 470 471 472
    DO_TEST_QEMU("2.8.0", "caps_2.8.0",
                 "/usr/bin/qemu-system-x86_64", NULL,
                 "x86_64", VIR_DOMAIN_VIRT_KVM);

    DO_TEST_QEMU("2.8.0-tcg", "caps_2.8.0",
                 "/usr/bin/qemu-system-x86_64", NULL,
                 "x86_64", VIR_DOMAIN_VIRT_QEMU);

473 474 475 476
    DO_TEST_QEMU("2.9.0", "caps_2.9.0",
                 "/usr/bin/qemu-system-x86_64", NULL,
                 "x86_64", VIR_DOMAIN_VIRT_KVM);

477 478 479 480
    DO_TEST_QEMU("2.9.0", "caps_2.9.0",
                 "/usr/bin/qemu-system-x86_64", "q35",
                 "x86_64", VIR_DOMAIN_VIRT_KVM);

481 482 483 484
    DO_TEST_QEMU("2.9.0-tcg", "caps_2.9.0",
                 "/usr/bin/qemu-system-x86_64", NULL,
                 "x86_64", VIR_DOMAIN_VIRT_QEMU);

485 486 487 488 489 490 491 492
    DO_TEST_QEMU("2.7.0", "caps_2.7.0",
                 "/usr/bin/qemu-system-s390x", NULL,
                 "s390x", VIR_DOMAIN_VIRT_KVM);

    DO_TEST_QEMU("2.8.0", "caps_2.8.0",
                 "/usr/bin/qemu-system-s390x", NULL,
                 "s390x", VIR_DOMAIN_VIRT_KVM);

P
Pavel Hrdina 已提交
493 494
    virObjectUnref(cfg);

495 496
#endif /* WITH_QEMU */

497 498
#if WITH_LIBXL

499 500 501 502 503 504 505 506 507
# ifdef LIBXL_HAVE_PVUSB
#  define LIBXL_XENPV_CAPS "libxl-xenpv-usb"
#  define LIBXL_XENFV_CAPS "libxl-xenfv-usb"
# else
#  define LIBXL_XENPV_CAPS "libxl-xenpv"
#  define LIBXL_XENFV_CAPS "libxl-xenfv"
# endif

    DO_TEST_LIBXL(LIBXL_XENPV_CAPS, "/usr/bin/qemu-system-x86_64",
508
                  "xenpv", "x86_64", VIR_DOMAIN_VIRT_XEN);
509
    DO_TEST_LIBXL(LIBXL_XENFV_CAPS, "/usr/bin/qemu-system-x86_64",
510 511 512 513
                  "xenfv", "x86_64", VIR_DOMAIN_VIRT_XEN);

#endif /* WITH_LIBXL */

514 515 516 517 518 519 520 521 522 523
#if WITH_BHYVE
    DO_TEST_BHYVE("basic", "/usr/sbin/bhyve", &bhyve_caps, VIR_DOMAIN_VIRT_BHYVE);

    bhyve_caps |= BHYVE_CAP_LPC_BOOTROM;
    DO_TEST_BHYVE("uefi", "/usr/sbin/bhyve", &bhyve_caps, VIR_DOMAIN_VIRT_BHYVE);

    bhyve_caps |= BHYVE_CAP_FBUF;
    DO_TEST_BHYVE("fbuf", "/usr/sbin/bhyve", &bhyve_caps, VIR_DOMAIN_VIRT_BHYVE);
#endif /* WITH_BHYVE */

M
Michal Privoznik 已提交
524 525 526
    return ret;
}

527
#if WITH_QEMU
528
VIR_TEST_MAIN_PRELOAD(mymain,
529 530 531
                       abs_builddir "/.libs/domaincapsmock.so",
                       abs_builddir "/.libs/qemucpumock.so")
#else
532
VIR_TEST_MAIN_PRELOAD(mymain, abs_builddir "/.libs/domaincapsmock.so")
533
#endif