cpu_ppc64.c 16.8 KB
Newer Older
P
Prerna Saxena 已提交
1
/*
2
 * cpu_ppc64.c: CPU driver for 64-bit PowerPC CPUs
P
Prerna Saxena 已提交
3
 *
4
 * Copyright (C) 2013 Red Hat, Inc.
L
Li Zhang 已提交
5
 * Copyright (C) IBM Corporation, 2010
P
Prerna Saxena 已提交
6 7 8 9 10 11 12 13 14 15 16 17
 *
 * 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
18
 * License along with this library.  If not, see
O
Osier Yang 已提交
19
 * <http://www.gnu.org/licenses/>.
P
Prerna Saxena 已提交
20 21 22 23
 *
 * Authors:
 *      Anton Blanchard <anton@au.ibm.com>
 *      Prerna Saxena <prerna@linux.vnet.ibm.com>
L
Li Zhang 已提交
24
 *      Li Zhang <zhlcindy@linux.vnet.ibm.com>
P
Prerna Saxena 已提交
25 26 27
 */

#include <config.h>
L
Li Zhang 已提交
28
#include <stdint.h>
P
Prerna Saxena 已提交
29

30
#include "virlog.h"
31
#include "viralloc.h"
P
Prerna Saxena 已提交
32
#include "cpu.h"
33
#include "virstring.h"
L
Li Zhang 已提交
34
#include "cpu_map.h"
35
#include "virbuffer.h"
P
Prerna Saxena 已提交
36 37 38

#define VIR_FROM_THIS VIR_FROM_CPU

39
VIR_LOG_INIT("cpu.cpu_ppc64");
40

41
static const virArch archs[] = { VIR_ARCH_PPC64, VIR_ARCH_PPC64LE };
P
Prerna Saxena 已提交
42

43
struct ppc64_vendor {
L
Li Zhang 已提交
44
    char *name;
45
    struct ppc64_vendor *next;
L
Li Zhang 已提交
46 47
};

48
struct ppc64_model {
L
Li Zhang 已提交
49
    char *name;
50
    const struct ppc64_vendor *vendor;
51
    struct cpuPPC64Data data;
52
    struct ppc64_model *next;
L
Li Zhang 已提交
53 54
};

55 56 57
struct ppc64_map {
    struct ppc64_vendor *vendors;
    struct ppc64_model *models;
L
Li Zhang 已提交
58 59
};

60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85
static void
ppc64VendorFree(struct ppc64_vendor *vendor)
{
    if (!vendor)
        return;

    VIR_FREE(vendor->name);
    VIR_FREE(vendor);
}

static struct ppc64_vendor *
ppc64VendorFind(const struct ppc64_map *map,
                const char *name)
{
    struct ppc64_vendor *vendor;

    vendor = map->vendors;
    while (vendor) {
        if (STREQ(vendor->name, name))
            return vendor;

        vendor = vendor->next;
    }

    return NULL;
}
L
Li Zhang 已提交
86 87

static void
88
ppc64ModelFree(struct ppc64_model *model)
L
Li Zhang 已提交
89
{
90
    if (!model)
L
Li Zhang 已提交
91 92 93 94 95 96
        return;

    VIR_FREE(model->name);
    VIR_FREE(model);
}

97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
static struct ppc64_model *
ppc64ModelCopy(const struct ppc64_model *model)
{
    struct ppc64_model *copy;

    if (VIR_ALLOC(copy) < 0 ||
        VIR_STRDUP(copy->name, model->name) < 0) {
        ppc64ModelFree(copy);
        return NULL;
    }

    copy->data.pvr = model->data.pvr;
    copy->vendor = model->vendor;

    return copy;
}

114 115 116
static struct ppc64_model *
ppc64ModelFind(const struct ppc64_map *map,
               const char *name)
L
Li Zhang 已提交
117
{
118
    struct ppc64_model *model;
L
Li Zhang 已提交
119 120

    model = map->models;
121
    while (model) {
L
Li Zhang 已提交
122 123 124 125 126 127 128 129 130
        if (STREQ(model->name, name))
            return model;

        model = model->next;
    }

    return NULL;
}

131 132 133
static struct ppc64_model *
ppc64ModelFindPVR(const struct ppc64_map *map,
                  uint32_t pvr)
134
{
135
    struct ppc64_model *model;
136 137

    model = map->models;
138
    while (model) {
139 140 141 142 143 144
        if (model->data.pvr == pvr)
            return model;

        model = model->next;
    }

145 146 147
    /* PowerPC Processor Version Register is interpreted as follows :
     * Higher order 16 bits : Power ISA generation.
     * Lower order 16 bits : CPU chip version number.
M
Martin Kletzander 已提交
148
     * If the exact CPU isn't found, return the nearest matching CPU generation
149 150
     */
    if (pvr & 0x0000FFFFul)
151
        return ppc64ModelFindPVR(map, (pvr & 0xFFFF0000ul));
152

153 154 155
    return NULL;
}

156
static struct ppc64_model *
157 158
ppc64ModelFromCPU(const virCPUDef *cpu,
                  const struct ppc64_map *map)
159
{
160
    struct ppc64_model *model;
161

162 163 164
    if (!(model = ppc64ModelFind(map, cpu->model))) {
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Unknown CPU model %s"), cpu->model);
165 166 167
        return NULL;
    }

168
    return ppc64ModelCopy(model);
L
Li Zhang 已提交
169 170 171
}

static void
172
ppc64MapFree(struct ppc64_map *map)
L
Li Zhang 已提交
173
{
174
    if (!map)
L
Li Zhang 已提交
175 176
        return;

177 178 179 180 181
    while (map->models) {
        struct ppc64_model *model = map->models;
        map->models = model->next;
        ppc64ModelFree(model);
    }
182

183 184 185 186
    while (map->vendors) {
        struct ppc64_vendor *vendor = map->vendors;
        map->vendors = vendor->next;
        ppc64VendorFree(vendor);
187 188
    }

189
    VIR_FREE(map);
190 191
}

L
Li Zhang 已提交
192
static int
193 194
ppc64VendorLoad(xmlXPathContextPtr ctxt,
                struct ppc64_map *map)
L
Li Zhang 已提交
195
{
196
    struct ppc64_vendor *vendor;
L
Li Zhang 已提交
197

198
    if (VIR_ALLOC(vendor) < 0)
199
        return -1;
L
Li Zhang 已提交
200 201 202 203 204 205 206 207

    vendor->name = virXPathString("string(@name)", ctxt);
    if (!vendor->name) {
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("Missing CPU vendor name"));
        goto ignore;
    }

208
    if (ppc64VendorFind(map, vendor->name)) {
L
Li Zhang 已提交
209 210 211 212 213
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("CPU vendor %s already defined"), vendor->name);
        goto ignore;
    }

214
    if (!map->vendors) {
L
Li Zhang 已提交
215
        map->vendors = vendor;
216
    } else {
L
Li Zhang 已提交
217 218 219 220
        vendor->next = map->vendors;
        map->vendors = vendor;
    }

221
 cleanup:
222
    return 0;
L
Li Zhang 已提交
223

224
 ignore:
225
    ppc64VendorFree(vendor);
226
    goto cleanup;
L
Li Zhang 已提交
227 228 229
}

static int
230 231
ppc64ModelLoad(xmlXPathContextPtr ctxt,
               struct ppc64_map *map)
L
Li Zhang 已提交
232
{
233
    struct ppc64_model *model;
L
Li Zhang 已提交
234
    char *vendor = NULL;
235
    unsigned long pvr;
L
Li Zhang 已提交
236

237
    if (VIR_ALLOC(model) < 0)
238
        return -1;
L
Li Zhang 已提交
239 240

    model->name = virXPathString("string(@name)", ctxt);
241
    if (!model->name) {
L
Li Zhang 已提交
242 243 244 245 246
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("Missing CPU model name"));
        goto ignore;
    }

247
    if (ppc64ModelFind(map, model->name)) {
248 249 250 251
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("CPU model %s already defined"), model->name);
        goto ignore;
    }
