cpu.c 21.3 KB
Newer Older
J
Jiri Denemark 已提交
1 2 3
/*
 * cpu.c: internal functions for CPU manipulation
 *
4
 * Copyright (C) 2009-2013 Red Hat, Inc.
J
Jiri Denemark 已提交
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/>.
J
Jiri Denemark 已提交
19 20 21 22 23 24 25
 *
 * Authors:
 *      Jiri Denemark <jdenemar@redhat.com>
 */

#include <config.h>

26
#include "virlog.h"
27
#include "viralloc.h"
28
#include "virxml.h"
J
Jiri Denemark 已提交
29
#include "cpu.h"
30
#include "cpu_map.h"
J
Jiri Denemark 已提交
31
#include "cpu_x86.h"
32
#include "cpu_ppc64.h"
T
Thang Pham 已提交
33
#include "cpu_s390.h"
C
Chuck Short 已提交
34
#include "cpu_arm.h"
J
Jiri Denemark 已提交
35
#include "cpu_generic.h"
36
#include "util/virstring.h"
J
Jiri Denemark 已提交
37 38 39 40 41


#define NR_DRIVERS ARRAY_CARDINALITY(drivers)
#define VIR_FROM_THIS VIR_FROM_CPU

42 43
VIR_LOG_INIT("cpu.cpu");

J
Jiri Denemark 已提交
44 45
static struct cpuArchDriver *drivers[] = {
    &cpuDriverX86,
46
    &cpuDriverPPC64,
T
Thang Pham 已提交
47
    &cpuDriverS390,
C
Chuck Short 已提交
48
    &cpuDriverArm,
J
Jiri Denemark 已提交
49 50 51 52 53 54
    /* generic driver must always be the last one */
    &cpuDriverGeneric
};


static struct cpuArchDriver *
55
cpuGetSubDriver(virArch arch)
J
Jiri Denemark 已提交
56
{
57 58
    size_t i;
    size_t j;
J
Jiri Denemark 已提交
59

60
    if (arch == VIR_ARCH_NONE) {
61 62
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("undefined hardware architecture"));
J
Jiri Denemark 已提交
63 64 65 66 67
        return NULL;
    }

    for (i = 0; i < NR_DRIVERS - 1; i++) {
        for (j = 0; j < drivers[i]->narch; j++) {
68
            if (arch == drivers[i]->arch[j])
J
Jiri Denemark 已提交
69 70 71 72 73 74 75 76 77
                return drivers[i];
        }
    }

    /* use generic driver by default */
    return drivers[NR_DRIVERS - 1];
}


78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
static struct cpuArchDriver *
cpuGetSubDriverByName(const char *name)
{
    size_t i;

    for (i = 0; i < NR_DRIVERS - 1; i++) {
        if (STREQ_NULLABLE(name, drivers[i]->name))
            return drivers[i];
    }

    virReportError(VIR_ERR_INTERNAL_ERROR,
                   _("CPU driver '%s' does not exist"),
                   name);
    return NULL;
}


95 96 97 98 99 100 101 102 103 104 105 106 107
/**
 * cpuCompareXML:
 *
 * @host: host CPU definition
 * @xml: XML description of either guest or host CPU to be compared with @host
 *
 * Compares the CPU described by @xml with @host CPU.
 *
 * Returns VIR_CPU_COMPARE_ERROR on error, VIR_CPU_COMPARE_INCOMPATIBLE when
 * the two CPUs are incompatible, VIR_CPU_COMPARE_IDENTICAL when the two CPUs
 * are identical, VIR_CPU_COMPARE_SUPERSET when the @xml CPU is a superset of
 * the @host CPU.
 */
J
Jiri Denemark 已提交
108
virCPUCompareResult
109
cpuCompareXML(virCPUDefPtr host,
110 111
              const char *xml,
              bool failIncompatible)
J
Jiri Denemark 已提交
112 113 114 115 116 117
{
    xmlDocPtr doc = NULL;
    xmlXPathContextPtr ctxt = NULL;
    virCPUDefPtr cpu = NULL;
    virCPUCompareResult ret = VIR_CPU_COMPARE_ERROR;

118 119
    VIR_DEBUG("host=%p, xml=%s", host, NULLSTR(xml));

120
    if (!(doc = virXMLParseStringCtxt(xml, _("(CPU_definition)"), &ctxt)))
121
        goto cleanup;
J
Jiri Denemark 已提交
122

123
    cpu = virCPUDefParseXML(ctxt->node, ctxt, VIR_CPU_TYPE_AUTO);
J
Jiri Denemark 已提交
124 125 126
    if (cpu == NULL)
        goto cleanup;

127
    ret = cpuCompare(host, cpu, failIncompatible);
J
Jiri Denemark 已提交
128

129
 cleanup:
J
Jiri Denemark 已提交
130 131 132 133 134 135 136 137
    virCPUDefFree(cpu);
    xmlXPathFreeContext(ctxt);
    xmlFreeDoc(doc);

    return ret;
}


138 139 140 141 142 143 144 145 146 147 148 149 150
/**
 * cpuCompare:
 *
 * @host: host CPU definition
 * @cpu: either guest or host CPU to be compared with @host
 *
 * Compares the CPU described by @cpu with @host CPU.
 *
 * Returns VIR_CPU_COMPARE_ERROR on error, VIR_CPU_COMPARE_INCOMPATIBLE when
 * the two CPUs are incompatible, VIR_CPU_COMPARE_IDENTICAL when the two CPUs
 * are identical, VIR_CPU_COMPARE_SUPERSET when the @cpu CPU is a superset of
 * the @host CPU.
 */
J
Jiri Denemark 已提交
151
virCPUCompareResult
152
cpuCompare(virCPUDefPtr host,
153 154
           virCPUDefPtr cpu,
           bool failIncompatible)
J
Jiri Denemark 已提交
155 156 157
{
    struct cpuArchDriver *driver;

158 159
    VIR_DEBUG("host=%p, cpu=%p", host, cpu);

160
    if ((driver = cpuGetSubDriver(host->arch)) == NULL)
J
Jiri Denemark 已提交
161 162 163
        return VIR_CPU_COMPARE_ERROR;

    if (driver->compare == NULL) {
164 165
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compare CPUs of %s architecture"),
166
                       virArchToString(host->arch));
J
Jiri Denemark 已提交
167 168 169
        return VIR_CPU_COMPARE_ERROR;
    }

170
    return driver->compare(host, cpu, failIncompatible);
J
Jiri Denemark 已提交
171 172 173
}


174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198
/**
 * cpuDecode:
 *
 * @cpu: CPU definition stub to be filled in
 * @data: internal CPU data to be decoded into @cpu definition
 * @models: list of CPU models that can be considered when decoding @data
 * @nmodels: number of CPU models in @models
 * @preferred: CPU models that should be used if possible
 *
 * Decodes internal CPU data into a CPU definition consisting of a CPU model
 * and a list of CPU features. The @cpu model stub is supposed to have arch,
 * type, match and fallback members set, this function will add the rest. If
 * @models list is NULL, all models supported by libvirt will be considered
 * when decoding the data. In general, this function will select the model
 * closest to the CPU specified by @data unless @preferred is non-NULL, in
 * which case the @preferred model will be used as long as it is compatible
 * with @data.
 *
 * For VIR_ARCH_I686 and VIR_ARCH_X86_64 architectures this means the computed
 * CPU definition will have the shortest possible list of additional features.
 * When @preferred is non-NULL, the @preferred model will be used even if
 * other models would result in a shorter list of additional features.
 *
 * Returns 0 on success, -1 on error.
 */
J
Jiri Denemark 已提交
199
int
200
cpuDecode(virCPUDefPtr cpu,
201
          const virCPUData *data,
202
          const char **models,
203 204
          unsigned int nmodels,
          const char *preferred)
J
Jiri Denemark 已提交
205 206 207
{
    struct cpuArchDriver *driver;

208 209
    VIR_DEBUG("cpu=%p, data=%p, nmodels=%u, preferred=%s",
              cpu, data, nmodels, NULLSTR(preferred));
210
    if (models) {
211
        size_t i;
212
        for (i = 0; i < nmodels; i++)
213
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
214 215
    }

216
    if (models == NULL && nmodels != 0) {
217 218
        virReportError(VIR_ERR_INVALID_ARG, "%s",
                       _("nonzero nmodels doesn't match with NULL models"));
219 220 221
        return -1;
    }

222 223 224 225
    if (cpu->type > VIR_CPU_TYPE_GUEST ||
        cpu->mode != VIR_CPU_MODE_CUSTOM) {
        virReportError(VIR_ERR_INVALID_ARG, "%s",
                       _("invalid CPU definition stub"));
J
Jiri Denemark 已提交
226 227 228
        return -1;
    }

229
    if ((driver = cpuGetSubDriver(cpu->arch)) == NULL)
J
Jiri Denemark 已提交
230 231 232
        return -1;

    if (driver->decode == NULL) {
233 234
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot decode CPU data for %s architecture"),
235
                       virArchToString(cpu->arch));
J
Jiri Denemark 已提交
236 237 238
        return -1;
    }

239
    return driver->decode(cpu, data, models, nmodels, preferred, 0);
J
Jiri Denemark 已提交
240 241 242
}


243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261
/**
 * cpuEncode:
 *
 * @arch: CPU architecture
 * @cpu: CPU definition to be encoded into internal CPU driver representation
 * @forced: where to store CPU data corresponding to forced features
 * @required: where to store CPU data corresponding to required features
 * @optional: where to store CPU data corresponding to optional features
 * @disabled: where to store CPU data corresponding to disabled features
 * @forbidden: where to store CPU data corresponding to forbidden features
 * @vendor: where to store CPU data corresponding to CPU vendor
 *
 * Encode CPU definition from @cpu into internal CPU driver representation.
 * Any of @forced, @required, @optional, @disabled, @forbidden and @vendor
 * arguments can be NULL in case the caller is not interested in the
 * corresponding data.
 *
 * Returns 0 on success, -1 on error.
 */
J
Jiri Denemark 已提交
262
int
263
cpuEncode(virArch arch,
264
          const virCPUDef *cpu,
265 266 267 268 269 270
          virCPUDataPtr *forced,
          virCPUDataPtr *required,
          virCPUDataPtr *optional,
          virCPUDataPtr *disabled,
          virCPUDataPtr *forbidden,
          virCPUDataPtr *vendor)
J
Jiri Denemark 已提交
271 272 273
{
    struct cpuArchDriver *driver;

274
    VIR_DEBUG("arch=%s, cpu=%p, forced=%p, required=%p, "
J
Jiri Denemark 已提交
275
              "optional=%p, disabled=%p, forbidden=%p, vendor=%p",
276
              virArchToString(arch), cpu, forced, required,
J
Jiri Denemark 已提交
277
              optional, disabled, forbidden, vendor);
278

279 280 281 282 283 284
    if (!cpu->model) {
        virReportError(VIR_ERR_INVALID_ARG, "%s",
                       _("no guest CPU model specified"));
        return -1;
    }

285
    if ((driver = cpuGetSubDriver(arch)) == NULL)
J
Jiri Denemark 已提交
286 287 288
        return -1;

    if (driver->encode == NULL) {
289 290
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot encode CPU data for %s architecture"),
291
                       virArchToString(arch));
J
Jiri Denemark 已提交
292 293 294
        return -1;
    }

J
Jiri Denemark 已提交
295
    return driver->encode(arch, cpu, forced, required,
J
Jiri Denemark 已提交
296
                          optional, disabled, forbidden, vendor);
J
Jiri Denemark 已提交
297 298 299
}


300 301 302 303 304 305 306 307 308
/**
 * cpuDataFree:
 *
 * @data: CPU data structure to be freed
 *
 * Free internal CPU data.
 *
 * Returns nothing.
 */
J
Jiri Denemark 已提交
309
void
J
Jiri Denemark 已提交
310
cpuDataFree(virCPUDataPtr data)
J
Jiri Denemark 已提交
311 312 313
{
    struct cpuArchDriver *driver;

J
Jiri Denemark 已提交
314
    VIR_DEBUG("data=%p", data);
315

J
Jiri Denemark 已提交
316 317 318
    if (data == NULL)
        return;

J
Jiri Denemark 已提交
319
    if ((driver = cpuGetSubDriver(data->arch)) == NULL)
J
Jiri Denemark 已提交
320 321 322
        return;

    if (driver->free == NULL) {
323 324
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot free CPU data for %s architecture"),
J
Jiri Denemark 已提交
325
                       virArchToString(data->arch));
J
Jiri Denemark 已提交
326 327 328
        return;
    }

E
Eric Blake 已提交
329
    (driver->free)(data);
J
Jiri Denemark 已提交
330 331 332
}


333 334 335 336 337 338 339
/**
 * cpuNodeData:
 *
 * @arch: CPU architecture
 *
 * Returns CPU data for host CPU or NULL on error.
 */
340
virCPUDataPtr
341
cpuNodeData(virArch arch)
J
Jiri Denemark 已提交
342 343 344
{
    struct cpuArchDriver *driver;

345
    VIR_DEBUG("arch=%s", virArchToString(arch));
346

347
    if ((driver = cpuGetSubDriver(arch)) == NULL)
J
Jiri Denemark 已提交
348 349 350
        return NULL;

    if (driver->nodeData == NULL) {
351 352
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot get node CPU data for %s architecture"),
353
                       virArchToString(arch));
J
Jiri Denemark 已提交
354 355 356
        return NULL;
    }

357
    return driver->nodeData(arch);
J
Jiri Denemark 已提交
358 359 360
}


361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377
/**
 * cpuGuestData:
 *
 * @host: host CPU definition
 * @guest: guest CPU definition
 * @data: computed guest CPU data
 * @msg: error message describing why the @guest and @host CPUs are considered
 *       incompatible
 *
 * Computes guest CPU data for the @guest CPU definition when run on the @host
 * CPU.
 *
 * Returns VIR_CPU_COMPARE_ERROR on error, VIR_CPU_COMPARE_INCOMPATIBLE when
 * the two CPUs are incompatible (@msg will describe the incompatibility),
 * VIR_CPU_COMPARE_IDENTICAL when the two CPUs are identical,
 * VIR_CPU_COMPARE_SUPERSET when the @guest CPU is a superset of the @host CPU.
 */
J
Jiri Denemark 已提交
378
virCPUCompareResult
379
cpuGuestData(virCPUDefPtr host,
J
Jiri Denemark 已提交
380
             virCPUDefPtr guest,
381
             virCPUDataPtr *data,
382
             char **msg)
J
Jiri Denemark 已提交
383 384 385
{
    struct cpuArchDriver *driver;

386
    VIR_DEBUG("host=%p, guest=%p, data=%p, msg=%p", host, guest, data, msg);
387

388
    if ((driver = cpuGetSubDriver(host->arch)) == NULL)
J
Jiri Denemark 已提交
389 390 391
        return VIR_CPU_COMPARE_ERROR;

    if (driver->guestData == NULL) {
392 393
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compute guest CPU data for %s architecture"),
394
                       virArchToString(host->arch));
J
Jiri Denemark 已提交
395 396 397
        return VIR_CPU_COMPARE_ERROR;
    }

398
    return driver->guestData(host, guest, data, msg);
J
Jiri Denemark 已提交
399
}
400 401


402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422
/**
 * cpuBaselineXML:
 *
 * @xmlCPUs: list of host CPU XML descriptions
 * @ncpus: number of CPUs in @xmlCPUs
 * @models: list of CPU models that can be considered for the baseline CPU
 * @nmodels: number of CPU models in @models
 * @flags: bitwise-OR of virConnectBaselineCPUFlags
 *
 * Computes the most feature-rich CPU which is compatible with all given
 * host CPUs. If @models array is NULL, all models supported by libvirt will
 * be considered when computing the baseline CPU model, otherwise the baseline
 * CPU model will be one of the provided CPU @models.
 *
 * If @flags includes VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES then libvirt
 * will explicitly list all CPU features that are part of the host CPU,
 * without this flag features that are part of the CPU model will not be
 * listed.
 *
 * Returns XML description of the baseline CPU or NULL on error.
 */
