2005-01-04 02:27:31 +03:00
|
|
|
/*
|
|
|
|
* APIC support
|
2007-09-17 01:08:06 +04:00
|
|
|
*
|
2005-01-04 02:27:31 +03:00
|
|
|
* Copyright (c) 2004-2005 Fabrice Bellard
|
|
|
|
*
|
|
|
|
* 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
|
2020-10-23 15:44:24 +03:00
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
2005-01-04 02:27:31 +03:00
|
|
|
*
|
|
|
|
* 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
|
2009-07-17 00:47:01 +04:00
|
|
|
* License along with this library; if not, see <http://www.gnu.org/licenses/>
|
2005-01-04 02:27:31 +03:00
|
|
|
*/
|
2016-01-26 21:17:03 +03:00
|
|
|
#include "qemu/osdep.h"
|
2012-12-17 21:20:00 +04:00
|
|
|
#include "qemu/thread.h"
|
2023-03-15 20:43:13 +03:00
|
|
|
#include "qemu/error-report.h"
|
2013-02-05 20:06:20 +04:00
|
|
|
#include "hw/i386/apic_internal.h"
|
|
|
|
#include "hw/i386/apic.h"
|
2023-02-13 20:30:31 +03:00
|
|
|
#include "hw/intc/ioapic.h"
|
2019-12-12 19:15:43 +03:00
|
|
|
#include "hw/intc/i8259.h"
|
2023-01-10 12:53:48 +03:00
|
|
|
#include "hw/intc/kvm_irqcount.h"
|
2013-02-04 18:40:22 +04:00
|
|
|
#include "hw/pci/msi.h"
|
2012-12-17 21:20:00 +04:00
|
|
|
#include "qemu/host-utils.h"
|
2021-03-01 14:14:14 +03:00
|
|
|
#include "sysemu/kvm.h"
|
2010-10-20 20:41:28 +04:00
|
|
|
#include "trace.h"
|
2013-02-05 20:06:20 +04:00
|
|
|
#include "hw/i386/apic-msidef.h"
|
2016-05-05 18:14:37 +03:00
|
|
|
#include "qapi/error.h"
|
2020-09-03 23:43:22 +03:00
|
|
|
#include "qom/object.h"
|
2005-01-04 02:27:31 +03:00
|
|
|
|
2012-02-17 21:31:19 +04:00
|
|
|
#define SYNC_FROM_VAPIC 0x1
|
|
|
|
#define SYNC_TO_VAPIC 0x2
|
|
|
|
#define SYNC_ISR_IRR_TO_VAPIC 0x4
|
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
static APICCommonState **local_apics;
|
|
|
|
static uint32_t max_apics;
|
|
|
|
static uint32_t max_apic_words;
|
2009-01-15 23:11:34 +03:00
|
|
|
|
2016-09-15 10:15:59 +03:00
|
|
|
#define TYPE_APIC "apic"
|
2020-09-01 00:07:36 +03:00
|
|
|
/*This is reusing the APICCommonState typedef from APIC_COMMON */
|
|
|
|
DECLARE_INSTANCE_CHECKER(APICCommonState, APIC,
|
|
|
|
TYPE_APIC)
|
2016-09-15 10:15:59 +03:00
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_set_irq(APICCommonState *s, int vector_num, int trigger_mode);
|
|
|
|
static void apic_update_irq(APICCommonState *s);
|
2009-03-12 23:25:12 +03:00
|
|
|
static void apic_get_delivery_bitmask(uint32_t *deliver_bitmask,
|
2024-01-11 18:43:59 +03:00
|
|
|
uint32_t dest, uint8_t dest_mode);
|
|
|
|
|
|
|
|
void apic_set_max_apic_id(uint32_t max_apic_id)
|
|
|
|
{
|
|
|
|
int word_size = 32;
|
|
|
|
|
|
|
|
/* round up the max apic id to next multiple of words */
|
|
|
|
max_apics = (max_apic_id + word_size - 1) & ~(word_size - 1);
|
|
|
|
|
|
|
|
local_apics = g_malloc0(sizeof(*local_apics) * max_apics);
|
|
|
|
max_apic_words = max_apics >> 5;
|
|
|
|
}
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
|
2008-12-06 13:46:35 +03:00
|
|
|
/* Find first bit starting from msb */
|
2013-05-13 15:11:02 +04:00
|
|
|
static int apic_fls_bit(uint32_t value)
|
2008-12-06 13:46:35 +03:00
|
|
|
{
|
|
|
|
return 31 - clz32(value);
|
|
|
|
}
|
|
|
|
|
2008-10-12 04:53:17 +04:00
|
|
|
/* Find first bit starting from lsb */
|
2013-05-13 15:11:02 +04:00
|
|
|
static int apic_ffs_bit(uint32_t value)
|
2005-12-17 04:27:28 +03:00
|
|
|
{
|
2008-10-13 00:16:03 +04:00
|
|
|
return ctz32(value);
|
2005-12-17 04:27:28 +03:00
|
|
|
}
|
|
|
|
|
2013-05-13 15:11:02 +04:00
|
|
|
static inline void apic_reset_bit(uint32_t *tab, int index)
|
2005-12-17 04:27:28 +03:00
|
|
|
{
|
|
|
|
int i, mask;
|
|
|
|
i = index >> 5;
|
|
|
|
mask = 1 << (index & 0x1f);
|
|
|
|
tab[i] &= ~mask;
|
|
|
|
}
|
|
|
|
|
2012-02-17 21:31:19 +04:00
|
|
|
/* return -1 if no bit is set */
|
|
|
|
static int get_highest_priority_int(uint32_t *tab)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 7; i >= 0; i--) {
|
|
|
|
if (tab[i] != 0) {
|
2013-05-13 15:11:02 +04:00
|
|
|
return i * 32 + apic_fls_bit(tab[i]);
|
2012-02-17 21:31:19 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void apic_sync_vapic(APICCommonState *s, int sync_type)
|
|
|
|
{
|
|
|
|
VAPICState vapic_state;
|
|
|
|
size_t length;
|
|
|
|
off_t start;
|
|
|
|
int vector;
|
|
|
|
|
|
|
|
if (!s->vapic_paddr) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (sync_type & SYNC_FROM_VAPIC) {
|
2014-04-07 22:28:23 +04:00
|
|
|
cpu_physical_memory_read(s->vapic_paddr, &vapic_state,
|
|
|
|
sizeof(vapic_state));
|
2012-02-17 21:31:19 +04:00
|
|
|
s->tpr = vapic_state.tpr;
|
|
|
|
}
|
|
|
|
if (sync_type & (SYNC_TO_VAPIC | SYNC_ISR_IRR_TO_VAPIC)) {
|
|
|
|
start = offsetof(VAPICState, isr);
|
|
|
|
length = offsetof(VAPICState, enabled) - offsetof(VAPICState, isr);
|
|
|
|
|
|
|
|
if (sync_type & SYNC_TO_VAPIC) {
|
2012-05-03 00:23:49 +04:00
|
|
|
assert(qemu_cpu_is_self(CPU(s->cpu)));
|
2012-02-17 21:31:19 +04:00
|
|
|
|
|
|
|
vapic_state.tpr = s->tpr;
|
|
|
|
vapic_state.enabled = 1;
|
|
|
|
start = 0;
|
|
|
|
length = sizeof(VAPICState);
|
|
|
|
}
|
|
|
|
|
|
|
|
vector = get_highest_priority_int(s->isr);
|
|
|
|
if (vector < 0) {
|
|
|
|
vector = 0;
|
|
|
|
}
|
|
|
|
vapic_state.isr = vector & 0xf0;
|
|
|
|
|
|
|
|
vapic_state.zero = 0;
|
|
|
|
|
|
|
|
vector = get_highest_priority_int(s->irr);
|
|
|
|
if (vector < 0) {
|
|
|
|
vector = 0;
|
|
|
|
}
|
|
|
|
vapic_state.irr = vector & 0xff;
|
|
|
|
|
2018-12-14 16:30:48 +03:00
|
|
|
address_space_write_rom(&address_space_memory,
|
|
|
|
s->vapic_paddr + start,
|
|
|
|
MEMTXATTRS_UNSPECIFIED,
|
|
|
|
((void *)&vapic_state) + start, length);
|
2012-02-17 21:31:19 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void apic_vapic_base_update(APICCommonState *s)
|
|
|
|
{
|
|
|
|
apic_sync_vapic(s, SYNC_TO_VAPIC);
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_local_deliver(APICCommonState *s, int vector)
|
2008-04-13 20:08:30 +04:00
|
|
|
{
|
|
|
|
uint32_t lvt = s->lvt[vector];
|
|
|
|
int trigger_mode;
|
|
|
|
|
2010-10-20 20:41:28 +04:00
|
|
|
trace_apic_local_deliver(vector, (lvt >> 8) & 7);
|
|
|
|
|
2008-04-13 20:08:30 +04:00
|
|
|
if (lvt & APIC_LVT_MASKED)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch ((lvt >> 8) & 7) {
|
|
|
|
case APIC_DM_SMI:
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(s->cpu), CPU_INTERRUPT_SMI);
|
2008-04-13 20:08:30 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APIC_DM_NMI:
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(s->cpu), CPU_INTERRUPT_NMI);
|
2008-04-13 20:08:30 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APIC_DM_EXTINT:
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(s->cpu), CPU_INTERRUPT_HARD);
|
2008-04-13 20:08:30 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
case APIC_DM_FIXED:
|
|
|
|
trigger_mode = APIC_TRIGGER_EDGE;
|
|
|
|
if ((vector == APIC_LVT_LINT0 || vector == APIC_LVT_LINT1) &&
|
|
|
|
(lvt & APIC_LVT_LEVEL_TRIGGER))
|
|
|
|
trigger_mode = APIC_TRIGGER_LEVEL;
|
|
|
|
apic_set_irq(s, lvt & 0xff, trigger_mode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
void apic_deliver_pic_intr(DeviceState *dev, int level)
|
2008-08-21 07:14:52 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2010-06-19 11:47:42 +04:00
|
|
|
|
2010-06-19 11:42:08 +04:00
|
|
|
if (level) {
|
|
|
|
apic_local_deliver(s, APIC_LVT_LINT0);
|
|
|
|
} else {
|
2008-08-21 07:14:52 +04:00
|
|
|
uint32_t lvt = s->lvt[APIC_LVT_LINT0];
|
|
|
|
|
|
|
|
switch ((lvt >> 8) & 7) {
|
|
|
|
case APIC_DM_FIXED:
|
|
|
|
if (!(lvt & APIC_LVT_LEVEL_TRIGGER))
|
|
|
|
break;
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_reset_bit(s->irr, lvt & 0xff);
|
2008-08-21 07:14:52 +04:00
|
|
|
/* fall through */
|
|
|
|
case APIC_DM_EXTINT:
|
2014-11-11 15:14:14 +03:00
|
|
|
apic_update_irq(s);
|
2008-08-21 07:14:52 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_external_nmi(APICCommonState *s)
|
2011-10-17 20:00:06 +04:00
|
|
|
{
|
|
|
|
apic_local_deliver(s, APIC_LVT_LINT1);
|
|
|
|
}
|
|
|
|
|
2005-12-17 04:27:28 +03:00
|
|
|
#define foreach_apic(apic, deliver_bitmask, code) \
|
|
|
|
{\
|
2014-03-17 20:00:31 +04:00
|
|
|
int __i, __j;\
|
2024-01-11 18:43:59 +03:00
|
|
|
for (__i = 0; __i < max_apic_words; __i++) {\
|
2014-03-17 20:00:31 +04:00
|
|
|
uint32_t __mask = deliver_bitmask[__i];\
|
2005-12-17 04:27:28 +03:00
|
|
|
if (__mask) {\
|
2024-01-11 18:43:59 +03:00
|
|
|
for (__j = 0; __j < 32; __j++) {\
|
2014-03-17 20:00:31 +04:00
|
|
|
if (__mask & (1U << __j)) {\
|
2005-12-17 04:27:28 +03:00
|
|
|
apic = local_apics[__i * 32 + __j];\
|
|
|
|
if (apic) {\
|
|
|
|
code;\
|
|
|
|
}\
|
|
|
|
}\
|
|
|
|
}\
|
|
|
|
}\
|
|
|
|
}\
|
|
|
|
}
|
|
|
|
|
2007-09-17 01:08:06 +04:00
|
|
|
static void apic_bus_deliver(const uint32_t *deliver_bitmask,
|
2011-08-22 19:46:31 +04:00
|
|
|
uint8_t delivery_mode, uint8_t vector_num,
|
2005-07-23 23:05:37 +04:00
|
|
|
uint8_t trigger_mode)
|
|
|
|
{
|
2011-10-16 13:16:36 +04:00
|
|
|
APICCommonState *apic_iter;
|
2005-07-23 23:05:37 +04:00
|
|
|
|
|
|
|
switch (delivery_mode) {
|
|
|
|
case APIC_DM_LOWPRI:
|
2005-11-23 23:59:44 +03:00
|
|
|
/* XXX: search for focus processor, arbitration */
|
2005-12-17 04:27:28 +03:00
|
|
|
{
|
|
|
|
int i, d;
|
|
|
|
d = -1;
|
2024-01-11 18:43:59 +03:00
|
|
|
for (i = 0; i < max_apic_words; i++) {
|
2005-12-17 04:27:28 +03:00
|
|
|
if (deliver_bitmask[i]) {
|
2013-05-13 15:11:02 +04:00
|
|
|
d = i * 32 + apic_ffs_bit(deliver_bitmask[i]);
|
2005-12-17 04:27:28 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (d >= 0) {
|
|
|
|
apic_iter = local_apics[d];
|
|
|
|
if (apic_iter) {
|
|
|
|
apic_set_irq(apic_iter, vector_num, trigger_mode);
|
|
|
|
}
|
|
|
|
}
|
2005-11-23 23:59:44 +03:00
|
|
|
}
|
2005-12-17 04:27:28 +03:00
|
|
|
return;
|
2005-11-23 23:59:44 +03:00
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
case APIC_DM_FIXED:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case APIC_DM_SMI:
|
2008-04-13 20:08:23 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(apic_iter->cpu), CPU_INTERRUPT_SMI)
|
2012-10-10 16:10:07 +04:00
|
|
|
);
|
2008-04-13 20:08:23 +04:00
|
|
|
return;
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
case APIC_DM_NMI:
|
2008-04-13 20:08:23 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(apic_iter->cpu), CPU_INTERRUPT_NMI)
|
2012-10-10 16:10:07 +04:00
|
|
|
);
|
2008-04-13 20:08:23 +04:00
|
|
|
return;
|
2005-07-23 23:05:37 +04:00
|
|
|
|
|
|
|
case APIC_DM_INIT:
|
|
|
|
/* normal INIT IPI sent to processors */
|
2007-09-17 01:08:06 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(apic_iter->cpu),
|
2012-10-10 16:10:07 +04:00
|
|
|
CPU_INTERRUPT_INIT)
|
|
|
|
);
|
2005-07-23 23:05:37 +04:00
|
|
|
return;
|
2007-09-17 12:09:54 +04:00
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
case APIC_DM_EXTINT:
|
2005-07-24 01:43:15 +04:00
|
|
|
/* handled in I/O APIC code */
|
2005-07-23 23:05:37 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2007-09-17 01:08:06 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2005-12-17 04:27:28 +03:00
|
|
|
apic_set_irq(apic_iter, vector_num, trigger_mode) );
|
2005-07-23 23:05:37 +04:00
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
static void apic_deliver_irq(uint32_t dest, uint8_t dest_mode,
|
|
|
|
uint8_t delivery_mode, uint8_t vector_num,
|
|
|
|
uint8_t trigger_mode)
|
2009-03-12 23:25:12 +03:00
|
|
|
{
|
2024-03-05 01:39:02 +03:00
|
|
|
g_autofree uint32_t *deliver_bitmask = g_new(uint32_t, max_apic_words);
|
2009-03-12 23:25:12 +03:00
|
|
|
|
2010-10-20 20:41:28 +04:00
|
|
|
trace_apic_deliver_irq(dest, dest_mode, delivery_mode, vector_num,
|
2011-08-22 19:46:31 +04:00
|
|
|
trigger_mode);
|
2010-10-20 20:41:28 +04:00
|
|
|
|
2009-03-12 23:25:12 +03:00
|
|
|
apic_get_delivery_bitmask(deliver_bitmask, dest, dest_mode);
|
2011-08-22 19:46:31 +04:00
|
|
|
apic_bus_deliver(deliver_bitmask, delivery_mode, vector_num, trigger_mode);
|
2009-03-12 23:25:12 +03:00
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:58 +03:00
|
|
|
bool is_x2apic_mode(DeviceState *dev)
|
|
|
|
{
|
|
|
|
APICCommonState *s = APIC(dev);
|
|
|
|
|
|
|
|
return s->apicbase & MSR_IA32_APICBASE_EXTD;
|
|
|
|
}
|
|
|
|
|
2024-01-11 18:44:00 +03:00
|
|
|
static int apic_set_base_check(APICCommonState *s, uint64_t val)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2024-01-11 18:44:00 +03:00
|
|
|
/* Enable x2apic when x2apic is not supported by CPU */
|
|
|
|
if (!cpu_has_x2apic_feature(&s->cpu->env) &&
|
|
|
|
val & MSR_IA32_APICBASE_EXTD) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Transition into invalid state
|
|
|
|
* (s->apicbase & MSR_IA32_APICBASE_ENABLE == 0) &&
|
|
|
|
* (s->apicbase & MSR_IA32_APICBASE_EXTD) == 1
|
|
|
|
*/
|
|
|
|
if (!(val & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
(val & MSR_IA32_APICBASE_EXTD)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invalid transition from disabled mode to x2APIC */
|
|
|
|
if (!(s->apicbase & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
!(s->apicbase & MSR_IA32_APICBASE_EXTD) &&
|
|
|
|
(val & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
(val & MSR_IA32_APICBASE_EXTD)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Invalid transition from x2APIC to xAPIC */
|
|
|
|
if ((s->apicbase & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
(s->apicbase & MSR_IA32_APICBASE_EXTD) &&
|
|
|
|
(val & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
!(val & MSR_IA32_APICBASE_EXTD)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int apic_set_base(APICCommonState *s, uint64_t val)
|
|
|
|
{
|
|
|
|
if (apic_set_base_check(s, val) < 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-09-17 01:08:06 +04:00
|
|
|
s->apicbase = (val & 0xfffff000) |
|
2005-01-04 02:27:31 +03:00
|
|
|
(s->apicbase & (MSR_IA32_APICBASE_BSP | MSR_IA32_APICBASE_ENABLE));
|
|
|
|
/* if disabled, cannot be enabled again */
|
|
|
|
if (!(val & MSR_IA32_APICBASE_ENABLE)) {
|
|
|
|
s->apicbase &= ~MSR_IA32_APICBASE_ENABLE;
|
2012-10-10 16:10:07 +04:00
|
|
|
cpu_clear_apic_feature(&s->cpu->env);
|
2005-01-04 02:27:31 +03:00
|
|
|
s->spurious_vec &= ~APIC_SV_ENABLE;
|
|
|
|
}
|
2024-01-11 18:44:00 +03:00
|
|
|
|
|
|
|
/* Transition from disabled mode to xAPIC */
|
|
|
|
if (!(s->apicbase & MSR_IA32_APICBASE_ENABLE) &&
|
|
|
|
(val & MSR_IA32_APICBASE_ENABLE)) {
|
|
|
|
s->apicbase |= MSR_IA32_APICBASE_ENABLE;
|
|
|
|
cpu_set_apic_feature(&s->cpu->env);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Transition from xAPIC to x2APIC */
|
|
|
|
if (cpu_has_x2apic_feature(&s->cpu->env) &&
|
|
|
|
!(s->apicbase & MSR_IA32_APICBASE_EXTD) &&
|
|
|
|
(val & MSR_IA32_APICBASE_EXTD)) {
|
|
|
|
s->apicbase |= MSR_IA32_APICBASE_EXTD;
|
|
|
|
|
|
|
|
s->log_dest = ((s->initial_apic_id & 0xffff0) << 16) |
|
|
|
|
(1 << (s->initial_apic_id & 0xf));
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_set_tpr(APICCommonState *s, uint8_t val)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2012-02-17 21:31:19 +04:00
|
|
|
/* Updates from cr8 are ignored while the VAPIC is active */
|
|
|
|
if (!s->vapic_paddr) {
|
|
|
|
s->tpr = val << 4;
|
|
|
|
apic_update_irq(s);
|
|
|
|
}
|
2005-01-23 23:46:56 +03:00
|
|
|
}
|
|
|
|
|
2017-09-13 12:05:15 +03:00
|
|
|
int apic_get_highest_priority_irr(DeviceState *dev)
|
|
|
|
{
|
|
|
|
APICCommonState *s;
|
|
|
|
|
|
|
|
if (!dev) {
|
|
|
|
/* no interrupts */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
s = APIC_COMMON(dev);
|
|
|
|
return get_highest_priority_int(s->irr);
|
|
|
|
}
|
|
|
|
|
2012-02-17 21:31:19 +04:00
|
|
|
static uint8_t apic_get_tpr(APICCommonState *s)
|
2005-07-23 23:05:37 +04:00
|
|
|
{
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
|
|
|
return s->tpr >> 4;
|
2005-07-23 23:05:37 +04:00
|
|
|
}
|
|
|
|
|
2015-09-22 16:18:13 +03:00
|
|
|
int apic_get_ppr(APICCommonState *s)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
|
|
|
int tpr, isrv, ppr;
|
|
|
|
|
|
|
|
tpr = (s->tpr >> 4);
|
|
|
|
isrv = get_highest_priority_int(s->isr);
|
|
|
|
if (isrv < 0)
|
|
|
|
isrv = 0;
|
|
|
|
isrv >>= 4;
|
|
|
|
if (tpr >= isrv)
|
|
|
|
ppr = s->tpr;
|
|
|
|
else
|
|
|
|
ppr = isrv << 4;
|
|
|
|
return ppr;
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static int apic_get_arb_pri(APICCommonState *s)
|
2005-07-23 23:05:37 +04:00
|
|
|
{
|
|
|
|
/* XXX: arbitration */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-02-07 17:14:44 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* <0 - low prio interrupt,
|
|
|
|
* 0 - no interrupt,
|
|
|
|
* >0 - interrupt number
|
|
|
|
*/
|
2011-10-16 13:16:36 +04:00
|
|
|
static int apic_irq_pending(APICCommonState *s)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2005-07-23 23:05:37 +04:00
|
|
|
int irrv, ppr;
|
2014-11-11 15:14:05 +03:00
|
|
|
|
|
|
|
if (!(s->spurious_vec & APIC_SV_ENABLE)) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2005-01-04 02:27:31 +03:00
|
|
|
irrv = get_highest_priority_int(s->irr);
|
2011-02-07 17:14:44 +03:00
|
|
|
if (irrv < 0) {
|
|
|
|
return 0;
|
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
ppr = apic_get_ppr(s);
|
2011-02-07 17:14:44 +03:00
|
|
|
if (ppr && (irrv & 0xf0) <= (ppr & 0xf0)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return irrv;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* signal the CPU if an irq is pending */
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_update_irq(APICCommonState *s)
|
2011-02-07 17:14:44 +03:00
|
|
|
{
|
2013-01-18 18:03:43 +04:00
|
|
|
CPUState *cpu;
|
2015-05-20 05:40:47 +03:00
|
|
|
DeviceState *dev = (DeviceState *)s;
|
2012-05-03 00:23:49 +04:00
|
|
|
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu = CPU(s->cpu);
|
2012-05-03 00:23:49 +04:00
|
|
|
if (!qemu_cpu_is_self(cpu)) {
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(cpu, CPU_INTERRUPT_POLL);
|
2012-07-09 18:42:32 +04:00
|
|
|
} else if (apic_irq_pending(s) > 0) {
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(cpu, CPU_INTERRUPT_HARD);
|
2015-05-20 05:40:47 +03:00
|
|
|
} else if (!apic_accept_pic_intr(dev) || !pic_get_output(isa_pic)) {
|
2014-11-11 15:14:14 +03:00
|
|
|
cpu_reset_interrupt(cpu, CPU_INTERRUPT_HARD);
|
2011-02-07 17:14:44 +03:00
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
void apic_poll_irq(DeviceState *dev)
|
2012-02-17 21:31:19 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2012-02-17 21:31:19 +04:00
|
|
|
|
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
|
|
|
apic_update_irq(s);
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_set_irq(APICCommonState *s, int vector_num, int trigger_mode)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2023-01-10 12:53:48 +03:00
|
|
|
kvm_report_irq_delivered(!apic_get_bit(s->irr, vector_num));
|
2009-01-15 23:11:34 +03:00
|
|
|
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_set_bit(s->irr, vector_num);
|
2005-01-04 02:27:31 +03:00
|
|
|
if (trigger_mode)
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_set_bit(s->tmr, vector_num);
|
2005-01-04 02:27:31 +03:00
|
|
|
else
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_reset_bit(s->tmr, vector_num);
|
2012-02-17 21:31:19 +04:00
|
|
|
if (s->vapic_paddr) {
|
|
|
|
apic_sync_vapic(s, SYNC_ISR_IRR_TO_VAPIC);
|
|
|
|
/*
|
|
|
|
* The vcpu thread needs to see the new IRR before we pull its current
|
|
|
|
* TPR value. That way, if we miss a lowering of the TRP, the guest
|
|
|
|
* has the chance to notice the new IRR and poll for IRQs on its own.
|
|
|
|
*/
|
|
|
|
smp_wmb();
|
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
apic_update_irq(s);
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_eoi(APICCommonState *s)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
|
|
|
int isrv;
|
|
|
|
isrv = get_highest_priority_int(s->isr);
|
|
|
|
if (isrv < 0)
|
|
|
|
return;
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_reset_bit(s->isr, isrv);
|
|
|
|
if (!(s->spurious_vec & APIC_SV_DIRECTED_IO) && apic_get_bit(s->tmr, isrv)) {
|
2011-02-04 00:54:11 +03:00
|
|
|
ioapic_eoi_broadcast(isrv);
|
|
|
|
}
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC | SYNC_TO_VAPIC);
|
2005-01-04 02:27:31 +03:00
|
|
|
apic_update_irq(s);
|
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
static bool apic_match_dest(APICCommonState *apic, uint32_t dest)
|
2009-06-10 16:40:48 +04:00
|
|
|
{
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(&apic->parent_obj)) {
|
|
|
|
return apic->initial_apic_id == dest;
|
|
|
|
} else {
|
|
|
|
return apic->id == (uint8_t)dest;
|
|
|
|
}
|
|
|
|
}
|
2009-06-10 16:40:48 +04:00
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
static void apic_find_dest(uint32_t *deliver_bitmask, uint32_t dest)
|
|
|
|
{
|
|
|
|
APICCommonState *apic = NULL;
|
|
|
|
int i;
|
2009-06-10 16:40:48 +04:00
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
for (i = 0; i < max_apics; i++) {
|
2009-06-10 16:40:48 +04:00
|
|
|
apic = local_apics[i];
|
2024-01-11 18:43:59 +03:00
|
|
|
if (apic && apic_match_dest(apic, dest)) {
|
|
|
|
apic_set_bit(deliver_bitmask, i);
|
|
|
|
}
|
2009-06-10 16:40:48 +04:00
|
|
|
}
|
2024-01-11 18:43:59 +03:00
|
|
|
}
|
2009-06-10 16:40:48 +04:00
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
/*
|
|
|
|
* Deliver interrupt to x2APIC CPUs if it is x2APIC broadcast.
|
|
|
|
* Otherwise, deliver interrupt to xAPIC CPUs if it is xAPIC
|
|
|
|
* broadcast.
|
|
|
|
*/
|
|
|
|
static void apic_get_broadcast_bitmask(uint32_t *deliver_bitmask,
|
|
|
|
bool is_x2apic_broadcast)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
APICCommonState *apic_iter;
|
|
|
|
|
|
|
|
for (i = 0; i < max_apics; i++) {
|
|
|
|
apic_iter = local_apics[i];
|
|
|
|
if (apic_iter) {
|
|
|
|
bool apic_in_x2apic = is_x2apic_mode(&apic_iter->parent_obj);
|
|
|
|
|
|
|
|
if (is_x2apic_broadcast && apic_in_x2apic) {
|
|
|
|
apic_set_bit(deliver_bitmask, i);
|
|
|
|
} else if (!is_x2apic_broadcast && !apic_in_x2apic) {
|
|
|
|
apic_set_bit(deliver_bitmask, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2009-06-10 16:40:48 +04:00
|
|
|
}
|
|
|
|
|
2005-12-17 04:27:28 +03:00
|
|
|
static void apic_get_delivery_bitmask(uint32_t *deliver_bitmask,
|
2024-01-11 18:43:59 +03:00
|
|
|
uint32_t dest, uint8_t dest_mode)
|
2005-07-23 23:05:37 +04:00
|
|
|
{
|
2024-01-11 18:43:59 +03:00
|
|
|
APICCommonState *apic;
|
2005-12-17 04:27:28 +03:00
|
|
|
int i;
|
2005-07-23 23:05:37 +04:00
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
memset(deliver_bitmask, 0x00, max_apic_words * sizeof(uint32_t));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* x2APIC broadcast is delivered to all x2APIC CPUs regardless of
|
|
|
|
* destination mode. In case the destination mode is physical, it is
|
|
|
|
* broadcasted to all xAPIC CPUs too. Otherwise, if the destination
|
|
|
|
* mode is logical, we need to continue checking if xAPIC CPUs accepts
|
|
|
|
* the interrupt.
|
|
|
|
*/
|
|
|
|
if (dest == 0xffffffff) {
|
|
|
|
if (dest_mode == APIC_DESTMODE_PHYSICAL) {
|
|
|
|
memset(deliver_bitmask, 0xff, max_apic_words * sizeof(uint32_t));
|
|
|
|
return;
|
2005-12-17 04:27:28 +03:00
|
|
|
} else {
|
2024-01-11 18:43:59 +03:00
|
|
|
apic_get_broadcast_bitmask(deliver_bitmask, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dest_mode == APIC_DESTMODE_PHYSICAL) {
|
|
|
|
apic_find_dest(deliver_bitmask, dest);
|
|
|
|
/* Any APIC in xAPIC mode will interpret 0xFF as broadcast */
|
|
|
|
if (dest == 0xff) {
|
|
|
|
apic_get_broadcast_bitmask(deliver_bitmask, false);
|
2005-12-17 04:27:28 +03:00
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
} else {
|
2024-01-11 18:43:59 +03:00
|
|
|
/* XXX: logical mode */
|
|
|
|
for (i = 0; i < max_apics; i++) {
|
|
|
|
apic = local_apics[i];
|
|
|
|
if (apic) {
|
|
|
|
/* x2APIC logical mode */
|
|
|
|
if (apic->apicbase & MSR_IA32_APICBASE_EXTD) {
|
|
|
|
if ((dest >> 16) == (apic->extended_log_dest >> 16) &&
|
|
|
|
(dest & apic->extended_log_dest & 0xffff)) {
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_set_bit(deliver_bitmask, i);
|
2005-12-17 04:27:28 +03:00
|
|
|
}
|
2024-01-11 18:43:59 +03:00
|
|
|
continue;
|
2005-12-17 04:27:28 +03:00
|
|
|
}
|
2024-01-11 18:43:59 +03:00
|
|
|
|
|
|
|
/* xAPIC logical mode */
|
|
|
|
dest = (uint8_t)dest;
|
|
|
|
if (apic->dest_mode == APIC_DESTMODE_LOGICAL_FLAT) {
|
|
|
|
if (dest & apic->log_dest) {
|
|
|
|
apic_set_bit(deliver_bitmask, i);
|
|
|
|
}
|
|
|
|
} else if (apic->dest_mode == APIC_DESTMODE_LOGICAL_CLUSTER) {
|
|
|
|
/*
|
|
|
|
* In cluster model of xAPIC logical mode IPI, 4 higher
|
|
|
|
* bits are used as cluster address, 4 lower bits are
|
|
|
|
* the bitmask for local APICs in the cluster. The IPI
|
|
|
|
* is delivered to an APIC if the cluster address
|
|
|
|
* matches and the APIC's address bit in the cluster is
|
|
|
|
* set in bitmask of destination ID in IPI.
|
|
|
|
*
|
|
|
|
* The cluster address ranges from 0 - 14, the cluster
|
|
|
|
* address 15 (0xf) is the broadcast address to all
|
|
|
|
* clusters.
|
|
|
|
*/
|
|
|
|
if ((dest & 0xf0) == 0xf0 ||
|
|
|
|
(dest & 0xf0) == (apic->log_dest & 0xf0)) {
|
|
|
|
if (dest & apic->log_dest & 0x0f) {
|
|
|
|
apic_set_bit(deliver_bitmask, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2005-12-17 04:27:28 +03:00
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_startup(APICCommonState *s, int vector_num)
|
2005-11-22 02:26:26 +03:00
|
|
|
{
|
2009-06-18 00:26:59 +04:00
|
|
|
s->sipi_vector = vector_num;
|
2013-01-18 18:03:43 +04:00
|
|
|
cpu_interrupt(CPU(s->cpu), CPU_INTERRUPT_SIPI);
|
2009-06-18 00:26:59 +04:00
|
|
|
}
|
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
void apic_sipi(DeviceState *dev)
|
2009-06-18 00:26:59 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2010-06-19 11:47:42 +04:00
|
|
|
|
2013-01-18 01:30:20 +04:00
|
|
|
cpu_reset_interrupt(CPU(s->cpu), CPU_INTERRUPT_SIPI);
|
2009-06-18 00:26:59 +04:00
|
|
|
|
|
|
|
if (!s->wait_for_sipi)
|
2005-11-22 02:26:26 +03:00
|
|
|
return;
|
2012-05-03 17:37:01 +04:00
|
|
|
cpu_x86_load_seg_cache_sipi(s->cpu, s->sipi_vector);
|
2009-06-18 00:26:59 +04:00
|
|
|
s->wait_for_sipi = 0;
|
2005-11-22 02:26:26 +03:00
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:59 +03:00
|
|
|
static void apic_deliver(DeviceState *dev, uint32_t dest, uint8_t dest_mode,
|
2005-07-23 23:05:37 +04:00
|
|
|
uint8_t delivery_mode, uint8_t vector_num,
|
2024-01-11 18:43:59 +03:00
|
|
|
uint8_t trigger_mode, uint8_t dest_shorthand)
|
2005-07-23 23:05:37 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2011-10-16 13:16:36 +04:00
|
|
|
APICCommonState *apic_iter;
|
2024-01-11 18:43:59 +03:00
|
|
|
uint32_t deliver_bitmask_size = max_apic_words * sizeof(uint32_t);
|
2024-03-05 01:39:02 +03:00
|
|
|
g_autofree uint32_t *deliver_bitmask = g_new(uint32_t, max_apic_words);
|
2024-01-11 18:43:59 +03:00
|
|
|
uint32_t current_apic_id;
|
|
|
|
|
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
current_apic_id = s->initial_apic_id;
|
|
|
|
} else {
|
|
|
|
current_apic_id = s->id;
|
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
|
2005-11-22 02:26:26 +03:00
|
|
|
switch (dest_shorthand) {
|
2005-12-17 04:27:28 +03:00
|
|
|
case 0:
|
|
|
|
apic_get_delivery_bitmask(deliver_bitmask, dest, dest_mode);
|
|
|
|
break;
|
|
|
|
case 1:
|
2024-01-11 18:43:59 +03:00
|
|
|
memset(deliver_bitmask, 0x00, deliver_bitmask_size);
|
|
|
|
apic_set_bit(deliver_bitmask, current_apic_id);
|
2005-12-17 04:27:28 +03:00
|
|
|
break;
|
|
|
|
case 2:
|
2024-01-11 18:43:59 +03:00
|
|
|
memset(deliver_bitmask, 0xff, deliver_bitmask_size);
|
2005-12-17 04:27:28 +03:00
|
|
|
break;
|
|
|
|
case 3:
|
2024-01-11 18:43:59 +03:00
|
|
|
memset(deliver_bitmask, 0xff, deliver_bitmask_size);
|
|
|
|
apic_reset_bit(deliver_bitmask, current_apic_id);
|
2005-12-17 04:27:28 +03:00
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
}
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
switch (delivery_mode) {
|
|
|
|
case APIC_DM_INIT:
|
|
|
|
{
|
|
|
|
int trig_mode = (s->icr[0] >> 15) & 1;
|
|
|
|
int level = (s->icr[0] >> 14) & 1;
|
|
|
|
if (level == 0 && trig_mode == 1) {
|
2007-09-17 01:08:06 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2005-12-17 04:27:28 +03:00
|
|
|
apic_iter->arb_id = apic_iter->id );
|
2005-07-23 23:05:37 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case APIC_DM_SIPI:
|
2007-09-17 01:08:06 +04:00
|
|
|
foreach_apic(apic_iter, deliver_bitmask,
|
2005-12-17 04:27:28 +03:00
|
|
|
apic_startup(apic_iter, vector_num) );
|
2005-07-23 23:05:37 +04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2011-08-22 19:46:31 +04:00
|
|
|
apic_bus_deliver(deliver_bitmask, delivery_mode, vector_num, trigger_mode);
|
2005-07-23 23:05:37 +04:00
|
|
|
}
|
|
|
|
|
2012-07-09 18:42:31 +04:00
|
|
|
static bool apic_check_pic(APICCommonState *s)
|
|
|
|
{
|
2015-05-20 05:40:47 +03:00
|
|
|
DeviceState *dev = (DeviceState *)s;
|
|
|
|
|
|
|
|
if (!apic_accept_pic_intr(dev) || !pic_get_output(isa_pic)) {
|
2012-07-09 18:42:31 +04:00
|
|
|
return false;
|
|
|
|
}
|
2015-05-20 05:40:47 +03:00
|
|
|
apic_deliver_pic_intr(dev, 1);
|
2012-07-09 18:42:31 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
int apic_get_interrupt(DeviceState *dev)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2005-01-04 02:27:31 +03:00
|
|
|
int intno;
|
|
|
|
|
|
|
|
/* if the APIC is installed or enabled, we let the 8259 handle the
|
|
|
|
IRQs */
|
|
|
|
if (!s)
|
|
|
|
return -1;
|
|
|
|
if (!(s->spurious_vec & APIC_SV_ENABLE))
|
|
|
|
return -1;
|
2007-09-17 12:09:54 +04:00
|
|
|
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
2011-02-07 17:14:44 +03:00
|
|
|
intno = apic_irq_pending(s);
|
|
|
|
|
2014-11-11 15:14:18 +03:00
|
|
|
/* if there is an interrupt from the 8259, let the caller handle
|
|
|
|
* that first since ExtINT interrupts ignore the priority.
|
|
|
|
*/
|
|
|
|
if (intno == 0 || apic_check_pic(s)) {
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_TO_VAPIC);
|
2005-01-04 02:27:31 +03:00
|
|
|
return -1;
|
2011-02-07 17:14:44 +03:00
|
|
|
} else if (intno < 0) {
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_TO_VAPIC);
|
2005-07-23 23:05:37 +04:00
|
|
|
return s->spurious_vec & 0xff;
|
2011-02-07 17:14:44 +03:00
|
|
|
}
|
2013-05-13 15:11:02 +04:00
|
|
|
apic_reset_bit(s->irr, intno);
|
|
|
|
apic_set_bit(s->isr, intno);
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_TO_VAPIC);
|
2012-07-09 18:42:30 +04:00
|
|
|
|
2005-01-04 02:27:31 +03:00
|
|
|
apic_update_irq(s);
|
2012-07-09 18:42:30 +04:00
|
|
|
|
2005-01-04 02:27:31 +03:00
|
|
|
return intno;
|
|
|
|
}
|
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
int apic_accept_pic_intr(DeviceState *dev)
|
2007-10-09 07:08:56 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2007-10-09 07:08:56 +04:00
|
|
|
uint32_t lvt0;
|
|
|
|
|
|
|
|
if (!s)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
lvt0 = s->lvt[APIC_LVT_LINT0];
|
|
|
|
|
2008-04-13 20:08:30 +04:00
|
|
|
if ((s->apicbase & MSR_IA32_APICBASE_ENABLE) == 0 ||
|
|
|
|
(lvt0 & APIC_LVT_MASKED) == 0)
|
2019-09-26 15:48:00 +03:00
|
|
|
return isa_pic != NULL;
|
2007-10-09 07:08:56 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-10-16 13:16:36 +04:00
|
|
|
static void apic_timer_update(APICCommonState *s, int64_t current_time)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2011-10-16 14:19:12 +04:00
|
|
|
if (apic_next_timer(s, current_time)) {
|
2013-08-21 19:03:08 +04:00
|
|
|
timer_mod(s->timer, s->next_time);
|
2005-01-04 02:27:31 +03:00
|
|
|
} else {
|
2013-08-21 19:03:08 +04:00
|
|
|
timer_del(s->timer);
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void apic_timer(void *opaque)
|
|
|
|
{
|
2011-10-16 13:16:36 +04:00
|
|
|
APICCommonState *s = opaque;
|
2005-01-04 02:27:31 +03:00
|
|
|
|
2010-06-19 11:42:08 +04:00
|
|
|
apic_local_deliver(s, APIC_LVT_TIMER);
|
2005-01-04 02:27:31 +03:00
|
|
|
apic_timer_update(s, s->next_time);
|
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:58 +03:00
|
|
|
static int apic_register_read(int index, uint64_t *value)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2013-11-05 14:16:02 +04:00
|
|
|
DeviceState *dev;
|
2011-10-16 13:16:36 +04:00
|
|
|
APICCommonState *s;
|
2005-01-04 02:27:31 +03:00
|
|
|
uint32_t val;
|
2024-01-11 18:43:58 +03:00
|
|
|
int ret = 0;
|
2018-08-03 13:19:43 +03:00
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
dev = cpu_get_current_apic();
|
|
|
|
if (!dev) {
|
2024-01-11 18:43:58 +03:00
|
|
|
return -1;
|
2010-06-19 11:42:34 +04:00
|
|
|
}
|
2016-09-15 10:15:59 +03:00
|
|
|
s = APIC(dev);
|
2005-01-04 02:27:31 +03:00
|
|
|
|
|
|
|
switch(index) {
|
|
|
|
case 0x02: /* id */
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
val = s->initial_apic_id;
|
|
|
|
} else {
|
|
|
|
val = s->id << 24;
|
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
|
|
|
case 0x03: /* version */
|
2014-05-05 18:52:51 +04:00
|
|
|
val = s->version | ((APIC_LVT_NB - 1) << 16);
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
|
|
|
case 0x08:
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
|
|
|
if (apic_report_tpr_access) {
|
2012-10-10 16:10:07 +04:00
|
|
|
cpu_report_tpr_access(&s->cpu->env, TPR_ACCESS_READ);
|
2012-02-17 21:31:19 +04:00
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
val = s->tpr;
|
|
|
|
break;
|
2005-07-23 23:05:37 +04:00
|
|
|
case 0x09:
|
|
|
|
val = apic_get_arb_pri(s);
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x0a:
|
|
|
|
/* ppr */
|
|
|
|
val = apic_get_ppr(s);
|
|
|
|
break;
|
2008-03-29 01:31:36 +03:00
|
|
|
case 0x0b:
|
|
|
|
val = 0;
|
|
|
|
break;
|
2005-07-23 23:05:37 +04:00
|
|
|
case 0x0d:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
val = s->extended_log_dest;
|
|
|
|
} else {
|
|
|
|
val = s->log_dest << 24;
|
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
break;
|
|
|
|
case 0x0e:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
val = 0;
|
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
val = (s->dest_mode << 28) | 0xfffffff;
|
|
|
|
}
|
2005-07-23 23:05:37 +04:00
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x0f:
|
|
|
|
val = s->spurious_vec;
|
|
|
|
break;
|
|
|
|
case 0x10 ... 0x17:
|
|
|
|
val = s->isr[index & 7];
|
|
|
|
break;
|
|
|
|
case 0x18 ... 0x1f:
|
|
|
|
val = s->tmr[index & 7];
|
|
|
|
break;
|
|
|
|
case 0x20 ... 0x27:
|
|
|
|
val = s->irr[index & 7];
|
|
|
|
break;
|
|
|
|
case 0x28:
|
|
|
|
val = s->esr;
|
|
|
|
break;
|
|
|
|
case 0x30:
|
|
|
|
case 0x31:
|
|
|
|
val = s->icr[index & 1];
|
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
case 0x32 ... 0x37:
|
|
|
|
val = s->lvt[index - 0x32];
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x38:
|
|
|
|
val = s->initial_count;
|
|
|
|
break;
|
|
|
|
case 0x39:
|
|
|
|
val = apic_get_current_count(s);
|
|
|
|
break;
|
|
|
|
case 0x3e:
|
|
|
|
val = s->divide_conf;
|
|
|
|
break;
|
|
|
|
default:
|
2015-09-22 16:18:14 +03:00
|
|
|
s->esr |= APIC_ESR_ILLEGAL_ADDRESS;
|
2005-01-04 02:27:31 +03:00
|
|
|
val = 0;
|
2024-01-11 18:43:58 +03:00
|
|
|
ret = -1;
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
|
|
|
}
|
2024-01-11 18:43:58 +03:00
|
|
|
|
|
|
|
trace_apic_register_read(index, val);
|
|
|
|
*value = val;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint64_t apic_mem_read(void *opaque, hwaddr addr, unsigned size)
|
|
|
|
{
|
|
|
|
uint64_t val;
|
|
|
|
int index;
|
|
|
|
|
|
|
|
if (size < 4) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
index = (addr >> 4) & 0xff;
|
|
|
|
apic_register_read(index, &val);
|
|
|
|
|
2005-01-04 02:27:31 +03:00
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:58 +03:00
|
|
|
int apic_msr_read(int index, uint64_t *val)
|
|
|
|
{
|
|
|
|
DeviceState *dev;
|
|
|
|
|
|
|
|
dev = cpu_get_current_apic();
|
|
|
|
if (!dev) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return apic_register_read(index, val);
|
|
|
|
}
|
|
|
|
|
2016-10-10 18:28:43 +03:00
|
|
|
static void apic_send_msi(MSIMessage *msi)
|
2009-06-21 20:50:03 +04:00
|
|
|
{
|
2016-10-10 18:28:43 +03:00
|
|
|
uint64_t addr = msi->address;
|
|
|
|
uint32_t data = msi->data;
|
2024-01-11 18:43:59 +03:00
|
|
|
uint32_t dest = (addr & MSI_ADDR_DEST_ID_MASK) >> MSI_ADDR_DEST_ID_SHIFT;
|
|
|
|
/*
|
|
|
|
* The higher 3 bytes of destination id is stored in higher word of
|
|
|
|
* msi address. See x86_iommu_irq_to_msi_message()
|
|
|
|
*/
|
|
|
|
dest = dest | (addr >> 32);
|
2009-06-21 20:50:03 +04:00
|
|
|
uint8_t vector = (data & MSI_DATA_VECTOR_MASK) >> MSI_DATA_VECTOR_SHIFT;
|
|
|
|
uint8_t dest_mode = (addr >> MSI_ADDR_DEST_MODE_SHIFT) & 0x1;
|
|
|
|
uint8_t trigger_mode = (data >> MSI_DATA_TRIGGER_SHIFT) & 0x1;
|
|
|
|
uint8_t delivery = (data >> MSI_DATA_DELIVERY_MODE_SHIFT) & 0x7;
|
|
|
|
/* XXX: Ignore redirection hint. */
|
2011-08-22 19:46:31 +04:00
|
|
|
apic_deliver_irq(dest, dest_mode, delivery, vector, trigger_mode);
|
2009-06-21 20:50:03 +04:00
|
|
|
}
|
|
|
|
|
2024-01-11 18:43:58 +03:00
|
|
|
static int apic_register_write(int index, uint64_t val)
|
2005-01-04 02:27:31 +03:00
|
|
|
{
|
2013-11-05 14:16:02 +04:00
|
|
|
DeviceState *dev;
|
2011-10-16 13:16:36 +04:00
|
|
|
APICCommonState *s;
|
2005-01-04 02:27:31 +03:00
|
|
|
|
2013-11-05 14:16:02 +04:00
|
|
|
dev = cpu_get_current_apic();
|
|
|
|
if (!dev) {
|
2024-01-11 18:43:58 +03:00
|
|
|
return -1;
|
2010-06-19 11:42:34 +04:00
|
|
|
}
|
2016-09-15 10:15:59 +03:00
|
|
|
s = APIC(dev);
|
2005-01-04 02:27:31 +03:00
|
|
|
|
2024-01-11 18:43:58 +03:00
|
|
|
trace_apic_register_write(index, val);
|
2005-01-04 02:27:31 +03:00
|
|
|
|
|
|
|
switch(index) {
|
|
|
|
case 0x02:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2005-01-04 02:27:31 +03:00
|
|
|
s->id = (val >> 24);
|
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
case 0x03:
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x08:
|
2012-02-17 21:31:19 +04:00
|
|
|
if (apic_report_tpr_access) {
|
2012-10-10 16:10:07 +04:00
|
|
|
cpu_report_tpr_access(&s->cpu->env, TPR_ACCESS_WRITE);
|
2012-02-17 21:31:19 +04:00
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
s->tpr = val;
|
2012-02-17 21:31:19 +04:00
|
|
|
apic_sync_vapic(s, SYNC_TO_VAPIC);
|
2005-07-23 23:05:37 +04:00
|
|
|
apic_update_irq(s);
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
case 0x09:
|
|
|
|
case 0x0a:
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x0b: /* EOI */
|
|
|
|
apic_eoi(s);
|
|
|
|
break;
|
2005-07-23 23:05:37 +04:00
|
|
|
case 0x0d:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
s->log_dest = val >> 24;
|
|
|
|
break;
|
|
|
|
case 0x0e:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
s->dest_mode = val >> 28;
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x0f:
|
|
|
|
s->spurious_vec = val & 0x1ff;
|
2005-07-23 23:05:37 +04:00
|
|
|
apic_update_irq(s);
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
case 0x10 ... 0x17:
|
|
|
|
case 0x18 ... 0x1f:
|
|
|
|
case 0x20 ... 0x27:
|
|
|
|
case 0x28:
|
|
|
|
break;
|
2024-01-11 18:43:59 +03:00
|
|
|
case 0x30: {
|
|
|
|
uint32_t dest;
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
s->icr[0] = val;
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
s->icr[1] = val >> 32;
|
|
|
|
dest = s->icr[1];
|
|
|
|
} else {
|
|
|
|
dest = (s->icr[1] >> 24) & 0xff;
|
|
|
|
}
|
|
|
|
|
|
|
|
apic_deliver(dev, dest, (s->icr[0] >> 11) & 1,
|
2005-07-23 23:05:37 +04:00
|
|
|
(s->icr[0] >> 8) & 7, (s->icr[0] & 0xff),
|
2024-01-11 18:43:59 +03:00
|
|
|
(s->icr[0] >> 15) & 1, (s->icr[0] >> 18) & 3);
|
2005-07-23 23:05:37 +04:00
|
|
|
break;
|
2024-01-11 18:43:59 +03:00
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x31:
|
2024-01-11 18:43:59 +03:00
|
|
|
if (is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2005-07-23 23:05:37 +04:00
|
|
|
s->icr[1] = val;
|
2005-01-04 02:27:31 +03:00
|
|
|
break;
|
|
|
|
case 0x32 ... 0x37:
|
|
|
|
{
|
|
|
|
int n = index - 0x32;
|
|
|
|
s->lvt[n] = val;
|
2012-07-09 18:42:31 +04:00
|
|
|
if (n == APIC_LVT_TIMER) {
|
2013-08-21 19:03:08 +04:00
|
|
|
apic_timer_update(s, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL));
|
2012-07-09 18:42:31 +04:00
|
|
|
} else if (n == APIC_LVT_LINT0 && apic_check_pic(s)) {
|
|
|
|
apic_update_irq(s);
|
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x38:
|
|
|
|
s->initial_count = val;
|
2013-08-21 19:03:08 +04:00
|
|
|
s->initial_count_load_time = qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL);
|
2005-01-04 02:27:31 +03:00
|
|
|
apic_timer_update(s, s->initial_count_load_time);
|
|
|
|
break;
|
2005-11-22 02:26:26 +03:00
|
|
|
case 0x39:
|
|
|
|
break;
|
2005-01-04 02:27:31 +03:00
|
|
|
case 0x3e:
|
|
|
|
{
|
|
|
|
int v;
|
|
|
|
s->divide_conf = val & 0xb;
|
|
|
|
v = (s->divide_conf & 3) | ((s->divide_conf >> 1) & 4);
|
|
|
|
s->count_shift = (v + 1) & 7;
|
|
|
|
}
|
|
|
|
break;
|
2024-01-11 18:43:59 +03:00
|
|
|
case 0x3f: {
|
|
|
|
int vector = val & 0xff;
|
|
|
|
|
|
|
|
if (!is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Self IPI is identical to IPI with
|
|
|
|
* - Destination shorthand: 1 (Self)
|
|
|
|
* - Trigger mode: 0 (Edge)
|
|
|
|
* - Delivery mode: 0 (Fixed)
|
|
|
|
*/
|
|
|
|
apic_deliver(dev, 0, 0, APIC_DM_FIXED, vector, 0, 1);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2005-01-04 02:27:31 +03:00
|
|
|
default:
|
2015-09-22 16:18:14 +03:00
|
|
|
s->esr |= APIC_ESR_ILLEGAL_ADDRESS;
|
2024-01-11 18:43:58 +03:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void apic_mem_write(void *opaque, hwaddr addr, uint64_t val,
|
|
|
|
unsigned size)
|
|
|
|
{
|
|
|
|
int index = (addr >> 4) & 0xff;
|
|
|
|
|
|
|
|
if (size < 4) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (addr > 0xfff || !index) {
|
|
|
|
/*
|
|
|
|
* MSI and MMIO APIC are at the same memory location,
|
|
|
|
* but actually not on the global bus: MSI is on PCI bus
|
|
|
|
* APIC is connected directly to the CPU.
|
|
|
|
* Mapping them on the global bus happens to work because
|
|
|
|
* MSI registers are reserved in APIC MMIO and vice versa.
|
|
|
|
*/
|
|
|
|
MSIMessage msi = { .address = addr, .data = val };
|
|
|
|
apic_send_msi(&msi);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
apic_register_write(index, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
int apic_msr_write(int index, uint64_t val)
|
|
|
|
{
|
|
|
|
DeviceState *dev;
|
|
|
|
|
|
|
|
dev = cpu_get_current_apic();
|
|
|
|
if (!dev) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!is_x2apic_mode(dev)) {
|
|
|
|
return -1;
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
2024-01-11 18:43:58 +03:00
|
|
|
|
|
|
|
return apic_register_write(index, val);
|
2005-01-04 02:27:31 +03:00
|
|
|
}
|
|
|
|
|
2012-02-17 21:31:19 +04:00
|
|
|
static void apic_pre_save(APICCommonState *s)
|
|
|
|
{
|
|
|
|
apic_sync_vapic(s, SYNC_FROM_VAPIC);
|
|
|
|
}
|
|
|
|
|
2011-10-16 14:19:12 +04:00
|
|
|
static void apic_post_load(APICCommonState *s)
|
|
|
|
{
|
|
|
|
if (s->timer_expiry != -1) {
|
2013-08-21 19:03:08 +04:00
|
|
|
timer_mod(s->timer, s->timer_expiry);
|
2011-10-16 14:19:12 +04:00
|
|
|
} else {
|
2013-08-21 19:03:08 +04:00
|
|
|
timer_del(s->timer);
|
2011-10-16 14:19:12 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-08-15 18:17:16 +04:00
|
|
|
static const MemoryRegionOps apic_io_ops = {
|
2018-08-03 13:19:43 +03:00
|
|
|
.read = apic_mem_read,
|
|
|
|
.write = apic_mem_write,
|
|
|
|
.impl.min_access_size = 1,
|
|
|
|
.impl.max_access_size = 4,
|
|
|
|
.valid.min_access_size = 1,
|
|
|
|
.valid.max_access_size = 4,
|
2011-08-15 18:17:16 +04:00
|
|
|
.endianness = DEVICE_NATIVE_ENDIAN,
|
2005-01-04 02:27:31 +03:00
|
|
|
};
|
|
|
|
|
2013-11-05 14:16:03 +04:00
|
|
|
static void apic_realize(DeviceState *dev, Error **errp)
|
2010-06-19 11:44:07 +04:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2016-05-05 18:14:37 +03:00
|
|
|
|
2021-03-01 14:14:14 +03:00
|
|
|
if (kvm_enabled()) {
|
|
|
|
warn_report("Userspace local APIC is deprecated for KVM.");
|
|
|
|
warn_report("Do not use kernel-irqchip except for the -M isapc machine type.");
|
|
|
|
}
|
|
|
|
|
2013-06-07 05:25:08 +04:00
|
|
|
memory_region_init_io(&s->io_memory, OBJECT(s), &apic_io_ops, s, "apic-msi",
|
2013-04-25 18:05:29 +04:00
|
|
|
APIC_SPACE_SIZE);
|
2010-06-19 11:44:07 +04:00
|
|
|
|
2023-04-28 00:10:13 +03:00
|
|
|
/*
|
|
|
|
* apic-msi's apic_mem_write can call into ioapic_eoi_broadcast, which can
|
|
|
|
* write back to apic-msi. As such mark the apic-msi region re-entrancy
|
|
|
|
* safe.
|
|
|
|
*/
|
|
|
|
s->io_memory.disable_reentrancy_guard = true;
|
|
|
|
|
2013-08-21 19:03:08 +04:00
|
|
|
s->timer = timer_new_ns(QEMU_CLOCK_VIRTUAL, apic_timer, s);
|
2024-01-11 18:43:59 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The --machine none does not call apic_set_max_apic_id before creating
|
|
|
|
* apic, so we need to call it here and set it to 1 which is the max cpus
|
|
|
|
* in machine none.
|
|
|
|
*/
|
|
|
|
if (!local_apics) {
|
|
|
|
apic_set_max_apic_id(1);
|
|
|
|
}
|
|
|
|
local_apics[s->initial_apic_id] = s;
|
2012-05-16 22:41:11 +04:00
|
|
|
|
2016-03-04 12:24:28 +03:00
|
|
|
msi_nonbroken = true;
|
2010-06-19 11:44:07 +04:00
|
|
|
}
|
|
|
|
|
qdev: Unrealize must not fail
Devices may have component devices and buses.
Device realization may fail. Realization is recursive: a device's
realize() method realizes its components, and device_set_realized()
realizes its buses (which should in turn realize the devices on that
bus, except bus_set_realized() doesn't implement that, yet).
When realization of a component or bus fails, we need to roll back:
unrealize everything we realized so far. If any of these unrealizes
failed, the device would be left in an inconsistent state. Must not
happen.
device_set_realized() lets it happen: it ignores errors in the roll
back code starting at label child_realize_fail.
Since realization is recursive, unrealization must be recursive, too.
But how could a partly failed unrealize be rolled back? We'd have to
re-realize, which can fail. This design is fundamentally broken.
device_set_realized() does not roll back at all. Instead, it keeps
unrealizing, ignoring further errors.
It can screw up even for a device with no buses: if the lone
dc->unrealize() fails, it still unregisters vmstate, and calls
listeners' unrealize() callback.
bus_set_realized() does not roll back either. Instead, it stops
unrealizing.
Fortunately, no unrealize method can fail, as we'll see below.
To fix the design error, drop parameter @errp from all the unrealize
methods.
Any unrealize method that uses @errp now needs an update. This leads
us to unrealize() methods that can fail. Merely passing it to another
unrealize method cannot cause failure, though. Here are the ones that
do other things with @errp:
* virtio_serial_device_unrealize()
Fails when qbus_set_hotplug_handler() fails, but still does all the
other work. On failure, the device would stay realized with its
resources completely gone. Oops. Can't happen, because
qbus_set_hotplug_handler() can't actually fail here. Pass
&error_abort to qbus_set_hotplug_handler() instead.
* hw/ppc/spapr_drc.c's unrealize()
Fails when object_property_del() fails, but all the other work is
already done. On failure, the device would stay realized with its
vmstate registration gone. Oops. Can't happen, because
object_property_del() can't actually fail here. Pass &error_abort
to object_property_del() instead.
* spapr_phb_unrealize()
Fails and bails out when remove_drcs() fails, but other work is
already done. On failure, the device would stay realized with some
of its resources gone. Oops. remove_drcs() fails only when
chassis_from_bus()'s object_property_get_uint() fails, and it can't
here. Pass &error_abort to remove_drcs() instead.
Therefore, no unrealize method can fail before this patch.
device_set_realized()'s recursive unrealization via bus uses
object_property_set_bool(). Can't drop @errp there, so pass
&error_abort.
We similarly unrealize with object_property_set_bool() elsewhere,
always ignoring errors. Pass &error_abort instead.
Several unrealize methods no longer handle errors from other unrealize
methods: virtio_9p_device_unrealize(),
virtio_input_device_unrealize(), scsi_qdev_unrealize(), ...
Much of the deleted error handling looks wrong anyway.
One unrealize methods no longer ignore such errors:
usb_ehci_pci_exit().
Several realize methods no longer ignore errors when rolling back:
v9fs_device_realize_common(), pci_qdev_unrealize(),
spapr_phb_realize(), usb_qdev_realize(), vfio_ccw_realize(),
virtio_device_realize().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-Id: <20200505152926.18877-17-armbru@redhat.com>
2020-05-05 18:29:24 +03:00
|
|
|
static void apic_unrealize(DeviceState *dev)
|
2016-06-23 18:30:14 +03:00
|
|
|
{
|
2016-09-15 10:15:59 +03:00
|
|
|
APICCommonState *s = APIC(dev);
|
2016-06-23 18:30:14 +03:00
|
|
|
|
|
|
|
timer_free(s->timer);
|
2024-01-11 18:43:59 +03:00
|
|
|
local_apics[s->initial_apic_id] = NULL;
|
2016-06-23 18:30:14 +03:00
|
|
|
}
|
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static void apic_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
|
|
|
APICCommonClass *k = APIC_COMMON_CLASS(klass);
|
|
|
|
|
2013-11-05 14:16:03 +04:00
|
|
|
k->realize = apic_realize;
|
2016-06-23 18:30:14 +03:00
|
|
|
k->unrealize = apic_unrealize;
|
2012-01-24 23:12:29 +04:00
|
|
|
k->set_base = apic_set_base;
|
|
|
|
k->set_tpr = apic_set_tpr;
|
2012-02-17 21:31:19 +04:00
|
|
|
k->get_tpr = apic_get_tpr;
|
|
|
|
k->vapic_base_update = apic_vapic_base_update;
|
2012-01-24 23:12:29 +04:00
|
|
|
k->external_nmi = apic_external_nmi;
|
2012-02-17 21:31:19 +04:00
|
|
|
k->pre_save = apic_pre_save;
|
2012-01-24 23:12:29 +04:00
|
|
|
k->post_load = apic_post_load;
|
2016-10-10 18:28:43 +03:00
|
|
|
k->send_msi = apic_send_msi;
|
2012-01-24 23:12:29 +04:00
|
|
|
}
|
|
|
|
|
2013-01-10 19:19:07 +04:00
|
|
|
static const TypeInfo apic_info = {
|
2016-09-15 10:15:59 +03:00
|
|
|
.name = TYPE_APIC,
|
2011-12-08 07:34:16 +04:00
|
|
|
.instance_size = sizeof(APICCommonState),
|
|
|
|
.parent = TYPE_APIC_COMMON,
|
|
|
|
.class_init = apic_class_init,
|
2010-06-19 11:44:07 +04:00
|
|
|
};
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
static void apic_register_types(void)
|
2010-06-19 11:44:07 +04:00
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
type_register_static(&apic_info);
|
2010-06-19 11:44:07 +04:00
|
|
|
}
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
type_init(apic_register_types)
|