msm: iommu: Remove dependency on IDR
authorStepan Moskovchenko <stepanm@codeaurora.org>
Fri, 25 Feb 2011 02:00:42 +0000 (18:00 -0800)
committerDavid Brown <davidb@codeaurora.org>
Tue, 8 Mar 2011 22:40:59 +0000 (14:40 -0800)
Remove the depencency on the IOMMU IDR register, as it may
not be accessible depending on the security configuraton.
This involves moving the NCB field of IDR into the platform
data.

Signed-off-by: Stepan Moskovchenko <stepanm@codeaurora.org>
Signed-off-by: David Brown <davidb@codeaurora.org>
arch/arm/mach-msm/devices-iommu.c
arch/arm/mach-msm/include/mach/iommu.h
arch/arm/mach-msm/iommu.c
arch/arm/mach-msm/iommu_dev.c

index af97afe0bfab46c59895545d8b1866389a3f54bb..24030d0da6e3c59ec9ddfa2a76995001f747bc62 100644 (file)
@@ -280,50 +280,62 @@ static struct platform_device msm_root_iommu_dev = {
 
 static struct msm_iommu_dev jpegd_iommu = {
        .name = "jpegd",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev vpe_iommu = {
-       .name = "vpe"
+       .name = "vpe",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev mdp0_iommu = {
-       .name = "mdp0"
+       .name = "mdp0",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev mdp1_iommu = {
-       .name = "mdp1"
+       .name = "mdp1",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev rot_iommu = {
-       .name = "rot"
+       .name = "rot",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev ijpeg_iommu = {
-       .name = "ijpeg"
+       .name = "ijpeg",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev vfe_iommu = {
        .name = "vfe",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev vcodec_a_iommu = {
-       .name = "vcodec_a"
+       .name = "vcodec_a",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev vcodec_b_iommu = {
-       .name = "vcodec_b"
+       .name = "vcodec_b",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev gfx3d_iommu = {
        .name = "gfx3d",
+       .ncb = 3,
 };
 
 static struct msm_iommu_dev gfx2d0_iommu = {
        .name = "gfx2d0",
+       .ncb = 2,
 };
 
 static struct msm_iommu_dev gfx2d1_iommu = {
        .name = "gfx2d1",
+       .ncb = 2,
 };
 
 static struct platform_device msm_device_iommu_jpegd = {
index 4dfe7efcf4eafaa2d5889efbbffcde7e4c5115aa..5c7c955e6d253a987c4a16d9c28f0c11e7c01771 100644 (file)
 /**
  * struct msm_iommu_dev - a single IOMMU hardware instance
  * name                Human-readable name given to this IOMMU HW instance
+ * ncb         Number of context banks present on this IOMMU HW instance
  */
 struct msm_iommu_dev {
        const char *name;
+       int ncb;
 };
 
 /**
@@ -69,6 +71,7 @@ struct msm_iommu_ctx_dev {
 /**
  * struct msm_iommu_drvdata - A single IOMMU hardware instance
  * @base:      IOMMU config port base address (VA)
+ * @ncb                The number of contexts on this IOMMU
  * @irq:       Interrupt number
  * @clk:       The bus clock for this IOMMU hardware instance
  * @pclk:      The clock for the IOMMU bus interconnect
@@ -79,6 +82,7 @@ struct msm_iommu_ctx_dev {
 struct msm_iommu_drvdata {
        void __iomem *base;
        int irq;
+       int ncb;
        struct clk *clk;
        struct clk *pclk;
 };
index 9c087405c6359cb1353398253c4b9dab4a043fd2..0146f519e85cb6d2a73f6547bdfa5bd5d5c5d1fa 100644 (file)
@@ -636,7 +636,7 @@ irqreturn_t msm_iommu_fault_handler(int irq, void *dev_id)
        struct msm_iommu_drvdata *drvdata = dev_id;
        void __iomem *base;
        unsigned int fsr;
-       int ncb, i, ret;
+       int i, ret;
 
        spin_lock(&msm_iommu_lock);
 
@@ -654,8 +654,7 @@ irqreturn_t msm_iommu_fault_handler(int irq, void *dev_id)
        if (ret)
                goto fail;
 
-       ncb = GET_NCB(base)+1;
-       for (i = 0; i < ncb; i++) {
+       for (i = 0; i < drvdata->ncb; i++) {
                fsr = GET_FSR(base, i);
                if (fsr) {
                        pr_err("Fault occurred in context %d.\n", i);
index 0e240c9d6e71aeca8198f5e580c791e1137352f9..8e8fb079852d22ca124a5c0ff7bbce9d3f5ec365 100644 (file)
@@ -85,9 +85,9 @@ fail:
 }
 EXPORT_SYMBOL(msm_iommu_get_ctx);
 
-static void msm_iommu_reset(void __iomem *base)
+static void msm_iommu_reset(void __iomem *base, int ncb)
 {
-       int ctx, ncb;
+       int ctx;
 
        SET_RPUE(base, 0);
        SET_RPUEIE(base, 0);
@@ -100,7 +100,6 @@ static void msm_iommu_reset(void __iomem *base)
        SET_GLOBAL_TLBIALL(base, 0);
        SET_RPU_ACR(base, 0);
        SET_TLBLKCRWE(base, 1);
-       ncb = GET_NCB(base)+1;
 
        for (ctx = 0; ctx < ncb; ctx++) {
                SET_BPRCOSH(base, ctx, 0);
@@ -136,7 +135,7 @@ static int msm_iommu_probe(struct platform_device *pdev)
        struct msm_iommu_dev *iommu_dev = pdev->dev.platform_data;
        void __iomem *regs_base;
        resource_size_t len;
-       int ret, ncb, nm2v, irq;
+       int ret, irq, par;
 
        if (pdev->id == -1) {
                msm_iommu_root_dev = pdev;
@@ -211,10 +210,18 @@ static int msm_iommu_probe(struct platform_device *pdev)
                goto fail_io;
        }
 
-       mb();
+       msm_iommu_reset(regs_base, iommu_dev->ncb);
 
-       if (GET_IDR(regs_base) == 0) {
-               pr_err("Invalid IDR value detected\n");
+       SET_M(regs_base, 0, 1);
+       SET_PAR(regs_base, 0, 0);
+       SET_V2PCFG(regs_base, 0, 1);
+       SET_V2PPR(regs_base, 0, 0);
+       par = GET_PAR(regs_base, 0);
+       SET_V2PCFG(regs_base, 0, 0);
+       SET_M(regs_base, 0, 0);
+
+       if (!par) {
+               pr_err("%s: Invalid PAR value detected\n", iommu_dev->name);
                ret = -ENODEV;
                goto fail_io;
        }
@@ -226,17 +233,15 @@ static int msm_iommu_probe(struct platform_device *pdev)
                goto fail_io;
        }
 
-       msm_iommu_reset(regs_base);
+
        drvdata->pclk = iommu_pclk;
        drvdata->clk = iommu_clk;
        drvdata->base = regs_base;
        drvdata->irq = irq;
-
-       nm2v = GET_NM2VCBMT((unsigned long) regs_base);
-       ncb = GET_NCB((unsigned long) regs_base);
+       drvdata->ncb = iommu_dev->ncb;
 
        pr_info("device %s mapped at %p, irq %d with %d ctx banks\n",
-                       iommu_dev->name, regs_base, irq, ncb+1);
+               iommu_dev->name, regs_base, irq, iommu_dev->ncb);
 
        platform_set_drvdata(pdev, drvdata);