domaincapstest.c 15.1 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 69 70 71 72 73
    virCPUDef host = {
        VIR_CPU_TYPE_HOST, 0, 0,
        VIR_ARCH_X86_64, (char *) "host",
        NULL, 0, (char *) "CPU Vendorrr",
        0, 0, 0, 0, 0, NULL,
    };
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 91 92 93 94 95
        virDomainCapsCPUModelsAdd(cpu->custom, "Model1", -1,
                                  VIR_DOMCAPS_CPU_USABLE_UNKNOWN) < 0 ||
        virDomainCapsCPUModelsAdd(cpu->custom, "Model2", -1,
                                  VIR_DOMCAPS_CPU_USABLE_NO) < 0 ||
        virDomainCapsCPUModelsAdd(cpu->custom, "Model3", -1,
                                  VIR_DOMCAPS_CPU_USABLE_YES) < 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"
120

121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
static virCPUDef aarch64Cpu = {
    0, 0, 0, 0, NULL, NULL, 0, NULL, 1, 1, 1, 0, 0, NULL,
};

static virCPUDef ppc64leCpu = {
    VIR_CPU_TYPE_HOST, 0, 0,
    VIR_ARCH_PPC64LE, (char *) "POWER8",
    NULL, 0, NULL, 1, 1, 1, 0, 0, NULL,
};

static virCPUDef x86Cpu = {
    VIR_CPU_TYPE_HOST, 0, 0,
    VIR_ARCH_X86_64, (char *) "Broadwell",
    NULL, 0, NULL, 1, 1, 1, 0, 0, NULL,
};

137 138 139 140 141 142
static virCPUDef s390Cpu = {
    VIR_CPU_TYPE_HOST, 0, 0,
    VIR_ARCH_S390X, NULL,
    NULL, 0, NULL, 1, 1, 1, 0, 0, NULL,
};

143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
static int
fakeHostCPU(virCapsPtr caps,
            virArch arch)
{
    virCPUDefPtr cpu;

    switch (arch) {
    case VIR_ARCH_AARCH64:
        cpu = &aarch64Cpu;
        break;

    case VIR_ARCH_PPC64LE:
        cpu = &ppc64leCpu;
        break;

    case VIR_ARCH_X86_64:
        cpu = &x86Cpu;
        break;

162 163 164 165
    case VIR_ARCH_S390X:
        cpu = &s390Cpu;
        break;

166 167 168 169 170 171 172 173 174 175 176 177 178
    default:
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "cannot fake host CPU for arch %s",
                       virArchToString(arch));
        return -1;
    }

    if (!(caps->host.cpu = virCPUDefCopy(cpu)))
        return -1;

    return 0;
}

179
static int
180
fillQemuCaps(virDomainCapsPtr domCaps,
J
Jiri Denemark 已提交
181
             const char *name,
J
Jiri Denemark 已提交
182
             const char *arch,
183
             const char *machine,
J
Jiri Denemark 已提交
184
             virQEMUDriverConfigPtr cfg)
185
{
J
Jiri Denemark 已提交
186 187
    int ret = -1;
    char *path = NULL;
188
    virCapsPtr caps = NULL;
J
Jiri Denemark 已提交
189
    virQEMUCapsPtr qemuCaps = NULL;
190
    virDomainCapsLoaderPtr loader = &domCaps->os.loader;
191

192 193 194 195
    if (!(caps = virCapabilitiesNew(domCaps->arch, false, false)) ||
        fakeHostCPU(caps, domCaps->arch) < 0)
        goto cleanup;

196
    if (virAsprintf(&path, "%s/qemucapabilitiesdata/%s.%s.xml",
J
Jiri Denemark 已提交
197
                    abs_srcdir, name, arch) < 0 ||
198
        !(qemuCaps = qemuTestParseCapabilities(caps, path)))
J
Jiri Denemark 已提交
199 200
        goto cleanup;

201 202 203 204 205 206 207 208 209 210
    if (machine &&
        VIR_STRDUP(domCaps->machine,
                   virQEMUCapsGetCanonicalMachine(qemuCaps, machine)) < 0)
        goto cleanup;

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

211
    if (virQEMUCapsFillDomainCaps(caps, domCaps, qemuCaps,
212
                                  cfg->firmwares,
213
                                  cfg->nfirmwares) < 0)
J
Jiri Denemark 已提交
214
        goto cleanup;
215 216 217 218 219 220 221 222 223 224

    /* 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);
225

J
Jiri Denemark 已提交
226 227
    /* As of f05b6a918e28 we are expecting to see OVMF_CODE.fd file which
     * may not exists everywhere. */
228 229 230 231
    while (loader->values.nvalues)
        VIR_FREE(loader->values.values[--loader->values.nvalues]);

    if (fillStringValues(&loader->values,
232
                         "/usr/share/AAVMF/AAVMF_CODE.fd",
233 234
                         "/usr/share/OVMF/OVMF_CODE.fd",
                         NULL) < 0)
M
Michal Privoznik 已提交
235 236
        goto cleanup;

J
Jiri Denemark 已提交
237
    ret = 0;
M
Michal Privoznik 已提交
238
 cleanup:
239
    virObjectUnref(caps);
J
Jiri Denemark 已提交
240 241
    virObjectUnref(qemuCaps);
    VIR_FREE(path);
242
    return ret;
M
Michal Privoznik 已提交
243
}
J
Jiri Denemark 已提交
244 245 246
#endif /* WITH_QEMU */


247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
#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 */


