apic_common.c 12.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
/*
 *  APIC support - common bits of emulated and KVM kernel model
 *
 *  Copyright (c) 2004-2005 Fabrice Bellard
 *  Copyright (c) 2011      Jan Kiszka, Siemens AG
 *
 * 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 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/>
 */
P
Paolo Bonzini 已提交
20 21
#include "hw/i386/apic.h"
#include "hw/i386/apic_internal.h"
22
#include "trace.h"
23
#include "sysemu/kvm.h"
24 25
#include "hw/qdev.h"
#include "hw/sysbus.h"
26 27

static int apic_irq_delivered;
28
bool apic_report_tpr_access;
29

X
xiaoqiang zhao 已提交
30
void cpu_set_apic_base(DeviceState *dev, uint64_t val)
31 32 33
{
    trace_cpu_set_apic_base(val);

X
xiaoqiang zhao 已提交
34 35
    if (dev) {
        APICCommonState *s = APIC_COMMON(dev);
36
        APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
37 38 39 40
        info->set_base(s, val);
    }
}

X
xiaoqiang zhao 已提交
41
uint64_t cpu_get_apic_base(DeviceState *dev)
42
{
X
xiaoqiang zhao 已提交
43 44
    if (dev) {
        APICCommonState *s = APIC_COMMON(dev);
45 46 47
        trace_cpu_get_apic_base((uint64_t)s->apicbase);
        return s->apicbase;
    } else {
48 49
        trace_cpu_get_apic_base(MSR_IA32_APICBASE_BSP);
        return MSR_IA32_APICBASE_BSP;
50
    }
51 52
}

X
xiaoqiang zhao 已提交
53
void cpu_set_apic_tpr(DeviceState *dev, uint8_t val)
54
{
55 56
    APICCommonState *s;
    APICCommonClass *info;
57

X
xiaoqiang zhao 已提交
58
    if (!dev) {
59
        return;
60
    }
61

X
xiaoqiang zhao 已提交
62
    s = APIC_COMMON(dev);
63 64 65
    info = APIC_COMMON_GET_CLASS(s);

    info->set_tpr(s, val);
66 67
}

X
xiaoqiang zhao 已提交
68
uint8_t cpu_get_apic_tpr(DeviceState *dev)
69 70 71 72
{
    APICCommonState *s;
    APICCommonClass *info;

X
xiaoqiang zhao 已提交
73
    if (!dev) {
74 75 76
        return 0;
    }

X
xiaoqiang zhao 已提交
77
    s = APIC_COMMON(dev);
78 79 80 81 82
    info = APIC_COMMON_GET_CLASS(s);

    return info->get_tpr(s);
}

X
xiaoqiang zhao 已提交
83
void apic_enable_tpr_access_reporting(DeviceState *dev, bool enable)
84
{
X
xiaoqiang zhao 已提交
85
    APICCommonState *s = APIC_COMMON(dev);
86 87 88 89 90 91 92 93
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);

    apic_report_tpr_access = enable;
    if (info->enable_tpr_reporting) {
        info->enable_tpr_reporting(s, enable);
    }
}

X
xiaoqiang zhao 已提交
94
void apic_enable_vapic(DeviceState *dev, hwaddr paddr)
95
{
X
xiaoqiang zhao 已提交
96
    APICCommonState *s = APIC_COMMON(dev);
97
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
98

99 100
    s->vapic_paddr = paddr;
    info->vapic_base_update(s);
101 102
}

X
xiaoqiang zhao 已提交
103
void apic_handle_tpr_access_report(DeviceState *dev, target_ulong ip,
104 105
                                   TPRAccess access)
{
X
xiaoqiang zhao 已提交
106
    APICCommonState *s = APIC_COMMON(dev);
107

108
    vapic_report_tpr_access(s->vapic, CPU(s->cpu), ip, access);
109 110
}

111 112 113 114 115 116 117 118 119
void apic_report_irq_delivered(int delivered)
{
    apic_irq_delivered += delivered;

    trace_apic_report_irq_delivered(apic_irq_delivered);
}

void apic_reset_irq_delivered(void)
{
120 121 122 123 124 125
    /* Copy this into a local variable to encourage gcc to emit a plain
     * register for a sys/sdt.h marker.  For details on this workaround, see:
     * https://sourceware.org/bugzilla/show_bug.cgi?id=13296
     */
    volatile int a_i_d = apic_irq_delivered;
    trace_apic_reset_irq_delivered(a_i_d);
126 127 128 129 130 131 132 133 134 135 136

    apic_irq_delivered = 0;
}