423 424 425 426
char *
cpuBaselineXML(const char **xmlCPUs,
               unsigned int ncpus,
               const char **models,
427 428
               unsigned int nmodels,
               unsigned int flags)
429 430 431 432 433 434
{
    xmlDocPtr doc = NULL;
    xmlXPathContextPtr ctxt = NULL;
    virCPUDefPtr *cpus = NULL;
    virCPUDefPtr cpu = NULL;
    char *cpustr;
435
    size_t i;
436

437 438 439
    VIR_DEBUG("ncpus=%u, nmodels=%u", ncpus, nmodels);
    if (xmlCPUs) {
        for (i = 0; i < ncpus; i++)
440
            VIR_DEBUG("xmlCPUs[%zu]=%s", i, NULLSTR(xmlCPUs[i]));
441 442 443
    }
    if (models) {
        for (i = 0; i < nmodels; i++)
444
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
445 446
    }

447
    if (xmlCPUs == NULL && ncpus != 0) {
448 449
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero ncpus doesn't match with NULL xmlCPUs"));
450 451 452 453
        return NULL;
    }

    if (ncpus < 1) {
454
        virReportError(VIR_ERR_INVALID_ARG, "%s", _("No CPUs given"));
455 456 457 458
        return NULL;
    }

    if (VIR_ALLOC_N(cpus, ncpus))
459
        goto error;
460 461

    for (i = 0; i < ncpus; i++) {
462
        if (!(doc = virXMLParseStringCtxt(xmlCPUs[i], _("(CPU_definition)"), &ctxt)))
463 464
            goto error;

465 466 467 468 469 470 471 472 473 474
        cpus[i] = virCPUDefParseXML(ctxt->node, ctxt, VIR_CPU_TYPE_HOST);
        if (cpus[i] == NULL)
            goto error;

        xmlXPathFreeContext(ctxt);
        xmlFreeDoc(doc);
        ctxt = NULL;
        doc = NULL;
    }

475
    if (!(cpu = cpuBaseline(cpus, ncpus, models, nmodels, flags)))
476 477
        goto error;

478
    cpustr = virCPUDefFormat(cpu, NULL, false);
479

480
 cleanup:
481 482 483 484 485 486 487 488 489 490 491
    if (cpus) {
        for (i = 0; i < ncpus; i++)
            virCPUDefFree(cpus[i]);
        VIR_FREE(cpus);
    }
    virCPUDefFree(cpu);
    xmlXPathFreeContext(ctxt);
    xmlFreeDoc(doc);

    return cpustr;

492
 error:
493 494 495 496 497
    cpustr = NULL;
    goto cleanup;
}


498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518
/**
 * cpuBaseline:
 *
 * @cpus: list of host CPU definitions
 * @ncpus: number of CPUs in @cpus
 * @models: list of CPU models that can be considered for the baseline CPU
 * @nmodels: number of CPU models in @models
 * @flags: bitwise-OR of virConnectBaselineCPUFlags
 *
 * Computes the most feature-rich CPU which is compatible with all given
 * host CPUs. If @models array is NULL, all models supported by libvirt will
 * be considered when computing the baseline CPU model, otherwise the baseline
 * CPU model will be one of the provided CPU @models.
 *
 * If @flags includes VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES then libvirt
 * will explicitly list all CPU features that are part of the host CPU,
 * without this flag features that are part of the CPU model will not be
 * listed.
 *
 * Returns baseline CPU definition or NULL on error.
 */
519 520 521 522
virCPUDefPtr
cpuBaseline(virCPUDefPtr *cpus,
            unsigned int ncpus,
            const char **models,
523 524
            unsigned int nmodels,
            unsigned int flags)
