
In an ideal world, machines can be built by wiring devices together with configuration, not code. Unfortunately, that's not the world we live in right now. We still have quite a few devices that need to be wired up by code. If you try to device_add such a device, it'll fail in sometimes mysterious ways. If you're lucky, you get an unmysterious immediate crash. To protect users from such badness, DeviceClass member no_user used to make device models unavailable with -device / device_add, but that regressed in commit 18b6dad. The device model is still omitted from help, but is available anyway. Attempts to fix the regression have been rejected with the argument that the purpose of no_user isn't clear, and it's prone to misuse. This commit clarifies no_user's purpose. Anthony suggested to rename it cannot_instantiate_with_device_add_yet_due_to_internal_bugs, which I shorten somewhat to keep checkpatch happy. While there, make it bool. Every use of cannot_instantiate_with_device_add_yet gets a FIXME comment asking for rationale. The next few commits will clean them all up, either by providing a rationale, or by getting rid of the use. With that done, the regression fix is hopefully acceptable. Signed-off-by: Markus Armbruster <armbru@redhat.com> Reviewed-by: Marcel Apfelbaum <marcel.a@redhat.com> Signed-off-by: Andreas Färber <afaerber@suse.de>
119 lines
3.4 KiB
C
119 lines
3.4 KiB
C
/*
|
|
* IOAPIC emulation logic - common bits of emulated and KVM kernel model
|
|
*
|
|
* Copyright (c) 2004-2005 Fabrice Bellard
|
|
* Copyright (c) 2009 Xiantao Zhang, Intel
|
|
* 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/>.
|
|
*/
|
|
|
|
#include "hw/i386/ioapic.h"
|
|
#include "hw/i386/ioapic_internal.h"
|
|
#include "hw/sysbus.h"
|
|
|
|
void ioapic_reset_common(DeviceState *dev)
|
|
{
|
|
IOAPICCommonState *s = IOAPIC_COMMON(dev);
|
|
int i;
|
|
|
|
s->id = 0;
|
|
s->ioregsel = 0;
|
|
s->irr = 0;
|
|
for (i = 0; i < IOAPIC_NUM_PINS; i++) {
|
|
s->ioredtbl[i] = 1 << IOAPIC_LVT_MASKED_SHIFT;
|
|
}
|
|
}
|
|
|
|
static void ioapic_dispatch_pre_save(void *opaque)
|
|
{
|
|
IOAPICCommonState *s = IOAPIC_COMMON(opaque);
|
|
IOAPICCommonClass *info = IOAPIC_COMMON_GET_CLASS(s);
|
|
|
|
if (info->pre_save) {
|
|
info->pre_save(s);
|
|
}
|
|
}
|
|
|
|
static int ioapic_dispatch_post_load(void *opaque, int version_id)
|
|
{
|
|
IOAPICCommonState *s = IOAPIC_COMMON(opaque);
|
|
IOAPICCommonClass *info = IOAPIC_COMMON_GET_CLASS(s);
|
|
|
|
if (info->post_load) {
|
|
info->post_load(s);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static void ioapic_common_realize(DeviceState *dev, Error **errp)
|
|
{
|
|
IOAPICCommonState *s = IOAPIC_COMMON(dev);
|
|
IOAPICCommonClass *info;
|
|
static int ioapic_no;
|
|
|
|
if (ioapic_no >= MAX_IOAPICS) {
|
|
error_setg(errp, "Only %d ioapics allowed", MAX_IOAPICS);
|
|
return;
|
|
}
|
|
|
|
info = IOAPIC_COMMON_GET_CLASS(s);
|
|
info->init(s, ioapic_no);
|
|
|
|
sysbus_init_mmio(SYS_BUS_DEVICE(s), &s->io_memory);
|
|
ioapic_no++;
|
|
}
|
|
|
|
static const VMStateDescription vmstate_ioapic_common = {
|
|
.name = "ioapic",
|
|
.version_id = 3,
|
|
.minimum_version_id = 1,
|
|
.minimum_version_id_old = 1,
|
|
.pre_save = ioapic_dispatch_pre_save,
|
|
.post_load = ioapic_dispatch_post_load,
|
|
.fields = (VMStateField[]) {
|
|
VMSTATE_UINT8(id, IOAPICCommonState),
|
|
VMSTATE_UINT8(ioregsel, IOAPICCommonState),
|
|
VMSTATE_UNUSED_V(2, 8), /* to account for qemu-kvm's v2 format */
|
|
VMSTATE_UINT32_V(irr, IOAPICCommonState, 2),
|
|
VMSTATE_UINT64_ARRAY(ioredtbl, IOAPICCommonState, IOAPIC_NUM_PINS),
|
|
VMSTATE_END_OF_LIST()
|
|
}
|
|
};
|
|
|
|
static void ioapic_common_class_init(ObjectClass *klass, void *data)
|
|
{
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
|
|
|
dc->realize = ioapic_common_realize;
|
|
dc->vmsd = &vmstate_ioapic_common;
|
|
dc->cannot_instantiate_with_device_add_yet = true; /* FIXME explain why */
|
|
}
|
|
|
|
static const TypeInfo ioapic_common_type = {
|
|
.name = TYPE_IOAPIC_COMMON,
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
|
.instance_size = sizeof(IOAPICCommonState),
|
|
.class_size = sizeof(IOAPICCommonClass),
|
|
.class_init = ioapic_common_class_init,
|
|
.abstract = true,
|
|
};
|
|
|
|
static void register_types(void)
|
|
{
|
|
type_register_static(&ioapic_common_type);
|
|
}
|
|
|
|
type_init(register_types)
|