int apic_get_irq_delivered(void)
{
    trace_apic_get_irq_delivered(apic_irq_delivered);

    return apic_irq_delivered;
}

X
xiaoqiang zhao 已提交
137
void apic_deliver_nmi(DeviceState *dev)
138
{
X
xiaoqiang zhao 已提交
139
    APICCommonState *s = APIC_COMMON(dev);
140
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
141 142 143 144

    info->external_nmi(s);
}

J
Jan Kiszka 已提交
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
bool apic_next_timer(APICCommonState *s, int64_t current_time)
{
    int64_t d;

    /* We need to store the timer state separately to support APIC
     * implementations that maintain a non-QEMU timer, e.g. inside the
     * host kernel. This open-coded state allows us to migrate between
     * both models. */
    s->timer_expiry = -1;

    if (s->lvt[APIC_LVT_TIMER] & APIC_LVT_MASKED) {
        return false;
    }

    d = (current_time - s->initial_count_load_time) >> s->count_shift;

    if (s->lvt[APIC_LVT_TIMER] & APIC_LVT_TIMER_PERIODIC) {
        if (!s->initial_count) {
            return false;
        }
        d = ((d / ((uint64_t)s->initial_count + 1)) + 1) *
            ((uint64_t)s->initial_count + 1);
    } else {
        if (d >= s->initial_count) {
            return false;
        }
        d = (uint64_t)s->initial_count + 1;
    }
    s->next_time = s->initial_count_load_time + (d << s->count_shift);
    s->timer_expiry = s->next_time;
    return true;
}

X
xiaoqiang zhao 已提交
178
void apic_init_reset(DeviceState *dev)
179
{
X
xiaoqiang zhao 已提交
180
    APICCommonState *s = APIC_COMMON(dev);
P
Paolo Bonzini 已提交
181
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203
    int i;

    if (!s) {
        return;
    }
    s->tpr = 0;
    s->spurious_vec = 0xff;
    s->log_dest = 0;
    s->dest_mode = 0xf;
    memset(s->isr, 0, sizeof(s->isr));
    memset(s->tmr, 0, sizeof(s->tmr));
    memset(s->irr, 0, sizeof(s->irr));
    for (i = 0; i < APIC_LVT_NB; i++) {
        s->lvt[i] = APIC_LVT_MASKED;
    }
    s->esr = 0;
    memset(s->icr, 0, sizeof(s->icr));
    s->divide_conf = 0;
    s->count_shift = 0;
    s->initial_count = 0;
    s->initial_count_load_time = 0;
    s->next_time = 0;
204
    s->wait_for_sipi = !cpu_is_bsp(s->cpu);
205

J
Jan Kiszka 已提交
206
    if (s->timer) {
207
        timer_del(s->timer);
J
Jan Kiszka 已提交
208 209
    }
    s->timer_expiry = -1;
P
Paolo Bonzini 已提交
210 211 212 213

    if (info->reset) {
        info->reset(s);
    }
214 215
}

X
xiaoqiang zhao 已提交
216
void apic_designate_bsp(DeviceState *dev)
217
{
X
xiaoqiang zhao 已提交
218
    if (dev == NULL) {
219 220 221
        return;
    }

X
xiaoqiang zhao 已提交
222
    APICCommonState *s = APIC_COMMON(dev);
223 224 225
    s->apicbase |= MSR_IA32_APICBASE_BSP;
}

X
xiaoqiang zhao 已提交
226
static void apic_reset_common(DeviceState *dev)
227
{
X
xiaoqiang zhao 已提交
228
    APICCommonState *s = APIC_COMMON(dev);
229
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
230 231
    bool bsp;

232
    bsp = cpu_is_bsp(s->cpu);
233
    s->apicbase = APIC_DEFAULT_ADDRESS |
234 235
        (bsp ? MSR_IA32_APICBASE_BSP : 0) | MSR_IA32_APICBASE_ENABLE;

236 237 238
    s->vapic_paddr = 0;
    info->vapic_base_update(s);

X
xiaoqiang zhao 已提交
239
    apic_init_reset(dev);
240 241 242 243 244 245 246 247 248 249 250 251 252 253 254

    if (bsp) {
        /*
         * LINT0 delivery mode on CPU #0 is set to ExtInt at initialization
         * time typically by BIOS, so PIC interrupt can be delivered to the
         * processor when local APIC is enabled.
         */
        s->lvt[APIC_LVT_LINT0] = 0x700;
    }
}