525 526
{
    struct cpuArchDriver *driver;
527
    size_t i;
528 529 530 531

    VIR_DEBUG("ncpus=%u, nmodels=%u", ncpus, nmodels);
    if (cpus) {
        for (i = 0; i < ncpus; i++)
532
            VIR_DEBUG("cpus[%zu]=%p", i, cpus[i]);
533 534 535
    }
    if (models) {
        for (i = 0; i < nmodels; i++)
536
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
537
    }
538 539

    if (cpus == NULL && ncpus != 0) {
540 541
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero ncpus doesn't match with NULL cpus"));
542 543 544 545
        return NULL;
    }

    if (ncpus < 1) {
546
        virReportError(VIR_ERR_INVALID_ARG, "%s", _("No CPUs given"));
547 548 549
        return NULL;
    }

550 551 552 553 554 555 556 557 558 559 560 561 562
    for (i = 0; i < ncpus; i++) {
        if (!cpus[i]) {
            virReportError(VIR_ERR_INVALID_ARG,
                           _("invalid CPU definition at index %zu"), i);
            return NULL;
        }
        if (!cpus[i]->model) {
            virReportError(VIR_ERR_INVALID_ARG,
                           _("no CPU model specified at index %zu"), i);
            return NULL;
        }
    }

563
    if (models == NULL && nmodels != 0) {
564 565
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero nmodels doesn't match with NULL models"));
566 567 568 569 570 571 572
        return NULL;
    }

    if ((driver = cpuGetSubDriver(cpus[0]->arch)) == NULL)
        return NULL;

    if (driver->baseline == NULL) {
573 574
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compute baseline CPU of %s architecture"),
575
                       virArchToString(cpus[0]->arch));
576 577 578
        return NULL;
    }

579
    return driver->baseline(cpus, ncpus, models, nmodels, flags);
580
}
581 582


583 584 585 586 587 588 589 590 591 592 593 594 595 596
/**
 * cpuUpdate:
 *
 * @guest: guest CPU definition
 * @host: host CPU definition
 *
 * Updates @guest CPU definition according to @host CPU. This is required to
 * support guest CPU definition which are relative to host CPU, such as CPUs
 * with VIR_CPU_MODE_CUSTOM and optional features or VIR_CPU_MATCH_MINIMUM, or
 * CPUs with non-custom mode (VIR_CPU_MODE_HOST_MODEL,
 * VIR_CPU_MODE_HOST_PASSTHROUGH).
 *
 * Returns 0 on success, -1 on error.
 */
597 598
int
cpuUpdate(virCPUDefPtr guest,
599
          const virCPUDef *host)
600 601 602 603 604 605 606 607 608
{
    struct cpuArchDriver *driver;

    VIR_DEBUG("guest=%p, host=%p", guest, host);

    if ((driver = cpuGetSubDriver(host->arch)) == NULL)
        return -1;

    if (driver->update == NULL) {
609 610
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot update guest CPU data for %s architecture"),
611
                       virArchToString(host->arch));
612 613 614 615 616
        return -1;
    }

    return driver->update(guest, host);
}
D
Daniel P. Berrange 已提交
617

618 619 620 621 622 623 624 625 626 627 628 629

/**
 * cpuHasFeature:
 *
 * @data: internal CPU representation
 * @feature: feature to be checked for
 *
 * Checks whether @feature is supported by the CPU described by @data.
 *
 * Returns 1 if the feature is supported, 0 if it's not supported, or
 * -1 on error.
 */
D
Daniel P. Berrange 已提交
630
int
631
cpuHasFeature(const virCPUData *data,
D
Daniel P. Berrange 已提交
632 633 634 635
              const char *feature)
{
    struct cpuArchDriver *driver;

J
Jiri Denemark 已提交
636
    VIR_DEBUG("data=%p, feature=%s", data, feature);
D
Daniel P. Berrange 已提交
637

J
Jiri Denemark 已提交
638
    if ((driver = cpuGetSubDriver(data->arch)) == NULL)
D
Daniel P. Berrange 已提交
639 640 641
        return -1;

    if (driver->hasFeature == NULL) {
642 643
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot check guest CPU data for %s architecture"),
J
Jiri Denemark 已提交
644
                       virArchToString(data->arch));
D
Daniel P. Berrange 已提交
645 646 647 648 649
        return -1;
    }

    return driver->hasFeature(data, feature);
}
650

