cpu.c 11.3 KB
Newer Older
J
Jiri Denemark 已提交
1 2 3
/*
 * cpu.c: internal functions for CPU manipulation
 *
E
Eric Blake 已提交
4
 * Copyright (C) 2009-2012 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 30
#include "cpu.h"
#include "cpu_x86.h"
P
Prerna Saxena 已提交
31
#include "cpu_powerpc.h"
T
Thang Pham 已提交
32
#include "cpu_s390.h"
C
Chuck Short 已提交
33
#include "cpu_arm.h"
J
Jiri Denemark 已提交
34 35 36 37 38 39 40 41
#include "cpu_generic.h"


#define NR_DRIVERS ARRAY_CARDINALITY(drivers)
#define VIR_FROM_THIS VIR_FROM_CPU

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


static struct cpuArchDriver *
51
cpuGetSubDriver(virArch arch)
J
Jiri Denemark 已提交
52
{
53 54
    size_t i;
    size_t j;
J
Jiri Denemark 已提交
55

56
    if (arch == VIR_ARCH_NONE) {
57 58
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("undefined hardware architecture"));
J
Jiri Denemark 已提交
59 60 61 62 63
        return NULL;
    }

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

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


virCPUCompareResult
75
cpuCompareXML(virCPUDefPtr host,
J
Jiri Denemark 已提交
76 77 78 79 80 81 82
              const char *xml)
{
    xmlDocPtr doc = NULL;
    xmlXPathContextPtr ctxt = NULL;
    virCPUDefPtr cpu = NULL;
    virCPUCompareResult ret = VIR_CPU_COMPARE_ERROR;

83 84
    VIR_DEBUG("host=%p, xml=%s", host, NULLSTR(xml));

85
    if (!(doc = virXMLParseStringCtxt(xml, _("(CPU_definition)"), &ctxt)))
86
        goto cleanup;
J
Jiri Denemark 已提交
87

88
    cpu = virCPUDefParseXML(ctxt->node, ctxt, VIR_CPU_TYPE_AUTO);
J
Jiri Denemark 已提交
89 90 91
    if (cpu == NULL)
        goto cleanup;

92
    if (!cpu->model) {
93 94
        virReportError(VIR_ERR_OPERATION_INVALID,
                       "%s", _("no CPU model specified"));
95 96 97
        goto cleanup;
    }

98
    ret = cpuCompare(host, cpu);
J
Jiri Denemark 已提交
99 100 101 102 103 104 105 106 107 108 109

cleanup:
    virCPUDefFree(cpu);
    xmlXPathFreeContext(ctxt);
    xmlFreeDoc(doc);

    return ret;
}


virCPUCompareResult
110
cpuCompare(virCPUDefPtr host,
J
Jiri Denemark 已提交
111 112 113 114
           virCPUDefPtr cpu)
{
    struct cpuArchDriver *driver;

115 116
    VIR_DEBUG("host=%p, cpu=%p", host, cpu);

117
    if ((driver = cpuGetSubDriver(host->arch)) == NULL)
J
Jiri Denemark 已提交
118 119 120
        return VIR_CPU_COMPARE_ERROR;

    if (driver->compare == NULL) {
121 122
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compare CPUs of %s architecture"),
123
                       virArchToString(host->arch));
J
Jiri Denemark 已提交
124 125 126 127 128 129 130 131
        return VIR_CPU_COMPARE_ERROR;
    }

    return driver->compare(host, cpu);
}


int
132
cpuDecode(virCPUDefPtr cpu,
133
          const virCPUDataPtr data,
134
          const char **models,
135 136
          unsigned int nmodels,
          const char *preferred)
J
Jiri Denemark 已提交
137 138 139
{
    struct cpuArchDriver *driver;

140 141
    VIR_DEBUG("cpu=%p, data=%p, nmodels=%u, preferred=%s",
              cpu, data, nmodels, NULLSTR(preferred));
142
    if (models) {
143
        size_t i;
144
        for (i = 0; i < nmodels; i++)
145
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
146 147
    }

148
    if (models == NULL && nmodels != 0) {
149 150
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero nmodels doesn't match with NULL models"));
151 152 153
        return -1;
    }

J
Jiri Denemark 已提交
154
    if (cpu == NULL) {
155 156
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("invalid CPU definition"));
J
Jiri Denemark 已提交
157 158 159
        return -1;
    }

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

    if (driver->decode == NULL) {
164 165
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot decode CPU data for %s architecture"),
166
                       virArchToString(cpu->arch));
J
Jiri Denemark 已提交
167 168 169
        return -1;
    }

170
    return driver->decode(cpu, data, models, nmodels, preferred, 0);
J
Jiri Denemark 已提交
171 172 173 174
}


int
175
cpuEncode(virArch arch,
J
Jiri Denemark 已提交
176
          const virCPUDefPtr cpu,
177 178 179 180 181 182
          virCPUDataPtr *forced,
          virCPUDataPtr *required,
          virCPUDataPtr *optional,
          virCPUDataPtr *disabled,
          virCPUDataPtr *forbidden,
          virCPUDataPtr *vendor)
J
Jiri Denemark 已提交
183 184 185
{
    struct cpuArchDriver *driver;

186
    VIR_DEBUG("arch=%s, cpu=%p, forced=%p, required=%p, "
J
Jiri Denemark 已提交
187
              "optional=%p, disabled=%p, forbidden=%p, vendor=%p",
188
              virArchToString(arch), cpu, forced, required,
J
Jiri Denemark 已提交
189
              optional, disabled, forbidden, vendor);
190

191
    if ((driver = cpuGetSubDriver(arch)) == NULL)
J
Jiri Denemark 已提交
192 193 194
        return -1;

    if (driver->encode == NULL) {
195 196
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot encode CPU data for %s architecture"),
197
                       virArchToString(arch));
J
Jiri Denemark 已提交
198 199 200
        return -1;
    }

J
Jiri Denemark 已提交
201
    return driver->encode(arch, cpu, forced, required,
J
Jiri Denemark 已提交
202
                          optional, disabled, forbidden, vendor);
J
Jiri Denemark 已提交
203 204 205 206
}


void
J
Jiri Denemark 已提交
207
cpuDataFree(virCPUDataPtr data)
J
Jiri Denemark 已提交
208 209 210
{
    struct cpuArchDriver *driver;

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

J
Jiri Denemark 已提交
213 214 215
    if (data == NULL)
        return;

J
Jiri Denemark 已提交
216
    if ((driver = cpuGetSubDriver(data->arch)) == NULL)
J
Jiri Denemark 已提交
217 218 219
        return;

    if (driver->free == NULL) {
220 221
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot free CPU data for %s architecture"),
J
Jiri Denemark 已提交
222
                       virArchToString(data->arch));
J
Jiri Denemark 已提交
223 224 225
        return;
    }

E
Eric Blake 已提交
226
    (driver->free)(data);
J
Jiri Denemark 已提交
227 228 229
}


230
virCPUDataPtr
231
cpuNodeData(virArch arch)
J
Jiri Denemark 已提交
232 233 234
{
    struct cpuArchDriver *driver;

235
    VIR_DEBUG("arch=%s", virArchToString(arch));
236

237
    if ((driver = cpuGetSubDriver(arch)) == NULL)
J
Jiri Denemark 已提交
238 239 240
        return NULL;

    if (driver->nodeData == NULL) {
241 242
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot get node CPU data for %s architecture"),
243
                       virArchToString(arch));
J
Jiri Denemark 已提交
244 245 246 247 248 249 250 251
        return NULL;
    }

    return driver->nodeData();
}


virCPUCompareResult
252
cpuGuestData(virCPUDefPtr host,
J
Jiri Denemark 已提交
253
             virCPUDefPtr guest,
254
             virCPUDataPtr *data,
255
             char **msg)
J
Jiri Denemark 已提交
256 257 258
{
    struct cpuArchDriver *driver;

259
    VIR_DEBUG("host=%p, guest=%p, data=%p, msg=%p", host, guest, data, msg);
260

261
    if ((driver = cpuGetSubDriver(host->arch)) == NULL)
J
Jiri Denemark 已提交
262 263 264
        return VIR_CPU_COMPARE_ERROR;

    if (driver->guestData == NULL) {
265 266
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compute guest CPU data for %s architecture"),
267
                       virArchToString(host->arch));
J
Jiri Denemark 已提交
268 269 270
        return VIR_CPU_COMPARE_ERROR;
    }

271
    return driver->guestData(host, guest, data, msg);
J
Jiri Denemark 已提交
272
}
273 274 275 276 277 278


char *
cpuBaselineXML(const char **xmlCPUs,
               unsigned int ncpus,
               const char **models,
279 280
               unsigned int nmodels,
               unsigned int flags)
281 282 283 284 285 286
{
    xmlDocPtr doc = NULL;
    xmlXPathContextPtr ctxt = NULL;
    virCPUDefPtr *cpus = NULL;
    virCPUDefPtr cpu = NULL;
    char *cpustr;
287
    size_t i;
288

289 290 291
    VIR_DEBUG("ncpus=%u, nmodels=%u", ncpus, nmodels);
    if (xmlCPUs) {
        for (i = 0; i < ncpus; i++)
292
            VIR_DEBUG("xmlCPUs[%zu]=%s", i, NULLSTR(xmlCPUs[i]));
293 294 295
    }
    if (models) {
        for (i = 0; i < nmodels; i++)
296
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
297 298
    }

299
    if (xmlCPUs == NULL && ncpus != 0) {
300 301
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero ncpus doesn't match with NULL xmlCPUs"));
302 303 304 305
        return NULL;
    }

    if (ncpus < 1) {
306
        virReportError(VIR_ERR_INVALID_ARG, "%s", _("No CPUs given"));
307 308 309 310
        return NULL;
    }

    if (VIR_ALLOC_N(cpus, ncpus))
311
        goto error;
312 313

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

317 318 319 320 321 322 323 324 325 326
        cpus[i] = virCPUDefParseXML(ctxt->node, ctxt, VIR_CPU_TYPE_HOST);
        if (cpus[i] == NULL)
            goto error;

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

327
    if (!(cpu = cpuBaseline(cpus, ncpus, models, nmodels, flags)))
328 329
        goto error;

330
    cpustr = virCPUDefFormat(cpu, 0);
331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353

cleanup:
    if (cpus) {
        for (i = 0; i < ncpus; i++)
            virCPUDefFree(cpus[i]);
        VIR_FREE(cpus);
    }
    virCPUDefFree(cpu);
    xmlXPathFreeContext(ctxt);
    xmlFreeDoc(doc);

    return cpustr;

error:
    cpustr = NULL;
    goto cleanup;
}


virCPUDefPtr
cpuBaseline(virCPUDefPtr *cpus,
            unsigned int ncpus,
            const char **models,
354 355
            unsigned int nmodels,
            unsigned int flags)
356 357
{
    struct cpuArchDriver *driver;
358
    size_t i;
359 360 361 362

    VIR_DEBUG("ncpus=%u, nmodels=%u", ncpus, nmodels);
    if (cpus) {
        for (i = 0; i < ncpus; i++)
363
            VIR_DEBUG("cpus[%zu]=%p", i, cpus[i]);
364 365 366
    }
    if (models) {
        for (i = 0; i < nmodels; i++)
367
            VIR_DEBUG("models[%zu]=%s", i, NULLSTR(models[i]));
368
    }
369 370

    if (cpus == NULL && ncpus != 0) {
371 372
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero ncpus doesn't match with NULL cpus"));
373 374 375 376
        return NULL;
    }

    if (ncpus < 1) {
377
        virReportError(VIR_ERR_INVALID_ARG, "%s", _("No CPUs given"));
378 379 380 381
        return NULL;
    }

    if (models == NULL && nmodels != 0) {
382 383
        virReportError(VIR_ERR_INTERNAL_ERROR,
                       "%s", _("nonzero nmodels doesn't match with NULL models"));
384 385 386 387 388 389 390
        return NULL;
    }

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

    if (driver->baseline == NULL) {
391 392
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot compute baseline CPU of %s architecture"),
393
                       virArchToString(cpus[0]->arch));
394 395 396
        return NULL;
    }

397
    return driver->baseline(cpus, ncpus, models, nmodels, flags);
398
}
399 400 401 402 403 404 405 406 407 408 409 410 411 412


int
cpuUpdate(virCPUDefPtr guest,
          const virCPUDefPtr host)
{
    struct cpuArchDriver *driver;

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

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

    if (driver->update == NULL) {
413 414
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot update guest CPU data for %s architecture"),
415
                       virArchToString(host->arch));
416 417 418 419 420
        return -1;
    }

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

int
J
Jiri Denemark 已提交
423
cpuHasFeature(const virCPUDataPtr data,
D
Daniel P. Berrange 已提交
424 425 426 427
              const char *feature)
{
    struct cpuArchDriver *driver;

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

J
Jiri Denemark 已提交
430
    if ((driver = cpuGetSubDriver(data->arch)) == NULL)
D
Daniel P. Berrange 已提交
431 432 433
        return -1;

    if (driver->hasFeature == NULL) {
434 435
        virReportError(VIR_ERR_NO_SUPPORT,
                       _("cannot check guest CPU data for %s architecture"),
J
Jiri Denemark 已提交
436
                       virArchToString(data->arch));
D
Daniel P. Berrange 已提交
437 438 439 440 441
        return -1;
    }

    return driver->hasFeature(data, feature);
}
442 443 444 445 446 447

bool
cpuModelIsAllowed(const char *model,
                  const char **models,
                  unsigned int nmodels)
{
448
    size_t i;
449 450 451 452 453 454 455 456 457 458

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

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