2 * IOAPIC emulation logic - common bits of emulated and KVM kernel model
4 * Copyright (c) 2004-2005 Fabrice Bellard
5 * Copyright (c) 2009 Xiantao Zhang, Intel
6 * Copyright (c) 2011 Jan Kiszka, Siemens AG
8 * This library is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU Lesser General Public
10 * License as published by the Free Software Foundation; either
11 * version 2 of the License, or (at your option) any later version.
13 * This library is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 * Lesser General Public License for more details.
18 * You should have received a copy of the GNU Lesser General Public
19 * License along with this library; if not, see <http://www.gnu.org/licenses/>.
22 #include "qemu/osdep.h"
23 #include "qapi/error.h"
24 #include "monitor/monitor.h"
25 #include "hw/i386/ioapic.h"
26 #include "hw/i386/ioapic_internal.h"
27 #include "hw/intc/intc.h"
28 #include "hw/sysbus.h"
30 /* ioapic_no count start from 0 to MAX_IOAPICS,
31 * remove as static variable from ioapic_common_init.
32 * now as a global variable, let child to increase the counter
33 * then we can drop the 'instance_no' argument
34 * and convert to our QOM's realize function
38 void ioapic_stat_update_irq(IOAPICCommonState
*s
, int irq
, int level
)
40 if (level
!= s
->irq_level
[irq
]) {
41 s
->irq_level
[irq
] = level
;
48 static bool ioapic_get_statistics(InterruptStatsProvider
*obj
,
49 uint64_t **irq_counts
,
50 unsigned int *nb_irqs
)
52 IOAPICCommonState
*s
= IOAPIC_COMMON(obj
);
54 *irq_counts
= s
->irq_count
;
55 *nb_irqs
= IOAPIC_NUM_PINS
;
60 static void ioapic_irr_dump(Monitor
*mon
, const char *name
, uint32_t bitmap
)
64 monitor_printf(mon
, "%-10s ", name
);
66 monitor_printf(mon
, "(none)\n");
69 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
70 if (bitmap
& (1 << i
)) {
71 monitor_printf(mon
, "%-2u ", i
);
74 monitor_printf(mon
, "\n");
77 void ioapic_print_redtbl(Monitor
*mon
, IOAPICCommonState
*s
)
79 static const char *delm_str
[] = {
80 "fixed", "lowest", "SMI", "...", "NMI", "INIT", "...", "extINT"};
81 uint32_t remote_irr
= 0;
84 monitor_printf(mon
, "ioapic0: ver=0x%x id=0x%02x sel=0x%02x",
85 s
->version
, s
->id
, s
->ioregsel
);
87 monitor_printf(mon
, " (redir[%u])\n",
88 (s
->ioregsel
- IOAPIC_REG_REDTBL_BASE
) >> 1);
90 monitor_printf(mon
, "\n");
92 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
93 uint64_t entry
= s
->ioredtbl
[i
];
94 uint32_t delm
= (uint32_t)((entry
& IOAPIC_LVT_DELIV_MODE
) >>
95 IOAPIC_LVT_DELIV_MODE_SHIFT
);
96 monitor_printf(mon
, " pin %-2u 0x%016"PRIx64
" dest=%"PRIx64
97 " vec=%-3"PRIu64
" %s %-5s %-6s %-6s %s\n",
99 (entry
>> IOAPIC_LVT_DEST_SHIFT
) &
100 (entry
& IOAPIC_LVT_DEST_MODE
? 0xff : 0xf),
101 entry
& IOAPIC_VECTOR_MASK
,
102 entry
& IOAPIC_LVT_POLARITY
? "active-lo" : "active-hi",
103 entry
& IOAPIC_LVT_TRIGGER_MODE
? "level" : "edge",
104 entry
& IOAPIC_LVT_MASKED
? "masked" : "",
106 entry
& IOAPIC_LVT_DEST_MODE
? "logical" : "physical");
108 remote_irr
|= entry
& IOAPIC_LVT_TRIGGER_MODE
?
109 (entry
& IOAPIC_LVT_REMOTE_IRR
? (1 << i
) : 0) : 0;
111 ioapic_irr_dump(mon
, " IRR", s
->irr
);
112 ioapic_irr_dump(mon
, " Remote IRR", remote_irr
);
115 void ioapic_reset_common(DeviceState
*dev
)
117 IOAPICCommonState
*s
= IOAPIC_COMMON(dev
);
123 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
124 s
->ioredtbl
[i
] = 1 << IOAPIC_LVT_MASKED_SHIFT
;
128 static int ioapic_dispatch_pre_save(void *opaque
)
130 IOAPICCommonState
*s
= IOAPIC_COMMON(opaque
);
131 IOAPICCommonClass
*info
= IOAPIC_COMMON_GET_CLASS(s
);
133 if (info
->pre_save
) {
140 static int ioapic_dispatch_post_load(void *opaque
, int version_id
)
142 IOAPICCommonState
*s
= IOAPIC_COMMON(opaque
);
143 IOAPICCommonClass
*info
= IOAPIC_COMMON_GET_CLASS(s
);
145 if (info
->post_load
) {
151 static void ioapic_common_realize(DeviceState
*dev
, Error
**errp
)
153 IOAPICCommonState
*s
= IOAPIC_COMMON(dev
);
154 IOAPICCommonClass
*info
;
156 if (ioapic_no
>= MAX_IOAPICS
) {
157 error_setg(errp
, "Only %d ioapics allowed", MAX_IOAPICS
);
161 info
= IOAPIC_COMMON_GET_CLASS(s
);
162 info
->realize(dev
, errp
);
164 sysbus_init_mmio(SYS_BUS_DEVICE(s
), &s
->io_memory
);
168 static void ioapic_print_info(InterruptStatsProvider
*obj
,
171 IOAPICCommonState
*s
= IOAPIC_COMMON(obj
);
173 ioapic_dispatch_pre_save(s
);
174 ioapic_print_redtbl(mon
, s
);
177 static const VMStateDescription vmstate_ioapic_common
= {
180 .minimum_version_id
= 1,
181 .pre_save
= ioapic_dispatch_pre_save
,
182 .post_load
= ioapic_dispatch_post_load
,
183 .fields
= (VMStateField
[]) {
184 VMSTATE_UINT8(id
, IOAPICCommonState
),
185 VMSTATE_UINT8(ioregsel
, IOAPICCommonState
),
186 VMSTATE_UNUSED_V(2, 8), /* to account for qemu-kvm's v2 format */
187 VMSTATE_UINT32_V(irr
, IOAPICCommonState
, 2),
188 VMSTATE_UINT64_ARRAY(ioredtbl
, IOAPICCommonState
, IOAPIC_NUM_PINS
),
189 VMSTATE_END_OF_LIST()
193 static void ioapic_common_class_init(ObjectClass
*klass
, void *data
)
195 DeviceClass
*dc
= DEVICE_CLASS(klass
);
196 InterruptStatsProviderClass
*ic
= INTERRUPT_STATS_PROVIDER_CLASS(klass
);
198 dc
->realize
= ioapic_common_realize
;
199 dc
->vmsd
= &vmstate_ioapic_common
;
200 ic
->print_info
= ioapic_print_info
;
201 ic
->get_statistics
= ioapic_get_statistics
;
204 static const TypeInfo ioapic_common_type
= {
205 .name
= TYPE_IOAPIC_COMMON
,
206 .parent
= TYPE_SYS_BUS_DEVICE
,
207 .instance_size
= sizeof(IOAPICCommonState
),
208 .class_size
= sizeof(IOAPICCommonClass
),
209 .class_init
= ioapic_common_class_init
,
211 .interfaces
= (InterfaceInfo
[]) {
212 { TYPE_INTERRUPT_STATS_PROVIDER
},
217 static void ioapic_common_register_types(void)
219 type_register_static(&ioapic_common_type
);
222 type_init(ioapic_common_register_types
)