mirror of
https://github.com/Motorhead1991/qemu.git
synced 2025-08-02 15:23:53 -06:00
hw/intc/aspeed: Introduce helper functions for enable and status registers
The behavior of the enable and status registers is almost identical between INTC(CPU Die) and INTCIO(IO Die). To reduce duplicated code, adds "aspeed_intc_enable_handler" functions to handle enable register write behavior and "aspeed_intc_status_handler" functions to handle status register write behavior. No functional change. Signed-off-by: Jamin Lin <jamin_lin@aspeedtech.com> Reviewed-by: Cédric Le Goater <clg@redhat.com> Link: https://lore.kernel.org/qemu-devel/20250307035945.3698802-7-jamin_lin@aspeedtech.com Signed-off-by: Cédric Le Goater <clg@redhat.com>
This commit is contained in:
parent
7ffee511fc
commit
3d6e15eafb
1 changed files with 108 additions and 83 deletions
|
@ -120,6 +120,112 @@ static void aspeed_intc_set_irq(void *opaque, int irq, int level)
|
|||
}
|
||||
}
|
||||
|
||||
static void aspeed_intc_enable_handler(AspeedINTCState *s, hwaddr offset,
|
||||
uint64_t data)
|
||||
{
|
||||
AspeedINTCClass *aic = ASPEED_INTC_GET_CLASS(s);
|
||||
uint32_t reg = offset >> 2;
|
||||
uint32_t old_enable;
|
||||
uint32_t change;
|
||||
uint32_t irq;
|
||||
|
||||
irq = (offset & 0x0f00) >> 8;
|
||||
|
||||
if (irq >= aic->num_ints) {
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid interrupt number: %d\n",
|
||||
__func__, irq);
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* The enable registers are used to enable source interrupts.
|
||||
* They also handle masking and unmasking of source interrupts
|
||||
* during the execution of the source ISR.
|
||||
*/
|
||||
|
||||
/* disable all source interrupt */
|
||||
if (!data && !s->enable[irq]) {
|
||||
s->regs[reg] = data;
|
||||
return;
|
||||
}
|
||||
|
||||
old_enable = s->enable[irq];
|
||||
s->enable[irq] |= data;
|
||||
|
||||
/* enable new source interrupt */
|
||||
if (old_enable != s->enable[irq]) {
|
||||
trace_aspeed_intc_enable(s->enable[irq]);
|
||||
s->regs[reg] = data;
|
||||
return;
|
||||
}
|
||||
|
||||
/* mask and unmask source interrupt */
|
||||
change = s->regs[reg] ^ data;
|
||||
if (change & data) {
|
||||
s->mask[irq] &= ~change;
|
||||
trace_aspeed_intc_unmask(change, s->mask[irq]);
|
||||
} else {
|
||||
s->mask[irq] |= change;
|
||||
trace_aspeed_intc_mask(change, s->mask[irq]);
|
||||
}
|
||||
|
||||
s->regs[reg] = data;
|
||||
}
|
||||
|
||||
static void aspeed_intc_status_handler(AspeedINTCState *s, hwaddr offset,
|
||||
uint64_t data)
|
||||
{
|
||||
AspeedINTCClass *aic = ASPEED_INTC_GET_CLASS(s);
|
||||
uint32_t reg = offset >> 2;
|
||||
uint32_t irq;
|
||||
|
||||
if (!data) {
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid data 0\n", __func__);
|
||||
return;
|
||||
}
|
||||
|
||||
irq = (offset & 0x0f00) >> 8;
|
||||
|
||||
if (irq >= aic->num_ints) {
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid interrupt number: %d\n",
|
||||
__func__, irq);
|
||||
return;
|
||||
}
|
||||
|
||||
/* clear status */
|
||||
s->regs[reg] &= ~data;
|
||||
|
||||
/*
|
||||
* These status registers are used for notify sources ISR are executed.
|
||||
* If one source ISR is executed, it will clear one bit.
|
||||
* If it clear all bits, it means to initialize this register status
|
||||
* rather than sources ISR are executed.
|
||||
*/
|
||||
if (data == 0xffffffff) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* All source ISR execution are done */
|
||||
if (!s->regs[reg]) {
|
||||
trace_aspeed_intc_all_isr_done(irq);
|
||||
if (s->pending[irq]) {
|
||||
/*
|
||||
* handle pending source interrupt
|
||||
* notify firmware which source interrupt are pending
|
||||
* by setting status register
|
||||
*/
|
||||
s->regs[reg] = s->pending[irq];
|
||||
s->pending[irq] = 0;
|
||||
trace_aspeed_intc_trigger_irq(irq, s->regs[reg]);
|
||||
aspeed_intc_update(s, irq, 1);
|
||||
} else {
|
||||
/* clear irq */
|
||||
trace_aspeed_intc_clear_irq(irq, 0);
|
||||
aspeed_intc_update(s, irq, 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static uint64_t aspeed_intc_read(void *opaque, hwaddr offset, unsigned int size)
|
||||
{
|
||||
AspeedINTCState *s = ASPEED_INTC(opaque);
|
||||
|
@ -136,11 +242,7 @@ static void aspeed_intc_write(void *opaque, hwaddr offset, uint64_t data,
|
|||
unsigned size)
|
||||
{
|
||||
AspeedINTCState *s = ASPEED_INTC(opaque);
|
||||
AspeedINTCClass *aic = ASPEED_INTC_GET_CLASS(s);
|
||||
uint32_t reg = offset >> 2;
|
||||
uint32_t old_enable;
|
||||
uint32_t change;
|
||||
uint32_t irq;
|
||||
|
||||
trace_aspeed_intc_write(offset, size, data);
|
||||
|
||||
|
@ -154,45 +256,7 @@ static void aspeed_intc_write(void *opaque, hwaddr offset, uint64_t data,
|
|||
case R_GICINT134_EN:
|
||||
case R_GICINT135_EN:
|
||||
case R_GICINT136_EN:
|
||||
irq = (offset & 0x0f00) >> 8;
|
||||
|
||||
if (irq >= aic->num_ints) {
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid interrupt number: %d\n",
|
||||
__func__, irq);
|
||||
return;
|
||||
}
|
||||
|
||||
/*
|
||||
* These registers are used for enable sources interrupt and
|
||||
* mask and unmask source interrupt while executing source ISR.
|
||||
*/
|
||||
|
||||
/* disable all source interrupt */
|
||||
if (!data && !s->enable[irq]) {
|
||||
s->regs[reg] = data;
|
||||
return;
|
||||
}
|
||||
|
||||
old_enable = s->enable[irq];
|
||||
s->enable[irq] |= data;
|
||||
|
||||
/* enable new source interrupt */
|
||||
if (old_enable != s->enable[irq]) {
|
||||
trace_aspeed_intc_enable(s->enable[irq]);
|
||||
s->regs[reg] = data;
|
||||
return;
|
||||
}
|
||||
|
||||
/* mask and unmask source interrupt */
|
||||
change = s->regs[reg] ^ data;
|
||||
if (change & data) {
|
||||
s->mask[irq] &= ~change;
|
||||
trace_aspeed_intc_unmask(change, s->mask[irq]);
|
||||
} else {
|
||||
s->mask[irq] |= change;
|
||||
trace_aspeed_intc_mask(change, s->mask[irq]);
|
||||
}
|
||||
s->regs[reg] = data;
|
||||
aspeed_intc_enable_handler(s, offset, data);
|
||||
break;
|
||||
case R_GICINT128_STATUS:
|
||||
case R_GICINT129_STATUS:
|
||||
|
@ -203,46 +267,7 @@ static void aspeed_intc_write(void *opaque, hwaddr offset, uint64_t data,
|
|||
case R_GICINT134_STATUS:
|
||||
case R_GICINT135_STATUS:
|
||||
case R_GICINT136_STATUS:
|
||||
irq = (offset & 0x0f00) >> 8;
|
||||
|
||||
if (irq >= aic->num_ints) {
|
||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: Invalid interrupt number: %d\n",
|
||||
__func__, irq);
|
||||
return;
|
||||
}
|
||||
|
||||
/* clear status */
|
||||
s->regs[reg] &= ~data;
|
||||
|
||||
/*
|
||||
* These status registers are used for notify sources ISR are executed.
|
||||
* If one source ISR is executed, it will clear one bit.
|
||||
* If it clear all bits, it means to initialize this register status
|
||||
* rather than sources ISR are executed.
|
||||
*/
|
||||
if (data == 0xffffffff) {
|
||||
return;
|
||||
}
|
||||
|
||||
/* All source ISR execution are done */
|
||||
if (!s->regs[reg]) {
|
||||
trace_aspeed_intc_all_isr_done(irq);
|
||||
if (s->pending[irq]) {
|
||||
/*
|
||||
* handle pending source interrupt
|
||||
* notify firmware which source interrupt are pending
|
||||
* by setting status register
|
||||
*/
|
||||
s->regs[reg] = s->pending[irq];
|
||||
s->pending[irq] = 0;
|
||||
trace_aspeed_intc_trigger_irq(irq, s->regs[reg]);
|
||||
aspeed_intc_update(s, irq, 1);
|
||||
} else {
|
||||
/* clear irq */
|
||||
trace_aspeed_intc_clear_irq(irq, 0);
|
||||
aspeed_intc_update(s, irq, 0);
|
||||
}
|
||||
}
|
||||
aspeed_intc_status_handler(s, offset, data);
|
||||
break;
|
||||
default:
|
||||
s->regs[reg] = data;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue