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 "monitor/monitor.h"
24 #include "hw/i386/ioapic.h"
25 #include "hw/i386/ioapic_internal.h"
26 #include "hw/sysbus.h"
28 /* ioapic_no count start from 0 to MAX_IOAPICS,
29 * remove as static variable from ioapic_common_init.
30 * now as a global variable, let child to increase the counter
31 * then we can drop the 'instance_no' argument
32 * and convert to our QOM's realize function
36 static void ioapic_irr_dump(Monitor
*mon
, const char *name
, uint32_t bitmap
)
40 monitor_printf(mon
, "%-10s ", name
);
42 monitor_printf(mon
, "(none)\n");
45 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
46 if (bitmap
& (1 << i
)) {
47 monitor_printf(mon
, "%-2u ", i
);
50 monitor_printf(mon
, "\n");
53 void ioapic_print_redtbl(Monitor
*mon
, IOAPICCommonState
*s
)
55 static const char *delm_str
[] = {
56 "fixed", "lowest", "SMI", "...", "NMI", "INIT", "...", "extINT"};
57 uint32_t remote_irr
= 0;
60 monitor_printf(mon
, "ioapic id=0x%02x sel=0x%02x", s
->id
, s
->ioregsel
);
62 monitor_printf(mon
, " (redir[%u])\n",
63 (s
->ioregsel
- IOAPIC_REG_REDTBL_BASE
) >> 1);
65 monitor_printf(mon
, "\n");
67 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
68 uint64_t entry
= s
->ioredtbl
[i
];
69 uint32_t delm
= (uint32_t)((entry
& IOAPIC_LVT_DELIV_MODE
) >>
70 IOAPIC_LVT_DELIV_MODE_SHIFT
);
71 monitor_printf(mon
, "pin %-2u 0x%016"PRIx64
" dest=%"PRIx64
72 " vec=%-3"PRIu64
" %s %-5s %-6s %-6s %s\n",
74 (entry
>> IOAPIC_LVT_DEST_SHIFT
) &
75 (entry
& IOAPIC_LVT_DEST_MODE
? 0xff : 0xf),
76 entry
& IOAPIC_VECTOR_MASK
,
77 entry
& IOAPIC_LVT_POLARITY
? "active-lo" : "active-hi",
78 entry
& IOAPIC_LVT_TRIGGER_MODE
? "level" : "edge",
79 entry
& IOAPIC_LVT_MASKED
? "masked" : "",
81 entry
& IOAPIC_LVT_DEST_MODE
? "logical" : "physical");
83 remote_irr
|= entry
& IOAPIC_LVT_TRIGGER_MODE
?
84 (entry
& IOAPIC_LVT_REMOTE_IRR
? (1 << i
) : 0) : 0;
86 ioapic_irr_dump(mon
, "IRR", s
->irr
);
87 ioapic_irr_dump(mon
, "Remote IRR", remote_irr
);
90 void ioapic_reset_common(DeviceState
*dev
)
92 IOAPICCommonState
*s
= IOAPIC_COMMON(dev
);
98 for (i
= 0; i
< IOAPIC_NUM_PINS
; i
++) {
99 s
->ioredtbl
[i
] = 1 << IOAPIC_LVT_MASKED_SHIFT
;
103 static void ioapic_dispatch_pre_save(void *opaque
)
105 IOAPICCommonState
*s
= IOAPIC_COMMON(opaque
);
106 IOAPICCommonClass
*info
= IOAPIC_COMMON_GET_CLASS(s
);
108 if (info
->pre_save
) {
113 static int ioapic_dispatch_post_load(void *opaque
, int version_id
)
115 IOAPICCommonState
*s
= IOAPIC_COMMON(opaque
);
116 IOAPICCommonClass
*info
= IOAPIC_COMMON_GET_CLASS(s
);
118 if (info
->post_load
) {
124 static void ioapic_common_realize(DeviceState
*dev
, Error
**errp
)
126 IOAPICCommonState
*s
= IOAPIC_COMMON(dev
);
127 IOAPICCommonClass
*info
;
129 if (ioapic_no
>= MAX_IOAPICS
) {
130 error_setg(errp
, "Only %d ioapics allowed", MAX_IOAPICS
);
134 info
= IOAPIC_COMMON_GET_CLASS(s
);
135 info
->realize(dev
, errp
);
137 sysbus_init_mmio(SYS_BUS_DEVICE(s
), &s
->io_memory
);
141 static const VMStateDescription vmstate_ioapic_common
= {
144 .minimum_version_id
= 1,
145 .pre_save
= ioapic_dispatch_pre_save
,
146 .post_load
= ioapic_dispatch_post_load
,
147 .fields
= (VMStateField
[]) {
148 VMSTATE_UINT8(id
, IOAPICCommonState
),
149 VMSTATE_UINT8(ioregsel
, IOAPICCommonState
),
150 VMSTATE_UNUSED_V(2, 8), /* to account for qemu-kvm's v2 format */
151 VMSTATE_UINT32_V(irr
, IOAPICCommonState
, 2),
152 VMSTATE_UINT64_ARRAY(ioredtbl
, IOAPICCommonState
, IOAPIC_NUM_PINS
),
153 VMSTATE_END_OF_LIST()
157 static void ioapic_common_class_init(ObjectClass
*klass
, void *data
)
159 DeviceClass
*dc
= DEVICE_CLASS(klass
);
161 dc
->realize
= ioapic_common_realize
;
162 dc
->vmsd
= &vmstate_ioapic_common
;
165 static const TypeInfo ioapic_common_type
= {
166 .name
= TYPE_IOAPIC_COMMON
,
167 .parent
= TYPE_SYS_BUS_DEVICE
,
168 .instance_size
= sizeof(IOAPICCommonState
),
169 .class_size
= sizeof(IOAPICCommonClass
),
170 .class_init
= ioapic_common_class_init
,
174 static void ioapic_common_register_types(void)
176 type_register_static(&ioapic_common_type
);
179 type_init(ioapic_common_register_types
)