mirror of https://github.com/xemu-project/xemu.git
spapr_pci: fix irq leak in RTAS ibm,change-msi
This RTAS call is used to request new interrupts or to free all interrupts. If the driver has already allocated interrupts and asks again for a non-null number of irqs, then the rtas_ibm_change_msi() function will silently leak the previous interrupts. It happens because xics_free() is only called when the driver releases all interrupts (!req_num case). Note that the previously allocated spapr_pci_msi is not leaked because the GHashTable is created with destroy functions and g_hash_table_insert() hence frees the old value. This patch makes sure any previously allocated MSIs are released when a new allocation succeeds. Signed-off-by: Greg Kurz <gkurz@linux.vnet.ibm.com> Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
This commit is contained in:
parent
d4a63ac8b1
commit
ce266b75fe
|
@ -305,9 +305,10 @@ static void rtas_ibm_change_msi(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
msi = (spapr_pci_msi *) g_hash_table_lookup(phb->msi, &config_addr);
|
||||||
|
|
||||||
/* Releasing MSIs */
|
/* Releasing MSIs */
|
||||||
if (!req_num) {
|
if (!req_num) {
|
||||||
msi = (spapr_pci_msi *) g_hash_table_lookup(phb->msi, &config_addr);
|
|
||||||
if (!msi) {
|
if (!msi) {
|
||||||
trace_spapr_pci_msi("Releasing wrong config", config_addr);
|
trace_spapr_pci_msi("Releasing wrong config", config_addr);
|
||||||
rtas_st(rets, 0, RTAS_OUT_HW_ERROR);
|
rtas_st(rets, 0, RTAS_OUT_HW_ERROR);
|
||||||
|
@ -360,6 +361,12 @@ static void rtas_ibm_change_msi(PowerPCCPU *cpu, sPAPRMachineState *spapr,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Release previous MSIs */
|
||||||
|
if (msi) {
|
||||||
|
xics_free(spapr->icp, msi->first_irq, msi->num);
|
||||||
|
g_hash_table_remove(phb->msi, &config_addr);
|
||||||
|
}
|
||||||
|
|
||||||
/* Setup MSI/MSIX vectors in the device (via cfgspace or MSIX BAR) */
|
/* Setup MSI/MSIX vectors in the device (via cfgspace or MSIX BAR) */
|
||||||
spapr_msi_setmsg(pdev, SPAPR_PCI_MSI_WINDOW, ret_intr_type == RTAS_TYPE_MSIX,
|
spapr_msi_setmsg(pdev, SPAPR_PCI_MSI_WINDOW, ret_intr_type == RTAS_TYPE_MSIX,
|
||||||
irq, req_num);
|
irq, req_num);
|
||||||
|
|
Loading…
Reference in New Issue