651 652 653 654 655 656 657 658 659 660

/**
 * cpuDataFormat:
 *
 * @data: internal CPU representation
 *
 * Formats @data into XML for test purposes.
 *
 * Returns string representation of the XML describing @data or NULL on error.
 */
661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680
char *
cpuDataFormat(const virCPUData *data)
{
    struct cpuArchDriver *driver;

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

    if (!(driver = cpuGetSubDriver(data->arch)))
        return NULL;

    if (!driver->dataFormat) {
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot format %s CPU data"),
                       virArchToString(data->arch));
        return NULL;
    }

    return driver->dataFormat(data);
}

681 682 683 684 685 686 687 688 689 690

/**
 * cpuDataParse:
 *
 * @xmlStr: XML string produced by cpuDataFormat
 *
 * Parses XML representation of virCPUData structure for test purposes.
 *
 * Returns internal CPU data structure parsed from the XML or NULL on error.
 */
691
virCPUDataPtr
692
cpuDataParse(const char *xmlStr)
693 694
{
    struct cpuArchDriver *driver;
695 696 697 698
    xmlDocPtr xml = NULL;
    xmlXPathContextPtr ctxt = NULL;
    virCPUDataPtr data = NULL;
    char *arch = NULL;
699

700
    VIR_DEBUG("xmlStr=%s", xmlStr);
701

702 703 704 705 706 707 708 709 710 711 712 713 714 715
    if (!(xml = virXMLParseStringCtxt(xmlStr, _("CPU data"), &ctxt))) {
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("cannot parse CPU data"));
        goto cleanup;
    }

    if (!(arch = virXPathString("string(/cpudata/@arch)", ctxt))) {
        virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
                       _("missing CPU data architecture"));
        goto cleanup;
    }

    if (!(driver = cpuGetSubDriverByName(arch)))
        goto cleanup;
716 717 718

    if (!driver->dataParse) {
        virReportError(VIR_ERR_NO_SUPPORT,
719 720
                       _("cannot parse %s CPU data"), arch);
        goto cleanup;
721 722
    }

723 724 725 726 727 728 729
    data = driver->dataParse(ctxt);

 cleanup:
    xmlXPathFreeContext(ctxt);
    xmlFreeDoc(xml);
    VIR_FREE(arch);
    return data;
730 731
}

732 733 734 735 736
bool
cpuModelIsAllowed(const char *model,
                  const char **models,
                  unsigned int nmodels)
{
737
    size_t i;
738 739 740 741 742 743 744 745 746 747

    if (!models || !nmodels)
        return true;

    for (i = 0; i < nmodels; i++) {
        if (models[i] && STREQ(models[i], model))
            return true;
    }
    return false;
}
748

749 750 751 752
/**
 * cpuGetModels:
 *
 * @archName: CPU architecture string
753
 * @models: where to store the NULL-terminated list of supported models
754 755 756 757 758
 *
 * Fetches all CPU models supported by libvirt on @archName.
 *
 * Returns number of supported CPU models or -1 on error.
 */
759 760 761 762
int
cpuGetModels(const char *archName, char ***models)
{
    struct cpuArchDriver *driver;
763 764 765
    virArch arch;

    VIR_DEBUG("arch=%s", archName);
766 767 768 769 770 771

    arch = virArchFromString(archName);
    if (arch == VIR_ARCH_NONE) {
        virReportError(VIR_ERR_INVALID_ARG,
                       _("cannot find architecture %s"),
                       archName);
772
        return -1;
773 774 775 776 777 778 779
    }

    driver = cpuGetSubDriver(arch);
    if (driver == NULL) {
        virReportError(VIR_ERR_INVALID_ARG,
                       _("cannot find a driver for the architecture %s"),
                       archName);
780
        return -1;
781 782
    }

783 784 785 786 787 788
    if (!driver->getModels) {
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("CPU driver for %s has no CPU model support"),
                       virArchToString(arch));
        return -1;
    }
789

790
    return driver->getModels(models);
791
}