mirror of
https://github.com/FEX-Emu/linux.git
synced 2025-01-27 13:43:53 +00:00
iommu/arm-smmu: Tidy up 64-bit/atomic I/O accesses
With {read,write}q_relaxed now able to fall back to the common nonatomic-hi-lo helper, make use of that so that we don't have to open-code our own. In the process, also convert the other remaining split accesses, and repurpose the custom accessor to smooth out the couple of troublesome instances where we really want to avoid nonatomic writes (and a 64-bit access is unnecessary in the 32-bit context formats we would use on a 32-bit CPU). This paves the way for getting rid of some of the assumptions currently baked into the driver which make it really awkward to use 32-bit context formats with SMMUv2 under a 64-bit kernel. Signed-off-by: Robin Murphy <robin.murphy@arm.com> Signed-off-by: Will Deacon <will.deacon@arm.com>
This commit is contained in:
parent
e511267bc2
commit
f9a05f05b1
@ -34,6 +34,7 @@
|
|||||||
#include <linux/err.h>
|
#include <linux/err.h>
|
||||||
#include <linux/interrupt.h>
|
#include <linux/interrupt.h>
|
||||||
#include <linux/io.h>
|
#include <linux/io.h>
|
||||||
|
#include <linux/io-64-nonatomic-hi-lo.h>
|
||||||
#include <linux/iommu.h>
|
#include <linux/iommu.h>
|
||||||
#include <linux/iopoll.h>
|
#include <linux/iopoll.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
@ -71,16 +72,15 @@
|
|||||||
((smmu->options & ARM_SMMU_OPT_SECURE_CFG_ACCESS) \
|
((smmu->options & ARM_SMMU_OPT_SECURE_CFG_ACCESS) \
|
||||||
? 0x400 : 0))
|
? 0x400 : 0))
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Some 64-bit registers only make sense to write atomically, but in such
|
||||||
|
* cases all the data relevant to AArch32 formats lies within the lower word,
|
||||||
|
* therefore this actually makes more sense than it might first appear.
|
||||||
|
*/
|
||||||
#ifdef CONFIG_64BIT
|
#ifdef CONFIG_64BIT
|
||||||
#define smmu_writeq writeq_relaxed
|
#define smmu_write_atomic_lq writeq_relaxed
|
||||||
#else
|
#else
|
||||||
#define smmu_writeq(reg64, addr) \
|
#define smmu_write_atomic_lq writel_relaxed
|
||||||
do { \
|
|
||||||
u64 __val = (reg64); \
|
|
||||||
void __iomem *__addr = (addr); \
|
|
||||||
writel_relaxed(__val >> 32, __addr + 4); \
|
|
||||||
writel_relaxed(__val, __addr); \
|
|
||||||
} while (0)
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Configuration registers */
|
/* Configuration registers */
|
||||||
@ -211,11 +211,9 @@
|
|||||||
#define ARM_SMMU_CB_TTBCR 0x30
|
#define ARM_SMMU_CB_TTBCR 0x30
|
||||||
#define ARM_SMMU_CB_S1_MAIR0 0x38
|
#define ARM_SMMU_CB_S1_MAIR0 0x38
|
||||||
#define ARM_SMMU_CB_S1_MAIR1 0x3c
|
#define ARM_SMMU_CB_S1_MAIR1 0x3c
|
||||||
#define ARM_SMMU_CB_PAR_LO 0x50
|
#define ARM_SMMU_CB_PAR 0x50
|
||||||
#define ARM_SMMU_CB_PAR_HI 0x54
|
|
||||||
#define ARM_SMMU_CB_FSR 0x58
|
#define ARM_SMMU_CB_FSR 0x58
|
||||||
#define ARM_SMMU_CB_FAR_LO 0x60
|
#define ARM_SMMU_CB_FAR 0x60
|
||||||
#define ARM_SMMU_CB_FAR_HI 0x64
|
|
||||||
#define ARM_SMMU_CB_FSYNR0 0x68
|
#define ARM_SMMU_CB_FSYNR0 0x68
|
||||||
#define ARM_SMMU_CB_S1_TLBIVA 0x600
|
#define ARM_SMMU_CB_S1_TLBIVA 0x600
|
||||||
#define ARM_SMMU_CB_S1_TLBIASID 0x610
|
#define ARM_SMMU_CB_S1_TLBIASID 0x610
|
||||||
@ -645,7 +643,7 @@ static void arm_smmu_tlb_inv_range_nosync(unsigned long iova, size_t size,
|
|||||||
ARM_SMMU_CB_S2_TLBIIPAS2;
|
ARM_SMMU_CB_S2_TLBIIPAS2;
|
||||||
iova >>= 12;
|
iova >>= 12;
|
||||||
do {
|
do {
|
||||||
writeq_relaxed(iova, reg);
|
smmu_write_atomic_lq(iova, reg);
|
||||||
iova += granule >> 12;
|
iova += granule >> 12;
|
||||||
} while (size -= granule);
|
} while (size -= granule);
|
||||||
#endif
|
#endif
|
||||||
@ -664,7 +662,7 @@ static struct iommu_gather_ops arm_smmu_gather_ops = {
|
|||||||
static irqreturn_t arm_smmu_context_fault(int irq, void *dev)
|
static irqreturn_t arm_smmu_context_fault(int irq, void *dev)
|
||||||
{
|
{
|
||||||
int flags, ret;
|
int flags, ret;
|
||||||
u32 fsr, far, fsynr, resume;
|
u32 fsr, fsynr, resume;
|
||||||
unsigned long iova;
|
unsigned long iova;
|
||||||
struct iommu_domain *domain = dev;
|
struct iommu_domain *domain = dev;
|
||||||
struct arm_smmu_domain *smmu_domain = to_smmu_domain(domain);
|
struct arm_smmu_domain *smmu_domain = to_smmu_domain(domain);
|
||||||
@ -686,13 +684,7 @@ static irqreturn_t arm_smmu_context_fault(int irq, void *dev)
|
|||||||
fsynr = readl_relaxed(cb_base + ARM_SMMU_CB_FSYNR0);
|
fsynr = readl_relaxed(cb_base + ARM_SMMU_CB_FSYNR0);
|
||||||
flags = fsynr & FSYNR0_WNR ? IOMMU_FAULT_WRITE : IOMMU_FAULT_READ;
|
flags = fsynr & FSYNR0_WNR ? IOMMU_FAULT_WRITE : IOMMU_FAULT_READ;
|
||||||
|
|
||||||
far = readl_relaxed(cb_base + ARM_SMMU_CB_FAR_LO);
|
iova = readq_relaxed(cb_base + ARM_SMMU_CB_FAR);
|
||||||
iova = far;
|
|
||||||
#ifdef CONFIG_64BIT
|
|
||||||
far = readl_relaxed(cb_base + ARM_SMMU_CB_FAR_HI);
|
|
||||||
iova |= ((unsigned long)far << 32);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (!report_iommu_fault(domain, smmu->dev, iova, flags)) {
|
if (!report_iommu_fault(domain, smmu->dev, iova, flags)) {
|
||||||
ret = IRQ_HANDLED;
|
ret = IRQ_HANDLED;
|
||||||
resume = RESUME_RETRY;
|
resume = RESUME_RETRY;
|
||||||
@ -788,14 +780,14 @@ static void arm_smmu_init_context_bank(struct arm_smmu_domain *smmu_domain,
|
|||||||
reg64 = pgtbl_cfg->arm_lpae_s1_cfg.ttbr[0];
|
reg64 = pgtbl_cfg->arm_lpae_s1_cfg.ttbr[0];
|
||||||
|
|
||||||
reg64 |= ((u64)ARM_SMMU_CB_ASID(smmu, cfg)) << TTBRn_ASID_SHIFT;
|
reg64 |= ((u64)ARM_SMMU_CB_ASID(smmu, cfg)) << TTBRn_ASID_SHIFT;
|
||||||
smmu_writeq(reg64, cb_base + ARM_SMMU_CB_TTBR0);
|
writeq_relaxed(reg64, cb_base + ARM_SMMU_CB_TTBR0);
|
||||||
|
|
||||||
reg64 = pgtbl_cfg->arm_lpae_s1_cfg.ttbr[1];
|
reg64 = pgtbl_cfg->arm_lpae_s1_cfg.ttbr[1];
|
||||||
reg64 |= ((u64)ARM_SMMU_CB_ASID(smmu, cfg)) << TTBRn_ASID_SHIFT;
|
reg64 |= ((u64)ARM_SMMU_CB_ASID(smmu, cfg)) << TTBRn_ASID_SHIFT;
|
||||||
smmu_writeq(reg64, cb_base + ARM_SMMU_CB_TTBR1);
|
writeq_relaxed(reg64, cb_base + ARM_SMMU_CB_TTBR1);
|
||||||
} else {
|
} else {
|
||||||
reg64 = pgtbl_cfg->arm_lpae_s2_cfg.vttbr;
|
reg64 = pgtbl_cfg->arm_lpae_s2_cfg.vttbr;
|
||||||
smmu_writeq(reg64, cb_base + ARM_SMMU_CB_TTBR0);
|
writeq_relaxed(reg64, cb_base + ARM_SMMU_CB_TTBR0);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* TTBCR */
|
/* TTBCR */
|
||||||
@ -1263,8 +1255,8 @@ static phys_addr_t arm_smmu_iova_to_phys_hard(struct iommu_domain *domain,
|
|||||||
/* ATS1 registers can only be written atomically */
|
/* ATS1 registers can only be written atomically */
|
||||||
va = iova & ~0xfffUL;
|
va = iova & ~0xfffUL;
|
||||||
if (smmu->version == ARM_SMMU_V2)
|
if (smmu->version == ARM_SMMU_V2)
|
||||||
smmu_writeq(va, cb_base + ARM_SMMU_CB_ATS1PR);
|
smmu_write_atomic_lq(va, cb_base + ARM_SMMU_CB_ATS1PR);
|
||||||
else
|
else /* Register is only 32-bit in v1 */
|
||||||
writel_relaxed(va, cb_base + ARM_SMMU_CB_ATS1PR);
|
writel_relaxed(va, cb_base + ARM_SMMU_CB_ATS1PR);
|
||||||
|
|
||||||
if (readl_poll_timeout_atomic(cb_base + ARM_SMMU_CB_ATSR, tmp,
|
if (readl_poll_timeout_atomic(cb_base + ARM_SMMU_CB_ATSR, tmp,
|
||||||
@ -1275,9 +1267,7 @@ static phys_addr_t arm_smmu_iova_to_phys_hard(struct iommu_domain *domain,
|
|||||||
return ops->iova_to_phys(ops, iova);
|
return ops->iova_to_phys(ops, iova);
|
||||||
}
|
}
|
||||||
|
|
||||||
phys = readl_relaxed(cb_base + ARM_SMMU_CB_PAR_LO);
|
phys = readq_relaxed(cb_base + ARM_SMMU_CB_PAR);
|
||||||
phys |= ((u64)readl_relaxed(cb_base + ARM_SMMU_CB_PAR_HI)) << 32;
|
|
||||||
|
|
||||||
if (phys & CB_PAR_F) {
|
if (phys & CB_PAR_F) {
|
||||||
dev_err(dev, "translation fault!\n");
|
dev_err(dev, "translation fault!\n");
|
||||||
dev_err(dev, "PAR = 0x%llx\n", phys);
|
dev_err(dev, "PAR = 0x%llx\n", phys);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user