/* This function is only used for old state version 1 and 2 */
static int apic_load_old(QEMUFile *f, void *opaque, int version_id)
{
    APICCommonState *s = opaque;
255
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287
    int i;

    if (version_id > 2) {
        return -EINVAL;
    }

    /* XXX: what if the base changes? (registered memory regions) */
    qemu_get_be32s(f, &s->apicbase);
    qemu_get_8s(f, &s->id);
    qemu_get_8s(f, &s->arb_id);
    qemu_get_8s(f, &s->tpr);
    qemu_get_be32s(f, &s->spurious_vec);
    qemu_get_8s(f, &s->log_dest);
    qemu_get_8s(f, &s->dest_mode);
    for (i = 0; i < 8; i++) {
        qemu_get_be32s(f, &s->isr[i]);
        qemu_get_be32s(f, &s->tmr[i]);
        qemu_get_be32s(f, &s->irr[i]);
    }
    for (i = 0; i < APIC_LVT_NB; i++) {
        qemu_get_be32s(f, &s->lvt[i]);
    }
    qemu_get_be32s(f, &s->esr);
    qemu_get_be32s(f, &s->icr[0]);
    qemu_get_be32s(f, &s->icr[1]);
    qemu_get_be32s(f, &s->divide_conf);
    s->count_shift = qemu_get_be32(f);
    qemu_get_be32s(f, &s->initial_count);
    s->initial_count_load_time = qemu_get_be64(f);
    s->next_time = qemu_get_be64(f);

    if (version_id >= 2) {
288 289 290 291 292
        s->timer_expiry = qemu_get_be64(f);
    }

    if (info->post_load) {
        info->post_load(s);
293 294 295 296
    }
    return 0;
}

X
xiaoqiang zhao 已提交
297
static void apic_common_realize(DeviceState *dev, Error **errp)
298
{
299 300
    APICCommonState *s = APIC_COMMON(dev);
    APICCommonClass *info;
301
    static DeviceState *vapic;
302
    static int apic_no;
303
    static bool mmio_registered;
304 305

    if (apic_no >= MAX_APICS) {
X
xiaoqiang zhao 已提交
306 307 308
        error_setg(errp, "%s initialization failed.",
                   object_get_typename(OBJECT(dev)));
        return;
309 310 311
    }
    s->idx = apic_no++;

312
    info = APIC_COMMON_GET_CLASS(s);
X
xiaoqiang zhao 已提交
313
    info->realize(dev, errp);
314
    if (!mmio_registered) {
X
xiaoqiang zhao 已提交
315
        ICCBus *b = ICC_BUS(qdev_get_parent_bus(dev));
316 317 318
        memory_region_add_subregion(b->apic_address_space, 0, &s->io_memory);
        mmio_registered = true;
    }
319

320 321 322
    /* Note: We need at least 1M to map the VAPIC option ROM */
    if (!vapic && s->vapic_control & VAPIC_ENABLE_MASK &&
        ram_size >= 1024 * 1024) {
323 324 325 326 327 328 329
        vapic = sysbus_create_simple("kvmvapic", -1, NULL);
    }
    s->vapic = vapic;
    if (apic_report_tpr_access && info->enable_tpr_reporting) {
        info->enable_tpr_reporting(s, true);
    }

330 331
}

332 333 334 335 336 337 338 339 340 341 342 343 344
static int apic_pre_load(void *opaque)
{
    APICCommonState *s = APIC_COMMON(opaque);

    /* The default is !cpu_is_bsp(s->cpu), but the common value is 0
     * so that's what apic_common_sipi_needed checks for.  Reset to
     * the value that is assumed when the apic_sipi subsection is
     * absent.
     */
    s->wait_for_sipi = 0;
    return 0;
}

345 346 347 348 349 350 351 352 353 354
static void apic_dispatch_pre_save(void *opaque)
{
    APICCommonState *s = APIC_COMMON(opaque);
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);

    if (info->pre_save) {
        info->pre_save(s);
    }
}

J
Jan Kiszka 已提交
355 356
static int apic_dispatch_post_load(void *opaque, int version_id)
{
357 358
    APICCommonState *s = APIC_COMMON(opaque);
    APICCommonClass *info = APIC_COMMON_GET_CLASS(s);
J
Jan Kiszka 已提交
359 360 361 362 363 364 365

    if (info->post_load) {
        info->post_load(s);
    }
    return 0;
}

