mirror of
https://github.com/qemu/qemu.git
synced 2024-11-28 22:33:36 +08:00
spapr_iommu: Realloc guest visible TCE table when hot(un)plugging vfio-pci
This replaces g_malloc() with spapr_tce_alloc_table() as this is the standard way of allocating tables and this allows moving the table back to KVM when unplugging a VFIO PCI device and VFIO TCE acceleration support is not present in the KVM. Although spapr_tce_alloc_table() is expected to fail with EBUSY if called when previous fd is not closed yet, in practice we will not see it because cap_spapr_vfio is false at the moment. Signed-off-by: Alexey Kardashevskiy <aik@ozlabs.ru> Signed-off-by: David Gibson <david@gibson.dropbear.id.au>
This commit is contained in:
parent
54a50dae93
commit
f5509b6beb
@ -275,33 +275,26 @@ static void spapr_tce_table_realize(DeviceState *dev, Error **errp)
|
|||||||
void spapr_tce_set_need_vfio(sPAPRTCETable *tcet, bool need_vfio)
|
void spapr_tce_set_need_vfio(sPAPRTCETable *tcet, bool need_vfio)
|
||||||
{
|
{
|
||||||
size_t table_size = tcet->nb_table * sizeof(uint64_t);
|
size_t table_size = tcet->nb_table * sizeof(uint64_t);
|
||||||
void *newtable;
|
uint64_t *oldtable;
|
||||||
|
int newfd = -1;
|
||||||
|
|
||||||
if (need_vfio == tcet->need_vfio) {
|
g_assert(need_vfio != tcet->need_vfio);
|
||||||
/* Nothing to do */
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!need_vfio) {
|
tcet->need_vfio = need_vfio;
|
||||||
/* FIXME: We don't support transition back to KVM accelerated
|
|
||||||
* TCEs yet */
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
tcet->need_vfio = true;
|
oldtable = tcet->table;
|
||||||
|
|
||||||
if (tcet->fd < 0) {
|
tcet->table = spapr_tce_alloc_table(tcet->liobn,
|
||||||
/* Table is already in userspace, nothing to be do */
|
tcet->page_shift,
|
||||||
return;
|
tcet->bus_offset,
|
||||||
}
|
tcet->nb_table,
|
||||||
|
&newfd,
|
||||||
|
need_vfio);
|
||||||
|
memcpy(tcet->table, oldtable, table_size);
|
||||||
|
|
||||||
newtable = g_malloc(table_size);
|
spapr_tce_free_table(oldtable, tcet->fd, tcet->nb_table);
|
||||||
memcpy(newtable, tcet->table, table_size);
|
|
||||||
|
|
||||||
kvmppc_remove_spapr_tce(tcet->table, tcet->fd, tcet->nb_table);
|
tcet->fd = newfd;
|
||||||
|
|
||||||
tcet->fd = -1;
|
|
||||||
tcet->table = newtable;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sPAPRTCETable *spapr_tce_new_table(DeviceState *owner, uint32_t liobn)
|
sPAPRTCETable *spapr_tce_new_table(DeviceState *owner, uint32_t liobn)
|
||||||
|
Loading…
Reference in New Issue
Block a user