|
@@ -1679,18 +1679,19 @@ static void pnv_ioda_setup_bus_dma(struct pnv_ioda_pe *pe,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-static void pnv_pci_ioda1_tce_invalidate(struct pnv_ioda_pe *pe,
|
|
|
- struct iommu_table *tbl,
|
|
|
- __be64 *startp, __be64 *endp, bool rm)
|
|
|
+static void pnv_pci_ioda1_tce_invalidate(struct iommu_table *tbl,
|
|
|
+ unsigned long index, unsigned long npages, bool rm)
|
|
|
{
|
|
|
+ struct pnv_ioda_pe *pe = tbl->data;
|
|
|
__be64 __iomem *invalidate = rm ?
|
|
|
(__be64 __iomem *)pe->tce_inval_reg_phys :
|
|
|
(__be64 __iomem *)tbl->it_index;
|
|
|
unsigned long start, end, inc;
|
|
|
const unsigned shift = tbl->it_page_shift;
|
|
|
|
|
|
- start = __pa(startp);
|
|
|
- end = __pa(endp);
|
|
|
+ start = __pa(((__be64 *)tbl->it_base) + index - tbl->it_offset);
|
|
|
+ end = __pa(((__be64 *)tbl->it_base) + index - tbl->it_offset +
|
|
|
+ npages - 1);
|
|
|
|
|
|
/* BML uses this case for p6/p7/galaxy2: Shift addr and put in node */
|
|
|
if (tbl->it_busno) {
|
|
@@ -1726,16 +1727,39 @@ static void pnv_pci_ioda1_tce_invalidate(struct pnv_ioda_pe *pe,
|
|
|
*/
|
|
|
}
|
|
|
|
|
|
+static int pnv_ioda1_tce_build(struct iommu_table *tbl, long index,
|
|
|
+ long npages, unsigned long uaddr,
|
|
|
+ enum dma_data_direction direction,
|
|
|
+ struct dma_attrs *attrs)
|
|
|
+{
|
|
|
+ int ret = pnv_tce_build(tbl, index, npages, uaddr, direction,
|
|
|
+ attrs);
|
|
|
+
|
|
|
+ if (!ret && (tbl->it_type & TCE_PCI_SWINV_CREATE))
|
|
|
+ pnv_pci_ioda1_tce_invalidate(tbl, index, npages, false);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static void pnv_ioda1_tce_free(struct iommu_table *tbl, long index,
|
|
|
+ long npages)
|
|
|
+{
|
|
|
+ pnv_tce_free(tbl, index, npages);
|
|
|
+
|
|
|
+ if (tbl->it_type & TCE_PCI_SWINV_FREE)
|
|
|
+ pnv_pci_ioda1_tce_invalidate(tbl, index, npages, false);
|
|
|
+}
|
|
|
+
|
|
|
static struct iommu_table_ops pnv_ioda1_iommu_ops = {
|
|
|
- .set = pnv_tce_build,
|
|
|
- .clear = pnv_tce_free,
|
|
|
+ .set = pnv_ioda1_tce_build,
|
|
|
+ .clear = pnv_ioda1_tce_free,
|
|
|
.get = pnv_tce_get,
|
|
|
};
|
|
|
|
|
|
-static void pnv_pci_ioda2_tce_invalidate(struct pnv_ioda_pe *pe,
|
|
|
- struct iommu_table *tbl,
|
|
|
- __be64 *startp, __be64 *endp, bool rm)
|
|
|
+static void pnv_pci_ioda2_tce_invalidate(struct iommu_table *tbl,
|
|
|
+ unsigned long index, unsigned long npages, bool rm)
|
|
|
{
|
|
|
+ struct pnv_ioda_pe *pe = tbl->data;
|
|
|
unsigned long start, end, inc;
|
|
|
__be64 __iomem *invalidate = rm ?
|
|
|
(__be64 __iomem *)pe->tce_inval_reg_phys :
|
|
@@ -1748,10 +1772,8 @@ static void pnv_pci_ioda2_tce_invalidate(struct pnv_ioda_pe *pe,
|
|
|
end = start;
|
|
|
|
|
|
/* Figure out the start, end and step */
|
|
|
- inc = tbl->it_offset + (((u64)startp - tbl->it_base) / sizeof(u64));
|
|
|
- start |= (inc << shift);
|
|
|
- inc = tbl->it_offset + (((u64)endp - tbl->it_base) / sizeof(u64));
|
|
|
- end |= (inc << shift);
|
|
|
+ start |= (index << shift);
|
|
|
+ end |= ((index + npages - 1) << shift);
|
|
|
inc = (0x1ull << shift);
|
|
|
mb();
|
|
|
|
|
@@ -1764,21 +1786,32 @@ static void pnv_pci_ioda2_tce_invalidate(struct pnv_ioda_pe *pe,
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-void pnv_pci_ioda_tce_invalidate(struct iommu_table *tbl,
|
|
|
- __be64 *startp, __be64 *endp, bool rm)
|
|
|
+static int pnv_ioda2_tce_build(struct iommu_table *tbl, long index,
|
|
|
+ long npages, unsigned long uaddr,
|
|
|
+ enum dma_data_direction direction,
|
|
|
+ struct dma_attrs *attrs)
|
|
|
{
|
|
|
- struct pnv_ioda_pe *pe = tbl->data;
|
|
|
- struct pnv_phb *phb = pe->phb;
|
|
|
+ int ret = pnv_tce_build(tbl, index, npages, uaddr, direction,
|
|
|
+ attrs);
|
|
|
|
|
|
- if (phb->type == PNV_PHB_IODA1)
|
|
|
- pnv_pci_ioda1_tce_invalidate(pe, tbl, startp, endp, rm);
|
|
|
- else
|
|
|
- pnv_pci_ioda2_tce_invalidate(pe, tbl, startp, endp, rm);
|
|
|
+ if (!ret && (tbl->it_type & TCE_PCI_SWINV_CREATE))
|
|
|
+ pnv_pci_ioda2_tce_invalidate(tbl, index, npages, false);
|
|
|
+
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+static void pnv_ioda2_tce_free(struct iommu_table *tbl, long index,
|
|
|
+ long npages)
|
|
|
+{
|
|
|
+ pnv_tce_free(tbl, index, npages);
|
|
|
+
|
|
|
+ if (tbl->it_type & TCE_PCI_SWINV_FREE)
|
|
|
+ pnv_pci_ioda2_tce_invalidate(tbl, index, npages, false);
|
|
|
}
|
|
|
|
|
|
static struct iommu_table_ops pnv_ioda2_iommu_ops = {
|
|
|
- .set = pnv_tce_build,
|
|
|
- .clear = pnv_tce_free,
|
|
|
+ .set = pnv_ioda2_tce_build,
|
|
|
+ .clear = pnv_ioda2_tce_free,
|
|
|
.get = pnv_tce_get,
|
|
|
};
|
|
|
|