J
Jiri Denemark 已提交
277 278 279 280
enum testCapsType {
    CAPS_NONE,
    CAPS_ALL,
    CAPS_QEMU,
281
    CAPS_LIBXL,
J
Jiri Denemark 已提交
282
};
M
Michal Privoznik 已提交
283

J
Jiri Denemark 已提交
284 285 286
struct testData {
    const char *name;
    const char *emulator;
M
Michal Privoznik 已提交
287
    const char *machine;
J
Jiri Denemark 已提交
288
    const char *arch;
M
Michal Privoznik 已提交
289
    virDomainVirtType type;
J
Jiri Denemark 已提交
290 291 292
    enum testCapsType capsType;
    const char *capsName;
    void *capsOpaque;
M
Michal Privoznik 已提交
293 294 295 296 297
};

static int
test_virDomainCapsFormat(const void *opaque)
{
J
Jiri Denemark 已提交
298
    const struct testData *data = opaque;
M
Michal Privoznik 已提交
299 300 301 302 303
    virDomainCapsPtr domCaps = NULL;
    char *path = NULL;
    char *domCapsXML = NULL;
    int ret = -1;

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

J
Jiri Denemark 已提交
308 309
    if (!(domCaps = virDomainCapsNew(data->emulator, data->machine,
                                     virArchFromString(data->arch),
J
Jiri Denemark 已提交
310
                                     data->type)))
M
Michal Privoznik 已提交
311 312
        goto cleanup;

J
Jiri Denemark 已提交
313 314 315 316 317 318 319 320 321 322 323
    switch (data->capsType) {
    case CAPS_NONE:
        break;

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

    case CAPS_QEMU:
#if WITH_QEMU
324
        if (fillQemuCaps(domCaps, data->capsName, data->arch, data->machine,
325
                         data->capsOpaque) < 0)
J
Jiri Denemark 已提交
326
            goto cleanup;
327 328 329 330 331 332 333
#endif
        break;

    case CAPS_LIBXL:
#if WITH_LIBXL
        if (fillXenCaps(domCaps) < 0)
            goto cleanup;
J
Jiri Denemark 已提交
334 335 336 337
#endif
        break;
    }

M
Michal Privoznik 已提交
338 339 340
    if (!(domCapsXML = virDomainCapsFormat(domCaps)))
        goto cleanup;

341
    if (virTestCompareToFile(domCapsXML, path) < 0)
M
Michal Privoznik 已提交
342 343 344 345 346 347 348 349 350 351 352 353 354 355 356
        goto cleanup;

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

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

J
Jiri Denemark 已提交
357
#if WITH_QEMU
358 359
    virQEMUDriverConfigPtr cfg = virQEMUDriverConfigNew(false);

360 361
    if (!cfg)
        return EXIT_FAILURE;
J
Jiri Denemark 已提交
362 363 364 365 366 367 368 369 370 371 372 373
#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,                                       \
        };                                                              \
374
        if (virTestRun(Name, test_virDomainCapsFormat, &data) < 0)      \
J
Jiri Denemark 已提交
375 376
            ret = -1;                                                   \
    } while (0)
377

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

403 404 405 406 407 408 409 410 411 412 413 414 415 416
#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 已提交
417
    DO_TEST("basic", "/bin/emulatorbin", "my-machine-type",
J
Jiri Denemark 已提交
418
            "x86_64", VIR_DOMAIN_VIRT_UML, CAPS_NONE);
J
Jiri Denemark 已提交
419
    DO_TEST("full", "/bin/emulatorbin", "my-machine-type",
J
Jiri Denemark 已提交
420
            "x86_64", VIR_DOMAIN_VIRT_KVM, CAPS_ALL);
J
Jiri Denemark 已提交
421 422 423

#if WITH_QEMU

424
    DO_TEST_QEMU("1.7.0", "caps_1.7.0",
425
                 "/usr/bin/qemu-system-x86_64", NULL,
426
                 "x86_64", VIR_DOMAIN_VIRT_KVM);
427

428
    DO_TEST_QEMU("2.6.0", "caps_2.6.0",
429
                 "/usr/bin/qemu-system-x86_64", NULL,
430
                 "x86_64", VIR_DOMAIN_VIRT_KVM);
431

432
    DO_TEST_QEMU("2.6.0", "caps_2.6.0-gicv2",
433
                 "/usr/bin/qemu-system-aarch64", NULL,
434
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
435

436
    DO_TEST_QEMU("2.6.0-gicv2", "caps_2.6.0-gicv2",
437
                 "/usr/bin/qemu-system-aarch64", "virt",
438
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
439

440
    DO_TEST_QEMU("2.6.0-gicv3", "caps_2.6.0-gicv3",
441
                 "/usr/bin/qemu-system-aarch64", "virt",
442
                 "aarch64", VIR_DOMAIN_VIRT_KVM);
443

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

448 449 450 451 452 453 454 455
    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);

456 457 458 459 460 461 462 463
    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);

464 465
#endif /* WITH_QEMU */

466 467
#if WITH_LIBXL

468 469 470 471 472 473 474 475 476
# 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",
477
                  "xenpv", "x86_64", VIR_DOMAIN_VIRT_XEN);
478
    DO_TEST_LIBXL(LIBXL_XENFV_CAPS, "/usr/bin/qemu-system-x86_64",
479 480 481 482
                  "xenfv", "x86_64", VIR_DOMAIN_VIRT_XEN);

#endif /* WITH_LIBXL */

M
Michal Privoznik 已提交
483 484 485
    return ret;
}

486
VIRT_TEST_MAIN_PRELOAD(mymain, abs_builddir "/.libs/domaincapsmock.so")