i8259: Convert to qdev
This key cleanup step requires to move the IRQ debugging bit from i8259_set_irq directly to the per-PIC pic_set_irq, to pass the PIC parameters (I/O base, ELCR address and mask, master/slave mode) as qdev properties, and to interconnect the PICs with their environment via GPIO pins. Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com> Signed-off-by: Blue Swirl <blauwirbel@gmail.com>
This commit is contained in:
parent
6835678c25
commit
747c70af78
157
hw/i8259.c
157
hw/i8259.c
@ -41,6 +41,7 @@
|
|||||||
//#define DEBUG_IRQ_COUNT
|
//#define DEBUG_IRQ_COUNT
|
||||||
|
|
||||||
struct PicState {
|
struct PicState {
|
||||||
|
ISADevice dev;
|
||||||
uint8_t last_irr; /* edge detection */
|
uint8_t last_irr; /* edge detection */
|
||||||
uint8_t irr; /* interrupt request register */
|
uint8_t irr; /* interrupt request register */
|
||||||
uint8_t imr; /* interrupt mask register */
|
uint8_t imr; /* interrupt mask register */
|
||||||
@ -58,8 +59,10 @@ struct PicState {
|
|||||||
uint8_t single_mode; /* true if slave pic is not initialized */
|
uint8_t single_mode; /* true if slave pic is not initialized */
|
||||||
uint8_t elcr; /* PIIX edge/trigger selection*/
|
uint8_t elcr; /* PIIX edge/trigger selection*/
|
||||||
uint8_t elcr_mask;
|
uint8_t elcr_mask;
|
||||||
qemu_irq int_out;
|
qemu_irq int_out[1];
|
||||||
bool master; /* reflects /SP input pin */
|
uint32_t master; /* reflects /SP input pin */
|
||||||
|
uint32_t iobase;
|
||||||
|
uint32_t elcr_addr;
|
||||||
MemoryRegion base_io;
|
MemoryRegion base_io;
|
||||||
MemoryRegion elcr_io;
|
MemoryRegion elcr_io;
|
||||||
};
|
};
|
||||||
@ -70,6 +73,9 @@ static int irq_level[16];
|
|||||||
#ifdef DEBUG_IRQ_COUNT
|
#ifdef DEBUG_IRQ_COUNT
|
||||||
static uint64_t irq_count[16];
|
static uint64_t irq_count[16];
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef DEBUG_IRQ_LATENCY
|
||||||
|
static int64_t irq_time[16];
|
||||||
|
#endif
|
||||||
PicState *isa_pic;
|
PicState *isa_pic;
|
||||||
static PicState *slave_pic;
|
static PicState *slave_pic;
|
||||||
|
|
||||||
@ -122,17 +128,39 @@ static void pic_update_irq(PicState *s)
|
|||||||
if (irq >= 0) {
|
if (irq >= 0) {
|
||||||
DPRINTF("pic%d: imr=%x irr=%x padd=%d\n",
|
DPRINTF("pic%d: imr=%x irr=%x padd=%d\n",
|
||||||
s->master ? 0 : 1, s->imr, s->irr, s->priority_add);
|
s->master ? 0 : 1, s->imr, s->irr, s->priority_add);
|
||||||
qemu_irq_raise(s->int_out);
|
qemu_irq_raise(s->int_out[0]);
|
||||||
} else {
|
} else {
|
||||||
qemu_irq_lower(s->int_out);
|
qemu_irq_lower(s->int_out[0]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* set irq level. If an edge is detected, then the IRR is set to 1 */
|
/* set irq level. If an edge is detected, then the IRR is set to 1 */
|
||||||
static void pic_set_irq1(PicState *s, int irq, int level)
|
static void pic_set_irq(void *opaque, int irq, int level)
|
||||||
{
|
{
|
||||||
int mask;
|
PicState *s = opaque;
|
||||||
mask = 1 << irq;
|
int mask = 1 << irq;
|
||||||
|
|
||||||
|
#if defined(DEBUG_PIC) || defined(DEBUG_IRQ_COUNT) || \
|
||||||
|
defined(DEBUG_IRQ_LATENCY)
|
||||||
|
int irq_index = s->master ? irq : irq + 8;
|
||||||
|
#endif
|
||||||
|
#if defined(DEBUG_PIC) || defined(DEBUG_IRQ_COUNT)
|
||||||
|
if (level != irq_level[irq_index]) {
|
||||||
|
DPRINTF("pic_set_irq: irq=%d level=%d\n", irq_index, level);
|
||||||
|
irq_level[irq_index] = level;
|
||||||
|
#ifdef DEBUG_IRQ_COUNT
|
||||||
|
if (level == 1) {
|
||||||
|
irq_count[irq_index]++;
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
#ifdef DEBUG_IRQ_LATENCY
|
||||||
|
if (level) {
|
||||||
|
irq_time[irq_index] = qemu_get_clock_ns(vm_clock);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
if (s->elcr & mask) {
|
if (s->elcr & mask) {
|
||||||
/* level triggered */
|
/* level triggered */
|
||||||
if (level) {
|
if (level) {
|
||||||
@ -156,32 +184,6 @@ static void pic_set_irq1(PicState *s, int irq, int level)
|
|||||||
pic_update_irq(s);
|
pic_update_irq(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG_IRQ_LATENCY
|
|
||||||
int64_t irq_time[16];
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static void i8259_set_irq(void *opaque, int irq, int level)
|
|
||||||
{
|
|
||||||
PicState *s = irq <= 7 ? isa_pic : slave_pic;
|
|
||||||
|
|
||||||
#if defined(DEBUG_PIC) || defined(DEBUG_IRQ_COUNT)
|
|
||||||
if (level != irq_level[irq]) {
|
|
||||||
DPRINTF("i8259_set_irq: irq=%d level=%d\n", irq, level);
|
|
||||||
irq_level[irq] = level;
|
|
||||||
#ifdef DEBUG_IRQ_COUNT
|
|
||||||
if (level == 1)
|
|
||||||
irq_count[irq]++;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
#ifdef DEBUG_IRQ_LATENCY
|
|
||||||
if (level) {
|
|
||||||
irq_time[irq] = qemu_get_clock_ns(vm_clock);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
pic_set_irq1(s, irq & 7, level);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* acknowledge interrupt 'irq' */
|
/* acknowledge interrupt 'irq' */
|
||||||
static void pic_intack(PicState *s, int irq)
|
static void pic_intack(PicState *s, int irq)
|
||||||
{
|
{
|
||||||
@ -258,9 +260,9 @@ static void pic_init_reset(PicState *s)
|
|||||||
pic_update_irq(s);
|
pic_update_irq(s);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void pic_reset(void *opaque)
|
static void pic_reset(DeviceState *dev)
|
||||||
{
|
{
|
||||||
PicState *s = opaque;
|
PicState *s = container_of(dev, PicState, dev.qdev);
|
||||||
|
|
||||||
pic_init_reset(s);
|
pic_init_reset(s);
|
||||||
s->elcr = 0;
|
s->elcr = 0;
|
||||||
@ -447,23 +449,24 @@ static const MemoryRegionOps pic_elcr_ioport_ops = {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
/* XXX: add generic master/slave system */
|
static int pic_initfn(ISADevice *dev)
|
||||||
static void pic_init(int io_addr, int elcr_addr, PicState *s, qemu_irq int_out,
|
|
||||||
bool master)
|
|
||||||
{
|
{
|
||||||
s->int_out = int_out;
|
PicState *s = DO_UPCAST(PicState, dev, dev);
|
||||||
s->master = master;
|
|
||||||
|
|
||||||
memory_region_init_io(&s->base_io, &pic_base_ioport_ops, s, "pic", 2);
|
memory_region_init_io(&s->base_io, &pic_base_ioport_ops, s, "pic", 2);
|
||||||
memory_region_init_io(&s->elcr_io, &pic_elcr_ioport_ops, s, "elcr", 1);
|
memory_region_init_io(&s->elcr_io, &pic_elcr_ioport_ops, s, "elcr", 1);
|
||||||
|
|
||||||
isa_register_ioport(NULL, &s->base_io, io_addr);
|
isa_register_ioport(NULL, &s->base_io, s->iobase);
|
||||||
if (elcr_addr >= 0) {
|
if (s->elcr_addr != -1) {
|
||||||
isa_register_ioport(NULL, &s->elcr_io, elcr_addr);
|
isa_register_ioport(NULL, &s->elcr_io, s->elcr_addr);
|
||||||
}
|
}
|
||||||
|
|
||||||
vmstate_register(NULL, io_addr, &vmstate_pic, s);
|
qdev_init_gpio_out(&dev->qdev, s->int_out, ARRAY_SIZE(s->int_out));
|
||||||
qemu_register_reset(pic_reset, s);
|
qdev_init_gpio_in(&dev->qdev, pic_set_irq, 8);
|
||||||
|
|
||||||
|
qdev_set_legacy_instance_id(&dev->qdev, s->iobase, 1);
|
||||||
|
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void pic_info(Monitor *mon)
|
void pic_info(Monitor *mon)
|
||||||
@ -503,20 +506,60 @@ void irq_info(Monitor *mon)
|
|||||||
|
|
||||||
qemu_irq *i8259_init(qemu_irq parent_irq)
|
qemu_irq *i8259_init(qemu_irq parent_irq)
|
||||||
{
|
{
|
||||||
qemu_irq *irqs;
|
qemu_irq *irq_set;
|
||||||
PicState *s;
|
ISADevice *dev;
|
||||||
|
int i;
|
||||||
|
|
||||||
irqs = qemu_allocate_irqs(i8259_set_irq, NULL, 16);
|
irq_set = g_malloc(ISA_NUM_IRQS * sizeof(qemu_irq));
|
||||||
|
|
||||||
s = g_malloc0(sizeof(PicState));
|
dev = isa_create("isa-i8259");
|
||||||
pic_init(0x20, 0x4d0, s, parent_irq, true);
|
qdev_prop_set_uint32(&dev->qdev, "iobase", 0x20);
|
||||||
s->elcr_mask = 0xf8;
|
qdev_prop_set_uint32(&dev->qdev, "elcr_addr", 0x4d0);
|
||||||
isa_pic = s;
|
qdev_prop_set_uint8(&dev->qdev, "elcr_mask", 0xf8);
|
||||||
|
qdev_prop_set_bit(&dev->qdev, "master", true);
|
||||||
|
qdev_init_nofail(&dev->qdev);
|
||||||
|
|
||||||
s = g_malloc0(sizeof(PicState));
|
qdev_connect_gpio_out(&dev->qdev, 0, parent_irq);
|
||||||
pic_init(0xa0, 0x4d1, s, irqs[2], false);
|
for (i = 0 ; i < 8; i++) {
|
||||||
s->elcr_mask = 0xde;
|
irq_set[i] = qdev_get_gpio_in(&dev->qdev, i);
|
||||||
slave_pic = s;
|
}
|
||||||
|
|
||||||
return irqs;
|
isa_pic = DO_UPCAST(PicState, dev, dev);
|
||||||
|
|
||||||
|
dev = isa_create("isa-i8259");
|
||||||
|
qdev_prop_set_uint32(&dev->qdev, "iobase", 0xa0);
|
||||||
|
qdev_prop_set_uint32(&dev->qdev, "elcr_addr", 0x4d1);
|
||||||
|
qdev_prop_set_uint8(&dev->qdev, "elcr_mask", 0xde);
|
||||||
|
qdev_init_nofail(&dev->qdev);
|
||||||
|
|
||||||
|
qdev_connect_gpio_out(&dev->qdev, 0, irq_set[2]);
|
||||||
|
for (i = 0 ; i < 8; i++) {
|
||||||
|
irq_set[i + 8] = qdev_get_gpio_in(&dev->qdev, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
slave_pic = DO_UPCAST(PicState, dev, dev);
|
||||||
|
|
||||||
|
return irq_set;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ISADeviceInfo i8259_info = {
|
||||||
|
.qdev.name = "isa-i8259",
|
||||||
|
.qdev.size = sizeof(PicState),
|
||||||
|
.qdev.vmsd = &vmstate_pic,
|
||||||
|
.qdev.reset = pic_reset,
|
||||||
|
.qdev.no_user = 1,
|
||||||
|
.init = pic_initfn,
|
||||||
|
.qdev.props = (Property[]) {
|
||||||
|
DEFINE_PROP_HEX32("iobase", PicState, iobase, -1),
|
||||||
|
DEFINE_PROP_HEX32("elcr_addr", PicState, elcr_addr, -1),
|
||||||
|
DEFINE_PROP_HEX8("elcr_mask", PicState, elcr_mask, -1),
|
||||||
|
DEFINE_PROP_BIT("master", PicState, master, 0, false),
|
||||||
|
DEFINE_PROP_END_OF_LIST(),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
static void pic_register(void)
|
||||||
|
{
|
||||||
|
isa_qdev_register(&i8259_info);
|
||||||
|
}
|
||||||
|
device_init(pic_register)
|
||||||
|
Loading…
Reference in New Issue
Block a user