366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382
static bool apic_common_sipi_needed(void *opaque)
{
    APICCommonState *s = APIC_COMMON(opaque);
    return s->wait_for_sipi != 0;
}

static const VMStateDescription vmstate_apic_common_sipi = {
    .name = "apic_sipi",
    .version_id = 1,
    .minimum_version_id = 1,
    .fields = (VMStateField[]) {
        VMSTATE_INT32(sipi_vector, APICCommonState),
        VMSTATE_INT32(wait_for_sipi, APICCommonState),
        VMSTATE_END_OF_LIST()
    }
};

383 384 385 386 387 388
static const VMStateDescription vmstate_apic_common = {
    .name = "apic",
    .version_id = 3,
    .minimum_version_id = 3,
    .minimum_version_id_old = 1,
    .load_state_old = apic_load_old,
389
    .pre_load = apic_pre_load,
390
    .pre_save = apic_dispatch_pre_save,
J
Jan Kiszka 已提交
391
    .post_load = apic_dispatch_post_load,
392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410
    .fields = (VMStateField[]) {
        VMSTATE_UINT32(apicbase, APICCommonState),
        VMSTATE_UINT8(id, APICCommonState),
        VMSTATE_UINT8(arb_id, APICCommonState),
        VMSTATE_UINT8(tpr, APICCommonState),
        VMSTATE_UINT32(spurious_vec, APICCommonState),
        VMSTATE_UINT8(log_dest, APICCommonState),
        VMSTATE_UINT8(dest_mode, APICCommonState),
        VMSTATE_UINT32_ARRAY(isr, APICCommonState, 8),
        VMSTATE_UINT32_ARRAY(tmr, APICCommonState, 8),
        VMSTATE_UINT32_ARRAY(irr, APICCommonState, 8),
        VMSTATE_UINT32_ARRAY(lvt, APICCommonState, APIC_LVT_NB),
        VMSTATE_UINT32(esr, APICCommonState),
        VMSTATE_UINT32_ARRAY(icr, APICCommonState, 2),
        VMSTATE_UINT32(divide_conf, APICCommonState),
        VMSTATE_INT32(count_shift, APICCommonState),
        VMSTATE_UINT32(initial_count, APICCommonState),
        VMSTATE_INT64(initial_count_load_time, APICCommonState),
        VMSTATE_INT64(next_time, APICCommonState),
J
Jan Kiszka 已提交
411 412
        VMSTATE_INT64(timer_expiry,
                      APICCommonState), /* open-coded timer state */
413
        VMSTATE_END_OF_LIST()
414 415 416 417 418 419 420
    },
    .subsections = (VMStateSubsection[]) {
        {
            .vmsd = &vmstate_apic_common_sipi,
            .needed = apic_common_sipi_needed,
        },
        VMSTATE_END_OF_LIST()
421 422 423 424 425
    }
};

static Property apic_properties_common[] = {
    DEFINE_PROP_UINT8("id", APICCommonState, id, -1),
426
    DEFINE_PROP_UINT8("version", APICCommonState, version, 0x14),
427 428
    DEFINE_PROP_BIT("vapic", APICCommonState, vapic_control, VAPIC_ENABLE_BIT,
                    true),
429 430 431
    DEFINE_PROP_END_OF_LIST(),
};

432 433
static void apic_common_class_init(ObjectClass *klass, void *data)
{
434
    ICCDeviceClass *idc = ICC_DEVICE_CLASS(klass);
435
    DeviceClass *dc = DEVICE_CLASS(klass);
436

437 438 439
    dc->vmsd = &vmstate_apic_common;
    dc->reset = apic_reset_common;
    dc->props = apic_properties_common;
X
xiaoqiang zhao 已提交
440
    idc->realize = apic_common_realize;
441 442 443 444 445
    /*
     * Reason: APIC and CPU need to be wired up by
     * x86_cpu_apic_create()
     */
    dc->cannot_instantiate_with_device_add_yet = true;
446
}
447

448
static const TypeInfo apic_common_type = {
449
    .name = TYPE_APIC_COMMON,
450
    .parent = TYPE_ICC_DEVICE,
451 452 453 454 455 456
    .instance_size = sizeof(APICCommonState),
    .class_size = sizeof(APICCommonClass),
    .class_init = apic_common_class_init,
    .abstract = true,
};

X
xiaoqiang zhao 已提交
457
static void apic_common_register_types(void)
458 459 460 461
{
    type_register_static(&apic_common_type);
}

X
xiaoqiang zhao 已提交
462
type_init(apic_common_register_types)