mirror of https://github.com/xqemu/xqemu.git
spapr: use memory core for iommu support
Now we can stop using a "translating" DMAContext, but we do not yet modify the sPAPRTCETable users to get an AddressSpace; they keep using the table via a DMAContext. Acked-by: David Gibson <david@gibson.dropbear.id.au> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
a71bfbfe9d
commit
a84bb43669
|
@ -37,12 +37,16 @@ enum sPAPRTCEAccess {
|
||||||
};
|
};
|
||||||
|
|
||||||
struct sPAPRTCETable {
|
struct sPAPRTCETable {
|
||||||
|
/* temporary until everyone has its own AddressSpace */
|
||||||
DMAContext dma;
|
DMAContext dma;
|
||||||
|
AddressSpace as;
|
||||||
|
|
||||||
uint32_t liobn;
|
uint32_t liobn;
|
||||||
uint32_t window_size;
|
uint32_t window_size;
|
||||||
sPAPRTCE *table;
|
sPAPRTCE *table;
|
||||||
bool bypass;
|
bool bypass;
|
||||||
int fd;
|
int fd;
|
||||||
|
MemoryRegion iommu;
|
||||||
QLIST_ENTRY(sPAPRTCETable) list;
|
QLIST_ENTRY(sPAPRTCETable) list;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -68,8 +72,9 @@ static sPAPRTCETable *spapr_tce_find_by_liobn(uint32_t liobn)
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static IOMMUTLBEntry spapr_tce_translate_iommu(sPAPRTCETable *tcet, hwaddr addr)
|
static IOMMUTLBEntry spapr_tce_translate_iommu(MemoryRegion *iommu, hwaddr addr)
|
||||||
{
|
{
|
||||||
|
sPAPRTCETable *tcet = container_of(iommu, sPAPRTCETable, iommu);
|
||||||
uint64_t tce;
|
uint64_t tce;
|
||||||
|
|
||||||
#ifdef DEBUG_TCE
|
#ifdef DEBUG_TCE
|
||||||
|
@ -111,24 +116,9 @@ static IOMMUTLBEntry spapr_tce_translate_iommu(sPAPRTCETable *tcet, hwaddr addr)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
static int spapr_tce_translate(DMAContext *dma,
|
static MemoryRegionIOMMUOps spapr_iommu_ops = {
|
||||||
dma_addr_t addr,
|
.translate = spapr_tce_translate_iommu,
|
||||||
hwaddr *paddr,
|
};
|
||||||
hwaddr *len,
|
|
||||||
DMADirection dir)
|
|
||||||
{
|
|
||||||
sPAPRTCETable *tcet = DO_UPCAST(sPAPRTCETable, dma, dma);
|
|
||||||
bool is_write = (dir == DMA_DIRECTION_FROM_DEVICE);
|
|
||||||
IOMMUTLBEntry entry = spapr_tce_translate_iommu(tcet, addr);
|
|
||||||
if (!(entry.perm & (1 << is_write))) {
|
|
||||||
return -EPERM;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Translate */
|
|
||||||
*paddr = entry.translated_addr | (addr & entry.addr_mask);
|
|
||||||
*len = (addr | entry.addr_mask) - addr + 1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size)
|
sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size)
|
||||||
{
|
{
|
||||||
|
@ -145,8 +135,6 @@ sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size)
|
||||||
}
|
}
|
||||||
|
|
||||||
tcet = g_malloc0(sizeof(*tcet));
|
tcet = g_malloc0(sizeof(*tcet));
|
||||||
dma_context_init(&tcet->dma, &address_space_memory, spapr_tce_translate, NULL, NULL);
|
|
||||||
|
|
||||||
tcet->liobn = liobn;
|
tcet->liobn = liobn;
|
||||||
tcet->window_size = window_size;
|
tcet->window_size = window_size;
|
||||||
|
|
||||||
|
@ -167,6 +155,11 @@ sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size)
|
||||||
"table @ %p, fd=%d\n", tcet, liobn, tcet->table, tcet->fd);
|
"table @ %p, fd=%d\n", tcet, liobn, tcet->table, tcet->fd);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
memory_region_init_iommu(&tcet->iommu, &spapr_iommu_ops,
|
||||||
|
"iommu-spapr", UINT64_MAX);
|
||||||
|
address_space_init(&tcet->as, &tcet->iommu);
|
||||||
|
dma_context_init(&tcet->dma, &tcet->as, NULL, NULL, NULL);
|
||||||
|
|
||||||
QLIST_INSERT_HEAD(&spapr_tce_tables, tcet, list);
|
QLIST_INSERT_HEAD(&spapr_tce_tables, tcet, list);
|
||||||
|
|
||||||
return tcet;
|
return tcet;
|
||||||
|
@ -190,6 +183,11 @@ DMAContext *spapr_tce_get_dma(sPAPRTCETable *tcet)
|
||||||
return &tcet->dma;
|
return &tcet->dma;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MemoryRegion *spapr_tce_get_iommu(sPAPRTCETable *tcet)
|
||||||
|
{
|
||||||
|
return &tcet->iommu;
|
||||||
|
}
|
||||||
|
|
||||||
void spapr_tce_set_bypass(sPAPRTCETable *tcet, bool bypass)
|
void spapr_tce_set_bypass(sPAPRTCETable *tcet, bool bypass)
|
||||||
{
|
{
|
||||||
tcet->bypass = bypass;
|
tcet->bypass = bypass;
|
||||||
|
@ -208,6 +206,7 @@ static target_ulong put_tce_emu(sPAPRTCETable *tcet, target_ulong ioba,
|
||||||
target_ulong tce)
|
target_ulong tce)
|
||||||
{
|
{
|
||||||
sPAPRTCE *tcep;
|
sPAPRTCE *tcep;
|
||||||
|
IOMMUTLBEntry entry;
|
||||||
|
|
||||||
if (ioba >= tcet->window_size) {
|
if (ioba >= tcet->window_size) {
|
||||||
hcall_dprintf("spapr_vio_put_tce on out-of-bounds IOBA 0x"
|
hcall_dprintf("spapr_vio_put_tce on out-of-bounds IOBA 0x"
|
||||||
|
@ -218,6 +217,13 @@ static target_ulong put_tce_emu(sPAPRTCETable *tcet, target_ulong ioba,
|
||||||
tcep = tcet->table + (ioba >> SPAPR_TCE_PAGE_SHIFT);
|
tcep = tcet->table + (ioba >> SPAPR_TCE_PAGE_SHIFT);
|
||||||
tcep->tce = tce;
|
tcep->tce = tce;
|
||||||
|
|
||||||
|
entry.target_as = &address_space_memory,
|
||||||
|
entry.iova = ioba & ~SPAPR_TCE_PAGE_MASK;
|
||||||
|
entry.translated_addr = tce & ~SPAPR_TCE_PAGE_MASK;
|
||||||
|
entry.addr_mask = SPAPR_TCE_PAGE_MASK;
|
||||||
|
entry.perm = tce;
|
||||||
|
memory_region_notify_iommu(&tcet->iommu, entry);
|
||||||
|
|
||||||
return H_SUCCESS;
|
return H_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -349,6 +349,7 @@ void spapr_events_init(sPAPREnvironment *spapr);
|
||||||
void spapr_events_fdt_skel(void *fdt, uint32_t epow_irq);
|
void spapr_events_fdt_skel(void *fdt, uint32_t epow_irq);
|
||||||
sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size);
|
sPAPRTCETable *spapr_tce_new_table(uint32_t liobn, size_t window_size);
|
||||||
DMAContext *spapr_tce_get_dma(sPAPRTCETable *tcet);
|
DMAContext *spapr_tce_get_dma(sPAPRTCETable *tcet);
|
||||||
|
MemoryRegion *spapr_tce_get_iommu(sPAPRTCETable *tcet);
|
||||||
void spapr_tce_free(sPAPRTCETable *tcet);
|
void spapr_tce_free(sPAPRTCETable *tcet);
|
||||||
void spapr_tce_reset(sPAPRTCETable *tcet);
|
void spapr_tce_reset(sPAPRTCETable *tcet);
|
||||||
void spapr_tce_set_bypass(sPAPRTCETable *tcet, bool bypass);
|
void spapr_tce_set_bypass(sPAPRTCETable *tcet, bool bypass);
|
||||||
|
|
Loading…
Reference in New Issue