L
Li Zhang 已提交
252 253 254 255 256 257 258 259 260 261

    if (virXPathBoolean("boolean(./vendor)", ctxt)) {
        vendor = virXPathString("string(./vendor/@name)", ctxt);
        if (!vendor) {
            virReportError(VIR_ERR_INTERNAL_ERROR,
                           _("Invalid vendor element in CPU model %s"),
                           model->name);
            goto ignore;
        }

262
        if (!(model->vendor = ppc64VendorFind(map, vendor))) {
L
Li Zhang 已提交
263 264 265 266 267 268 269
            virReportError(VIR_ERR_INTERNAL_ERROR,
                           _("Unknown vendor %s referenced by CPU model %s"),
                           vendor, model->name);
            goto ignore;
        }
    }

270 271 272 273 274 275 276 277 278
    if (!virXPathBoolean("boolean(./pvr)", ctxt) ||
        virXPathULongHex("string(./pvr/@value)", ctxt, &pvr) < 0) {
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Missing or invalid PVR value in CPU model %s"),
                       model->name);
        goto ignore;
    }
    model->data.pvr = pvr;

279
    if (!map->models) {
L
Li Zhang 已提交
280
        map->models = model;
281
    } else {
L
Li Zhang 已提交
282 283 284 285
        model->next = map->models;
        map->models = model;
    }

286
 cleanup:
L
Li Zhang 已提交
287
    VIR_FREE(vendor);
288
    return 0;
L
Li Zhang 已提交
289

290
 ignore:
291
    ppc64ModelFree(model);
292
    goto cleanup;
L
Li Zhang 已提交
293 294 295
}

static int
296 297 298
ppc64MapLoadCallback(cpuMapElement element,
                     xmlXPathContextPtr ctxt,
                     void *data)
L
Li Zhang 已提交
299
{
300
    struct ppc64_map *map = data;
L
Li Zhang 已提交
301 302 303

    switch (element) {
    case CPU_MAP_ELEMENT_VENDOR:
304
        return ppc64VendorLoad(ctxt, map);
L
Li Zhang 已提交
305
    case CPU_MAP_ELEMENT_MODEL:
306
        return ppc64ModelLoad(ctxt, map);
307 308
    case CPU_MAP_ELEMENT_FEATURE:
    case CPU_MAP_ELEMENT_LAST:
L
Li Zhang 已提交
309 310 311 312 313 314
        break;
    }

    return 0;
}

315 316
static struct ppc64_map *
ppc64LoadMap(void)
L
Li Zhang 已提交
317
{
318
    struct ppc64_map *map;
L
Li Zhang 已提交
319

320
    if (VIR_ALLOC(map) < 0)
L
Li Zhang 已提交
321 322
        return NULL;

323
    if (cpuMapLoad("ppc64", ppc64MapLoadCallback, map) < 0)
L
Li Zhang 已提交
324 325 326 327
        goto error;

    return map;

328
 error:
329
    ppc64MapFree(map);
L
Li Zhang 已提交
330 331 332
    return NULL;
}

333
static virCPUDataPtr
334 335
ppc64MakeCPUData(virArch arch,
                 struct cpuPPC64Data *data)
336 337 338 339 340 341 342
{
    virCPUDataPtr cpuData;

    if (VIR_ALLOC(cpuData) < 0)
        return NULL;

    cpuData->arch = arch;
343
    cpuData->data.ppc64 = *data;
344 345 346 347 348

    return cpuData;
}

static virCPUCompareResult
349 350 351 352
ppc64Compute(virCPUDefPtr host,
             const virCPUDef *cpu,
             virCPUDataPtr *guestData,
             char **message)
353 354

{
355 356 357
    struct ppc64_map *map = NULL;
    struct ppc64_model *host_model = NULL;
    struct ppc64_model *guest_model = NULL;
358

359
    virCPUCompareResult ret = VIR_CPU_COMPARE_ERROR;
360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379
    virArch arch;
    size_t i;

    if (cpu->arch != VIR_ARCH_NONE) {
        bool found = false;

        for (i = 0; i < ARRAY_CARDINALITY(archs); i++) {
            if (archs[i] == cpu->arch) {
                found = true;
                break;
            }
        }

        if (!found) {
            VIR_DEBUG("CPU arch %s does not match host arch",
                      virArchToString(cpu->arch));
            if (message &&
                virAsprintf(message,
                            _("CPU arch %s does not match host arch"),
                            virArchToString(cpu->arch)) < 0)
380 381 382 383
                goto cleanup;

            ret = VIR_CPU_COMPARE_INCOMPATIBLE;
            goto cleanup;
384 385 386 387 388 389 390 391 392 393 394 395 396 397 398
        }
        arch = cpu->arch;
    } else {
        arch = host->arch;
    }

    if (cpu->vendor &&
        (!host->vendor || STRNEQ(cpu->vendor, host->vendor))) {
        VIR_DEBUG("host CPU vendor does not match required CPU vendor %s",
                  cpu->vendor);
        if (message &&
            virAsprintf(message,
                        _("host CPU vendor does not match required "
                        "CPU vendor %s"),
                        cpu->vendor) < 0)
399 400 401 402
            goto cleanup;

        ret = VIR_CPU_COMPARE_INCOMPATIBLE;
        goto cleanup;
403 404
    }

405 406 407
    if (!(map = ppc64LoadMap()) ||
        !(host_model = ppc64ModelFromCPU(host, map)) ||
        !(guest_model = ppc64ModelFromCPU(cpu, map)))
408
        goto cleanup;
409

410
    if (guestData) {
411 412 413 414 415 416 417 418 419 420
        if (cpu->type == VIR_CPU_TYPE_GUEST &&
            cpu->match == VIR_CPU_MATCH_STRICT &&
            STRNEQ(guest_model->name, host_model->name)) {
            VIR_DEBUG("host CPU model does not match required CPU model %s",
                      guest_model->name);
            if (message &&
                virAsprintf(message,
                            _("host CPU model does not match required "
                            "CPU model %s"),
                            guest_model->name) < 0)
421 422 423 424
                goto cleanup;

            ret = VIR_CPU_COMPARE_INCOMPATIBLE;
            goto cleanup;
425 426
        }

427
        if (!(*guestData = ppc64MakeCPUData(arch, &guest_model->data)))
428
            goto cleanup;
429 430 431 432
    }

    ret = VIR_CPU_COMPARE_IDENTICAL;

433
 cleanup:
434 435 436
    ppc64MapFree(map);
    ppc64ModelFree(host_model);
    ppc64ModelFree(guest_model);
437 438 439
    return ret;
}

L
Li Zhang 已提交
440
static virCPUCompareResult
441 442 443
ppc64DriverCompare(virCPUDefPtr host,
                   virCPUDefPtr cpu,
                   bool failIncompatible)
L
Li Zhang 已提交
444
{
445 446 447
    if ((cpu->arch == VIR_ARCH_NONE || host->arch == cpu->arch) &&
        STREQ(host->model, cpu->model))
        return VIR_CPU_COMPARE_IDENTICAL;
L
Li Zhang 已提交
448

449 450 451 452 453 454
    if (failIncompatible) {
        virReportError(VIR_ERR_CPU_INCOMPATIBLE, NULL);
        return VIR_CPU_COMPARE_ERROR;
    } else {
        return VIR_CPU_COMPARE_INCOMPATIBLE;
    }
L
Li Zhang 已提交
455 456 457
}

static int
458 459 460 461 462 463
ppc64DriverDecode(virCPUDefPtr cpu,
                  const virCPUData *data,
                  const char **models,
                  unsigned int nmodels,
                  const char *preferred ATTRIBUTE_UNUSED,
                  unsigned int flags)
L
Li Zhang 已提交
464 465
{
    int ret = -1;
466 467
    struct ppc64_map *map;
    const struct ppc64_model *model;
L
Li Zhang 已提交
468

469 470
    virCheckFlags(VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES, -1);

471
    if (!data || !(map = ppc64LoadMap()))
L
Li Zhang 已提交
472 473
        return -1;

474
    if (!(model = ppc64ModelFindPVR(map, data->data.ppc64.pvr))) {
475 476
        virReportError(VIR_ERR_OPERATION_FAILED,
                       _("Cannot find CPU model with PVR 0x%08x"),
477
                       data->data.ppc64.pvr);
J
Jiri Denemark 已提交
478
        goto cleanup;
479
    }
L
Li Zhang 已提交
480

481
    if (!cpuModelIsAllowed(model->name, models, nmodels)) {
J
Jiri Denemark 已提交
482 483
        virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
                       _("CPU model %s is not supported by hypervisor"),
484
                       model->name);
J
Jiri Denemark 已提交
485
        goto cleanup;
L
Li Zhang 已提交
486 487
    }

488 489
    if (VIR_STRDUP(cpu->model, model->name) < 0 ||
        (model->vendor && VIR_STRDUP(cpu->vendor, model->vendor->name) < 0)) {
490 491
        goto cleanup;
    }
L
Li Zhang 已提交
492 493 494

    ret = 0;

495
 cleanup:
496
    ppc64MapFree(map);
L
Li Zhang 已提交
497 498 499 500 501

    return ret;
}

static void
502
ppc64DriverFree(virCPUDataPtr data)
L
Li Zhang 已提交
503
{
504
    if (!data)
L
Li Zhang 已提交
505 506 507 508 509
        return;

    VIR_FREE(data);
}

510
static virCPUDataPtr
511
ppc64DriverNodeData(virArch arch)
P
Prerna Saxena 已提交
512
{
L
Li Zhang 已提交
513
    virCPUDataPtr cpuData;
P
Prerna Saxena 已提交
514

L
Li Zhang 已提交
515
    if (VIR_ALLOC(cpuData) < 0)
P
Prerna Saxena 已提交
516 517
        return NULL;

518 519 520
    cpuData->arch = arch;

#if defined(__powerpc__) || defined(__powerpc64__)
J
Jiri Denemark 已提交
521
    asm("mfpvr %0"
522
        : "=r" (cpuData->data.ppc64.pvr));
523
#endif
L
Li Zhang 已提交
524

L
Li Zhang 已提交
525
    return cpuData;
P
Prerna Saxena 已提交
526 527
}

528
static virCPUCompareResult
529 530 531 532
ppc64DriverGuestData(virCPUDefPtr host,
                     virCPUDefPtr guest,
                     virCPUDataPtr *data,
                     char **message)
533
{
534
    return ppc64Compute(host, guest, data, message);
535 536
}

P
Prerna Saxena 已提交
537
static int
538 539
ppc64DriverUpdate(virCPUDefPtr guest,
                  const virCPUDef *host)
P
Prerna Saxena 已提交
540
{
541
    switch ((virCPUMode) guest->mode) {
542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557
    case VIR_CPU_MODE_HOST_MODEL:
    case VIR_CPU_MODE_HOST_PASSTHROUGH:
        guest->match = VIR_CPU_MATCH_EXACT;
        virCPUDefFreeModel(guest);
        return virCPUDefCopyModel(guest, host, true);

    case VIR_CPU_MODE_CUSTOM:
        return 0;

    case VIR_CPU_MODE_LAST:
        break;
    }

    virReportError(VIR_ERR_INTERNAL_ERROR,
                   _("Unexpected CPU mode: %d"), guest->mode);
    return -1;
P
Prerna Saxena 已提交
558
}
559

L
Li Zhang 已提交
560
static virCPUDefPtr
561 562 563 564 565
ppc64DriverBaseline(virCPUDefPtr *cpus,
                    unsigned int ncpus,
                    const char **models ATTRIBUTE_UNUSED,
                    unsigned int nmodels ATTRIBUTE_UNUSED,
                    unsigned int flags)
P
Prerna Saxena 已提交
566
{
567
    struct ppc64_map *map;
568 569
    const struct ppc64_model *model;
    const struct ppc64_vendor *vendor = NULL;
L
Li Zhang 已提交
570
    virCPUDefPtr cpu = NULL;
571
    size_t i;
J
Jiri Denemark 已提交
572

573 574
    virCheckFlags(VIR_CONNECT_BASELINE_CPU_EXPAND_FEATURES |
                  VIR_CONNECT_BASELINE_CPU_MIGRATABLE, NULL);
575

576
    if (!(map = ppc64LoadMap()))
J
Jiri Denemark 已提交
577
        goto error;
L
Li Zhang 已提交
578

579
    if (!(model = ppc64ModelFind(map, cpus[0]->model))) {
J
Jiri Denemark 已提交
580 581
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       _("Unknown CPU model %s"), cpus[0]->model);
L
Li Zhang 已提交
582 583 584
        goto error;
    }

J
Jiri Denemark 已提交
585
    for (i = 0; i < ncpus; i++) {
586
        const struct ppc64_vendor *vnd;
587

J
Jiri Denemark 已提交
588 589 590 591 592
        if (STRNEQ(cpus[i]->model, model->name)) {
            virReportError(VIR_ERR_OPERATION_FAILED, "%s",
                           _("CPUs are incompatible"));
            goto error;
        }
L
Li Zhang 已提交
593

J
Jiri Denemark 已提交
594 595 596
        if (!cpus[i]->vendor)
            continue;

597
        if (!(vnd = ppc64VendorFind(map, cpus[i]->vendor))) {
J
Jiri Denemark 已提交
598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620
            virReportError(VIR_ERR_OPERATION_FAILED,
                           _("Unknown CPU vendor %s"), cpus[i]->vendor);
            goto error;
        }

        if (model->vendor) {
            if (model->vendor != vnd) {
                virReportError(VIR_ERR_OPERATION_FAILED,
                               _("CPU vendor %s of model %s differs from "
                                 "vendor %s"),
                               model->vendor->name, model->name,
                               vnd->name);
                goto error;
            }
        } else if (vendor) {
            if (vendor != vnd) {
                virReportError(VIR_ERR_OPERATION_FAILED, "%s",
                               _("CPU vendors do not match"));
                goto error;
            }
        } else {
            vendor = vnd;
        }
L
Li Zhang 已提交
621 622
    }

J
Jiri Denemark 已提交
623
    if (VIR_ALLOC(cpu) < 0 ||
624
        VIR_STRDUP(cpu->model, model->name) < 0)
625
        goto error;
L
Li Zhang 已提交
626

627 628
    if (vendor && VIR_STRDUP(cpu->vendor, vendor->name) < 0)
        goto error;
J
Jiri Denemark 已提交
629 630 631

    cpu->type = VIR_CPU_TYPE_GUEST;
    cpu->match = VIR_CPU_MATCH_EXACT;
L
Li Zhang 已提交
632

633
 cleanup:
634
    ppc64MapFree(map);
P
Prerna Saxena 已提交
635

L
Li Zhang 已提交
636
    return cpu;
J
Jiri Denemark 已提交
637

638
 error:
L
Li Zhang 已提交
639 640 641
    virCPUDefFree(cpu);
    cpu = NULL;
    goto cleanup;
P
Prerna Saxena 已提交
642 643
}

644
static int
645
ppc64DriverGetModels(char ***models)
646
{
647 648
    struct ppc64_map *map;
    struct ppc64_model *model;
649 650 651
    char *name;
    size_t nmodels = 0;

652
    if (!(map = ppc64LoadMap()))
653 654 655 656 657 658
        goto error;

    if (models && VIR_ALLOC_N(*models, 0) < 0)
        goto error;

    model = map->models;
659
    while (model) {
660 661 662
        if (models) {
            if (VIR_STRDUP(name, model->name) < 0)
                goto error;
663

664 665 666 667 668
            if (VIR_APPEND_ELEMENT(*models, nmodels, name) < 0)
                goto error;
        } else {
            nmodels++;
        }
669 670 671 672 673

        model = model->next;
    }

 cleanup:
674
    ppc64MapFree(map);
675 676 677 678

    return nmodels;

 error:
679 680 681 682
    if (models) {
        virStringFreeList(*models);
        *models = NULL;
    }
683 684 685 686
    nmodels = -1;
    goto cleanup;
}

687
struct cpuArchDriver cpuDriverPPC64 = {
688 689 690
    .name       = "ppc64",
    .arch       = archs,
    .narch      = ARRAY_CARDINALITY(archs),
691 692
    .compare    = ppc64DriverCompare,
    .decode     = ppc64DriverDecode,
P
Prerna Saxena 已提交
693
    .encode     = NULL,
694 695 696 697 698
    .free       = ppc64DriverFree,
    .nodeData   = ppc64DriverNodeData,
    .guestData  = ppc64DriverGuestData,
    .baseline   = ppc64DriverBaseline,
    .update     = ppc64DriverUpdate,
P
Prerna Saxena 已提交
699
    .hasFeature = NULL,
700
    .getModels  = ppc64DriverGetModels,
P
Prerna Saxena 已提交
701
};