提交 227014b3 编写于 作者: B Benjamin Gaignard 提交者: Joerg Roedel

iommu: rockchip: Add internal ops to handle variants

Add internal ops to be able to handle incoming variant v2.
The goal is to keep the overall structure of the framework but
to allow to add the evolution of this hardware block.

The ops are global for a SoC because iommu domains are not
attached to a specific devices if they are for a virtuel device like
drm. Use a global variable shouldn't be since SoC usually doesn't
embedded different versions of the iommu hardware block.
If that happen one day a WARN_ON will be displayed at probe time.
Signed-off-by: NBenjamin Gaignard <benjamin.gaignard@collabora.com>
Reviewed-by: NHeiko Stuebner <heiko@sntech.de>
Link: https://lore.kernel.org/r/20210604164441.798362-4-benjamin.gaignard@collabora.comSigned-off-by: NJoerg Roedel <jroedel@suse.de>
上级 9e6f3cd5
...@@ -96,6 +96,15 @@ static const char * const rk_iommu_clocks[] = { ...@@ -96,6 +96,15 @@ static const char * const rk_iommu_clocks[] = {
"aclk", "iface", "aclk", "iface",
}; };
struct rk_iommu_ops {
phys_addr_t (*pt_address)(u32 dte);
u32 (*mk_dtentries)(dma_addr_t pt_dma);
u32 (*mk_ptentries)(phys_addr_t page, int prot);
phys_addr_t (*dte_addr_phys)(u32 addr);
u32 (*dma_addr_dte)(dma_addr_t dt_dma);
u64 dma_bit_mask;
};
struct rk_iommu { struct rk_iommu {
struct device *dev; struct device *dev;
void __iomem **bases; void __iomem **bases;
...@@ -116,6 +125,7 @@ struct rk_iommudata { ...@@ -116,6 +125,7 @@ struct rk_iommudata {
}; };
static struct device *dma_dev; static struct device *dma_dev;
static const struct rk_iommu_ops *rk_ops;
static inline void rk_table_flush(struct rk_iommu_domain *dom, dma_addr_t dma, static inline void rk_table_flush(struct rk_iommu_domain *dom, dma_addr_t dma,
unsigned int count) unsigned int count)
...@@ -215,11 +225,6 @@ static inline u32 rk_mk_dte(dma_addr_t pt_dma) ...@@ -215,11 +225,6 @@ static inline u32 rk_mk_dte(dma_addr_t pt_dma)
#define RK_PTE_PAGE_READABLE BIT(1) #define RK_PTE_PAGE_READABLE BIT(1)
#define RK_PTE_PAGE_VALID BIT(0) #define RK_PTE_PAGE_VALID BIT(0)
static inline phys_addr_t rk_pte_page_address(u32 pte)
{
return (phys_addr_t)pte & RK_PTE_PAGE_ADDRESS_MASK;
}
static inline bool rk_pte_is_page_valid(u32 pte) static inline bool rk_pte_is_page_valid(u32 pte)
{ {
return pte & RK_PTE_PAGE_VALID; return pte & RK_PTE_PAGE_VALID;
...@@ -448,10 +453,10 @@ static int rk_iommu_force_reset(struct rk_iommu *iommu) ...@@ -448,10 +453,10 @@ static int rk_iommu_force_reset(struct rk_iommu *iommu)
* and verifying that upper 5 nybbles are read back. * and verifying that upper 5 nybbles are read back.
*/ */
for (i = 0; i < iommu->num_mmu; i++) { for (i = 0; i < iommu->num_mmu; i++) {
rk_iommu_write(iommu->bases[i], RK_MMU_DTE_ADDR, DTE_ADDR_DUMMY); dte_addr = rk_ops->pt_address(DTE_ADDR_DUMMY);
rk_iommu_write(iommu->bases[i], RK_MMU_DTE_ADDR, dte_addr);
dte_addr = rk_iommu_read(iommu->bases[i], RK_MMU_DTE_ADDR); if (dte_addr != rk_iommu_read(iommu->bases[i], RK_MMU_DTE_ADDR)) {
if (dte_addr != (DTE_ADDR_DUMMY & RK_DTE_PT_ADDRESS_MASK)) {
dev_err(iommu->dev, "Error during raw reset. MMU_DTE_ADDR is not functioning\n"); dev_err(iommu->dev, "Error during raw reset. MMU_DTE_ADDR is not functioning\n");
return -EFAULT; return -EFAULT;
} }
...@@ -470,6 +475,16 @@ static int rk_iommu_force_reset(struct rk_iommu *iommu) ...@@ -470,6 +475,16 @@ static int rk_iommu_force_reset(struct rk_iommu *iommu)
return 0; return 0;
} }
static inline phys_addr_t rk_dte_addr_phys(u32 addr)
{
return (phys_addr_t)addr;
}
static inline u32 rk_dma_addr_dte(dma_addr_t dt_dma)
{
return dt_dma;
}
static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova) static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova)
{ {
void __iomem *base = iommu->bases[index]; void __iomem *base = iommu->bases[index];
...@@ -489,7 +504,7 @@ static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova) ...@@ -489,7 +504,7 @@ static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova)
page_offset = rk_iova_page_offset(iova); page_offset = rk_iova_page_offset(iova);
mmu_dte_addr = rk_iommu_read(base, RK_MMU_DTE_ADDR); mmu_dte_addr = rk_iommu_read(base, RK_MMU_DTE_ADDR);
mmu_dte_addr_phys = (phys_addr_t)mmu_dte_addr; mmu_dte_addr_phys = rk_ops->dte_addr_phys(mmu_dte_addr);
dte_addr_phys = mmu_dte_addr_phys + (4 * dte_index); dte_addr_phys = mmu_dte_addr_phys + (4 * dte_index);
dte_addr = phys_to_virt(dte_addr_phys); dte_addr = phys_to_virt(dte_addr_phys);
...@@ -498,14 +513,14 @@ static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova) ...@@ -498,14 +513,14 @@ static void log_iova(struct rk_iommu *iommu, int index, dma_addr_t iova)
if (!rk_dte_is_pt_valid(dte)) if (!rk_dte_is_pt_valid(dte))
goto print_it; goto print_it;
pte_addr_phys = rk_dte_pt_address(dte) + (pte_index * 4); pte_addr_phys = rk_ops->pt_address(dte) + (pte_index * 4);
pte_addr = phys_to_virt(pte_addr_phys); pte_addr = phys_to_virt(pte_addr_phys);
pte = *pte_addr; pte = *pte_addr;
if (!rk_pte_is_page_valid(pte)) if (!rk_pte_is_page_valid(pte))
goto print_it; goto print_it;
page_addr_phys = rk_pte_page_address(pte) + page_offset; page_addr_phys = rk_ops->pt_address(pte) + page_offset;
page_flags = pte & RK_PTE_PAGE_FLAGS_MASK; page_flags = pte & RK_PTE_PAGE_FLAGS_MASK;
print_it: print_it:
...@@ -601,13 +616,13 @@ static phys_addr_t rk_iommu_iova_to_phys(struct iommu_domain *domain, ...@@ -601,13 +616,13 @@ static phys_addr_t rk_iommu_iova_to_phys(struct iommu_domain *domain,
if (!rk_dte_is_pt_valid(dte)) if (!rk_dte_is_pt_valid(dte))
goto out; goto out;
pt_phys = rk_dte_pt_address(dte); pt_phys = rk_ops->pt_address(dte);
page_table = (u32 *)phys_to_virt(pt_phys); page_table = (u32 *)phys_to_virt(pt_phys);
pte = page_table[rk_iova_pte_index(iova)]; pte = page_table[rk_iova_pte_index(iova)];
if (!rk_pte_is_page_valid(pte)) if (!rk_pte_is_page_valid(pte))
goto out; goto out;
phys = rk_pte_page_address(pte) + rk_iova_page_offset(iova); phys = rk_ops->pt_address(pte) + rk_iova_page_offset(iova);
out: out:
spin_unlock_irqrestore(&rk_domain->dt_lock, flags); spin_unlock_irqrestore(&rk_domain->dt_lock, flags);
...@@ -679,13 +694,13 @@ static u32 *rk_dte_get_page_table(struct rk_iommu_domain *rk_domain, ...@@ -679,13 +694,13 @@ static u32 *rk_dte_get_page_table(struct rk_iommu_domain *rk_domain,
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
} }
dte = rk_mk_dte(pt_dma); dte = rk_ops->mk_dtentries(pt_dma);
*dte_addr = dte; *dte_addr = dte;
rk_table_flush(rk_domain, rk_table_flush(rk_domain,
rk_domain->dt_dma + dte_index * sizeof(u32), 1); rk_domain->dt_dma + dte_index * sizeof(u32), 1);
done: done:
pt_phys = rk_dte_pt_address(dte); pt_phys = rk_ops->pt_address(dte);
return (u32 *)phys_to_virt(pt_phys); return (u32 *)phys_to_virt(pt_phys);
} }
...@@ -727,7 +742,7 @@ static int rk_iommu_map_iova(struct rk_iommu_domain *rk_domain, u32 *pte_addr, ...@@ -727,7 +742,7 @@ static int rk_iommu_map_iova(struct rk_iommu_domain *rk_domain, u32 *pte_addr,
if (rk_pte_is_page_valid(pte)) if (rk_pte_is_page_valid(pte))
goto unwind; goto unwind;
pte_addr[pte_count] = rk_mk_pte(paddr, prot); pte_addr[pte_count] = rk_ops->mk_ptentries(paddr, prot);
paddr += SPAGE_SIZE; paddr += SPAGE_SIZE;
} }
...@@ -749,7 +764,7 @@ static int rk_iommu_map_iova(struct rk_iommu_domain *rk_domain, u32 *pte_addr, ...@@ -749,7 +764,7 @@ static int rk_iommu_map_iova(struct rk_iommu_domain *rk_domain, u32 *pte_addr,
pte_count * SPAGE_SIZE); pte_count * SPAGE_SIZE);
iova += pte_count * SPAGE_SIZE; iova += pte_count * SPAGE_SIZE;
page_phys = rk_pte_page_address(pte_addr[pte_count]); page_phys = rk_ops->pt_address(pte_addr[pte_count]);
pr_err("iova: %pad already mapped to %pa cannot remap to phys: %pa prot: %#x\n", pr_err("iova: %pad already mapped to %pa cannot remap to phys: %pa prot: %#x\n",
&iova, &page_phys, &paddr, prot); &iova, &page_phys, &paddr, prot);
...@@ -784,7 +799,8 @@ static int rk_iommu_map(struct iommu_domain *domain, unsigned long _iova, ...@@ -784,7 +799,8 @@ static int rk_iommu_map(struct iommu_domain *domain, unsigned long _iova,
dte_index = rk_domain->dt[rk_iova_dte_index(iova)]; dte_index = rk_domain->dt[rk_iova_dte_index(iova)];
pte_index = rk_iova_pte_index(iova); pte_index = rk_iova_pte_index(iova);
pte_addr = &page_table[pte_index]; pte_addr = &page_table[pte_index];
pte_dma = rk_dte_pt_address(dte_index) + pte_index * sizeof(u32);
pte_dma = rk_ops->pt_address(dte_index) + pte_index * sizeof(u32);
ret = rk_iommu_map_iova(rk_domain, pte_addr, pte_dma, iova, ret = rk_iommu_map_iova(rk_domain, pte_addr, pte_dma, iova,
paddr, size, prot); paddr, size, prot);
...@@ -820,7 +836,7 @@ static size_t rk_iommu_unmap(struct iommu_domain *domain, unsigned long _iova, ...@@ -820,7 +836,7 @@ static size_t rk_iommu_unmap(struct iommu_domain *domain, unsigned long _iova,
return 0; return 0;
} }
pt_phys = rk_dte_pt_address(dte); pt_phys = rk_ops->pt_address(dte);
pte_addr = (u32 *)phys_to_virt(pt_phys) + rk_iova_pte_index(iova); pte_addr = (u32 *)phys_to_virt(pt_phys) + rk_iova_pte_index(iova);
pte_dma = pt_phys + rk_iova_pte_index(iova) * sizeof(u32); pte_dma = pt_phys + rk_iova_pte_index(iova) * sizeof(u32);
unmap_size = rk_iommu_unmap_iova(rk_domain, pte_addr, pte_dma, size); unmap_size = rk_iommu_unmap_iova(rk_domain, pte_addr, pte_dma, size);
...@@ -878,7 +894,7 @@ static int rk_iommu_enable(struct rk_iommu *iommu) ...@@ -878,7 +894,7 @@ static int rk_iommu_enable(struct rk_iommu *iommu)
for (i = 0; i < iommu->num_mmu; i++) { for (i = 0; i < iommu->num_mmu; i++) {
rk_iommu_write(iommu->bases[i], RK_MMU_DTE_ADDR, rk_iommu_write(iommu->bases[i], RK_MMU_DTE_ADDR,
rk_domain->dt_dma); rk_ops->dma_addr_dte(rk_domain->dt_dma));
rk_iommu_base_command(iommu->bases[i], RK_MMU_CMD_ZAP_CACHE); rk_iommu_base_command(iommu->bases[i], RK_MMU_CMD_ZAP_CACHE);
rk_iommu_write(iommu->bases[i], RK_MMU_INT_MASK, RK_MMU_IRQ_MASK); rk_iommu_write(iommu->bases[i], RK_MMU_INT_MASK, RK_MMU_IRQ_MASK);
} }
...@@ -1034,7 +1050,7 @@ static void rk_iommu_domain_free(struct iommu_domain *domain) ...@@ -1034,7 +1050,7 @@ static void rk_iommu_domain_free(struct iommu_domain *domain)
for (i = 0; i < NUM_DT_ENTRIES; i++) { for (i = 0; i < NUM_DT_ENTRIES; i++) {
u32 dte = rk_domain->dt[i]; u32 dte = rk_domain->dt[i];
if (rk_dte_is_pt_valid(dte)) { if (rk_dte_is_pt_valid(dte)) {
phys_addr_t pt_phys = rk_dte_pt_address(dte); phys_addr_t pt_phys = rk_ops->pt_address(dte);
u32 *page_table = phys_to_virt(pt_phys); u32 *page_table = phys_to_virt(pt_phys);
dma_unmap_single(dma_dev, pt_phys, dma_unmap_single(dma_dev, pt_phys,
SPAGE_SIZE, DMA_TO_DEVICE); SPAGE_SIZE, DMA_TO_DEVICE);
...@@ -1124,6 +1140,7 @@ static int rk_iommu_probe(struct platform_device *pdev) ...@@ -1124,6 +1140,7 @@ static int rk_iommu_probe(struct platform_device *pdev)
struct device *dev = &pdev->dev; struct device *dev = &pdev->dev;
struct rk_iommu *iommu; struct rk_iommu *iommu;
struct resource *res; struct resource *res;
const struct rk_iommu_ops *ops;
int num_res = pdev->num_resources; int num_res = pdev->num_resources;
int err, i; int err, i;
...@@ -1135,6 +1152,17 @@ static int rk_iommu_probe(struct platform_device *pdev) ...@@ -1135,6 +1152,17 @@ static int rk_iommu_probe(struct platform_device *pdev)
iommu->dev = dev; iommu->dev = dev;
iommu->num_mmu = 0; iommu->num_mmu = 0;
ops = of_device_get_match_data(dev);
if (!rk_ops)
rk_ops = ops;
/*
* That should not happen unless different versions of the
* hardware block are embedded the same SoC
*/
if (WARN_ON(rk_ops != ops))
return -EINVAL;
iommu->bases = devm_kcalloc(dev, num_res, sizeof(*iommu->bases), iommu->bases = devm_kcalloc(dev, num_res, sizeof(*iommu->bases),
GFP_KERNEL); GFP_KERNEL);
if (!iommu->bases) if (!iommu->bases)
...@@ -1223,6 +1251,8 @@ static int rk_iommu_probe(struct platform_device *pdev) ...@@ -1223,6 +1251,8 @@ static int rk_iommu_probe(struct platform_device *pdev)
} }
} }
dma_set_mask_and_coherent(dev, rk_ops->dma_bit_mask);
return 0; return 0;
err_remove_sysfs: err_remove_sysfs:
iommu_device_sysfs_remove(&iommu->iommu); iommu_device_sysfs_remove(&iommu->iommu);
...@@ -1274,8 +1304,20 @@ static const struct dev_pm_ops rk_iommu_pm_ops = { ...@@ -1274,8 +1304,20 @@ static const struct dev_pm_ops rk_iommu_pm_ops = {
pm_runtime_force_resume) pm_runtime_force_resume)
}; };
static struct rk_iommu_ops iommu_data_ops_v1 = {
.pt_address = &rk_dte_pt_address,
.mk_dtentries = &rk_mk_dte,
.mk_ptentries = &rk_mk_pte,
.dte_addr_phys = &rk_dte_addr_phys,
.dma_addr_dte = &rk_dma_addr_dte,
.dma_bit_mask = DMA_BIT_MASK(32),
};
static const struct of_device_id rk_iommu_dt_ids[] = { static const struct of_device_id rk_iommu_dt_ids[] = {
{ .compatible = "rockchip,iommu" }, { .compatible = "rockchip,iommu",
.data = &iommu_data_ops_v1,
},
{ /* sentinel */ } { /* sentinel */ }
}; };
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册