Commit 9ae2fdde authored by Christoph Hellwig's avatar Christoph Hellwig Committed by Michael Ellerman

powerpc/pseries: use the generic iommu bypass code

Use the generic iommu bypass code instead of overriding set_dma_mask.
Signed-off-by: default avatarChristoph Hellwig <hch@lst.de>
Signed-off-by: default avatarMichael Ellerman <mpe@ellerman.id.au>
parent cd7c11ed
...@@ -978,7 +978,7 @@ static phys_addr_t ddw_memory_hotplug_max(void) ...@@ -978,7 +978,7 @@ static phys_addr_t ddw_memory_hotplug_max(void)
* pdn: the parent pe node with the ibm,dma_window property * pdn: the parent pe node with the ibm,dma_window property
* Future: also check if we can remap the base window for our base page size * Future: also check if we can remap the base window for our base page size
* *
* returns the dma offset for use by dma_set_mask * returns the dma offset for use by the direct mapped DMA code.
*/ */
static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn) static u64 enable_ddw(struct pci_dev *dev, struct device_node *pdn)
{ {
...@@ -1198,26 +1198,17 @@ static void pci_dma_dev_setup_pSeriesLP(struct pci_dev *dev) ...@@ -1198,26 +1198,17 @@ static void pci_dma_dev_setup_pSeriesLP(struct pci_dev *dev)
iommu_add_device(pci->table_group, &dev->dev); iommu_add_device(pci->table_group, &dev->dev);
} }
static int dma_set_mask_pSeriesLP(struct device *dev, u64 dma_mask) static bool iommu_bypass_supported_pSeriesLP(struct pci_dev *pdev, u64 dma_mask)
{ {
bool ddw_enabled = false; struct device_node *dn = pci_device_to_OF_node(pdev), *pdn;
struct device_node *pdn, *dn;
struct pci_dev *pdev;
const __be32 *dma_window = NULL; const __be32 *dma_window = NULL;
u64 dma_offset; u64 dma_offset;
if (!dev->dma_mask)
return -EIO;
if (!dev_is_pci(dev))
goto check_mask;
pdev = to_pci_dev(dev);
/* only attempt to use a new window if 64-bit DMA is requested */ /* only attempt to use a new window if 64-bit DMA is requested */
if (!disable_ddw && dma_mask == DMA_BIT_MASK(64)) { if (dma_mask < DMA_BIT_MASK(64))
dn = pci_device_to_OF_node(pdev); return false;
dev_dbg(dev, "node is %pOF\n", dn);
dev_dbg(&pdev->dev, "node is %pOF\n", dn);
/* /*
* the device tree might contain the dma-window properties * the device tree might contain the dma-window properties
...@@ -1231,54 +1222,16 @@ static int dma_set_mask_pSeriesLP(struct device *dev, u64 dma_mask) ...@@ -1231,54 +1222,16 @@ static int dma_set_mask_pSeriesLP(struct device *dev, u64 dma_mask)
if (dma_window) if (dma_window)
break; break;
} }
if (pdn && PCI_DN(pdn)) { if (pdn && PCI_DN(pdn)) {
dma_offset = enable_ddw(pdev, pdn); dma_offset = enable_ddw(pdev, pdn);
if (dma_offset != 0) { if (dma_offset != 0) {
dev_info(dev, "Using 64-bit direct DMA at offset %llx\n", dma_offset); set_dma_offset(&pdev->dev, dma_offset);
set_dma_offset(dev, dma_offset); return true;
set_dma_ops(dev, &dma_nommu_ops);
ddw_enabled = true;
} }
} }
}
/* fall back on iommu ops */
if (!ddw_enabled && get_dma_ops(dev) != &dma_iommu_ops) {
dev_info(dev, "Restoring 32-bit DMA via iommu\n");
set_dma_ops(dev, &dma_iommu_ops);
}
check_mask:
if (!dma_supported(dev, dma_mask))
return -EIO;
*dev->dma_mask = dma_mask;
return 0;
}
static u64 dma_get_required_mask_pSeriesLP(struct device *dev)
{
if (!dev->dma_mask)
return 0;
if (!disable_ddw && dev_is_pci(dev)) {
struct pci_dev *pdev = to_pci_dev(dev);
struct device_node *dn;
dn = pci_device_to_OF_node(pdev);
/* search upwards for ibm,dma-window */
for (; dn && PCI_DN(dn) && !PCI_DN(dn)->table_group;
dn = dn->parent)
if (of_get_property(dn, "ibm,dma-window", NULL))
break;
/* if there is a ibm,ddw-applicable property require 64 bits */
if (dn && PCI_DN(dn) &&
of_get_property(dn, "ibm,ddw-applicable", NULL))
return DMA_BIT_MASK(64);
}
return dma_iommu_get_required_mask(dev); return false;
} }
static int iommu_mem_notifier(struct notifier_block *nb, unsigned long action, static int iommu_mem_notifier(struct notifier_block *nb, unsigned long action,
...@@ -1373,8 +1326,9 @@ void iommu_init_early_pSeries(void) ...@@ -1373,8 +1326,9 @@ void iommu_init_early_pSeries(void)
if (firmware_has_feature(FW_FEATURE_LPAR)) { if (firmware_has_feature(FW_FEATURE_LPAR)) {
pseries_pci_controller_ops.dma_bus_setup = pci_dma_bus_setup_pSeriesLP; pseries_pci_controller_ops.dma_bus_setup = pci_dma_bus_setup_pSeriesLP;
pseries_pci_controller_ops.dma_dev_setup = pci_dma_dev_setup_pSeriesLP; pseries_pci_controller_ops.dma_dev_setup = pci_dma_dev_setup_pSeriesLP;
ppc_md.dma_set_mask = dma_set_mask_pSeriesLP; if (!disable_ddw)
ppc_md.dma_get_required_mask = dma_get_required_mask_pSeriesLP; pseries_pci_controller_ops.iommu_bypass_supported =
iommu_bypass_supported_pSeriesLP;
} else { } else {
pseries_pci_controller_ops.dma_bus_setup = pci_dma_bus_setup_pSeries; pseries_pci_controller_ops.dma_bus_setup = pci_dma_bus_setup_pSeries;
pseries_pci_controller_ops.dma_dev_setup = pci_dma_dev_setup_pSeries; pseries_pci_controller_ops.dma_dev_setup = pci_dma_dev_setup_pSeries;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment