#include <linux/of_iommu.h>
 #include <linux/pci.h>
 #include <linux/platform_device.h>
+#include <linux/pm_runtime.h>
 #include <linux/slab.h>
 #include <linux/spinlock.h>
 
        u32                             num_global_irqs;
        u32                             num_context_irqs;
        unsigned int                    *irqs;
+       struct clk_bulk_data            *clks;
+       int                             num_clks;
 
        u32                             cavium_id_base; /* Specific to Cavium */
 
 };
 
 #define ARM_SMMU_MATCH_DATA(name, ver, imp)    \
-static struct arm_smmu_match_data name = { .version = ver, .model = imp }
+static const struct arm_smmu_match_data name = { .version = ver, .model = imp }
 
 ARM_SMMU_MATCH_DATA(smmu_generic_v1, ARM_SMMU_V1, GENERIC_SMMU);
 ARM_SMMU_MATCH_DATA(smmu_generic_v2, ARM_SMMU_V2, GENERIC_SMMU);
                smmu->irqs[i] = irq;
        }
 
+       err = devm_clk_bulk_get_all(dev, &smmu->clks);
+       if (err < 0) {
+               dev_err(dev, "failed to get clocks %d\n", err);
+               return err;
+       }
+       smmu->num_clks = err;
+
+       err = clk_bulk_prepare_enable(smmu->num_clks, smmu->clks);
+       if (err)
+               return err;
+
        err = arm_smmu_device_cfg_probe(smmu);
        if (err)
                return err;
 
        /* Turn the thing off */
        writel(sCR0_CLIENTPD, ARM_SMMU_GR0_NS(smmu) + ARM_SMMU_GR0_sCR0);
+
+       clk_bulk_disable_unprepare(smmu->num_clks, smmu->clks);
+
        return 0;
 }
 
        arm_smmu_device_remove(pdev);
 }
 
-static int __maybe_unused arm_smmu_pm_resume(struct device *dev)
+static int __maybe_unused arm_smmu_runtime_resume(struct device *dev)
 {
        struct arm_smmu_device *smmu = dev_get_drvdata(dev);
+       int ret;
+
+       ret = clk_bulk_enable(smmu->num_clks, smmu->clks);
+       if (ret)
+               return ret;
 
        arm_smmu_device_reset(smmu);
+
        return 0;
 }
 
-static SIMPLE_DEV_PM_OPS(arm_smmu_pm_ops, NULL, arm_smmu_pm_resume);
+static int __maybe_unused arm_smmu_runtime_suspend(struct device *dev)
+{
+       struct arm_smmu_device *smmu = dev_get_drvdata(dev);
+
+       clk_bulk_disable(smmu->num_clks, smmu->clks);
+
+       return 0;
+}
+
+static int __maybe_unused arm_smmu_pm_resume(struct device *dev)
+{
+       if (pm_runtime_suspended(dev))
+               return 0;
+
+       return arm_smmu_runtime_resume(dev);
+}
+
+static int __maybe_unused arm_smmu_pm_suspend(struct device *dev)
+{
+       if (pm_runtime_suspended(dev))
+               return 0;
+
+       return arm_smmu_runtime_suspend(dev);
+}
+
+static const struct dev_pm_ops arm_smmu_pm_ops = {
+       SET_SYSTEM_SLEEP_PM_OPS(arm_smmu_pm_suspend, arm_smmu_pm_resume)
+       SET_RUNTIME_PM_OPS(arm_smmu_runtime_suspend,
+                          arm_smmu_runtime_resume, NULL)
+};
 
 static struct platform_driver arm_smmu_driver = {
        .driver = {