firmware: qcom_scm: Reduce locking section for __get_convention()
authorStephen Boyd <swboyd@chromium.org>
Tue, 23 Feb 2021 21:45:35 +0000 (13:45 -0800)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Fri, 14 May 2021 07:50:14 +0000 (09:50 +0200)
[ Upstream commit f6ea568f0ddcdfad52807110ed8983e610f0e03b ]

We shouldn't need to hold this spinlock here around the entire SCM call
into the firmware and back. Instead, we should be able to query the
firmware, potentially in parallel with other CPUs making the same
convention detection firmware call, and then grab the lock to update the
calling convention detected. The convention doesn't change at runtime so
calling into firmware more than once is possibly wasteful but simpler.
Besides, this is the slow path, not the fast path where we've already
detected the convention used.

More importantly, this allows us to add more logic here to workaround
the case where the firmware call to check for availability isn't
implemented in the firmware at all. In that case we can check the
firmware node compatible string and force a calling convention.

Note that we remove the 'has_queried' logic that is repeated twice. That
could lead to the calling convention being printed multiple times to the
kernel logs if the bool is true but __query_convention() is running on
multiple CPUs. We also shorten the time where the lock is held, but we
keep the lock held around the printk because it doesn't seem hugely
important to drop it for that.

Cc: Elliot Berman <eberman@codeaurora.org>
Cc: Brian Masney <masneyb@onstation.org>
Cc: Stephan Gerhold <stephan@gerhold.net>
Cc: Jeffrey Hugo <jhugo@codeaurora.org>
Cc: Douglas Anderson <dianders@chromium.org>
Fixes: 9a434cee773a ("firmware: qcom_scm: Dynamically support SMCCC and legacy conventions")
Signed-off-by: Stephen Boyd <swboyd@chromium.org>
Link: https://lore.kernel.org/r/20210223214539.1336155-3-swboyd@chromium.org
Signed-off-by: Bjorn Andersson <bjorn.andersson@linaro.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
drivers/firmware/qcom_scm-smc.c
drivers/firmware/qcom_scm.c
drivers/firmware/qcom_scm.h

index 497c13ba98d67f15651fd3979dfacf98475d7c29..d111833364ba4f3f5b93faad475be307569ec0e4 100644 (file)
@@ -77,8 +77,10 @@ static void __scm_smc_do(const struct arm_smccc_args *smc,
        }  while (res->a0 == QCOM_SCM_V2_EBUSY);
 }
 
-int scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
-                struct qcom_scm_res *res, bool atomic)
+
+int __scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
+                  enum qcom_scm_convention qcom_convention,
+                  struct qcom_scm_res *res, bool atomic)
 {
        int arglen = desc->arginfo & 0xf;
        int i;
@@ -87,9 +89,8 @@ int scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
        size_t alloc_len;
        gfp_t flag = atomic ? GFP_ATOMIC : GFP_KERNEL;
        u32 smccc_call_type = atomic ? ARM_SMCCC_FAST_CALL : ARM_SMCCC_STD_CALL;
-       u32 qcom_smccc_convention =
-                       (qcom_scm_convention == SMC_CONVENTION_ARM_32) ?
-                       ARM_SMCCC_SMC_32 : ARM_SMCCC_SMC_64;
+       u32 qcom_smccc_convention = (qcom_convention == SMC_CONVENTION_ARM_32) ?
+                                   ARM_SMCCC_SMC_32 : ARM_SMCCC_SMC_64;
        struct arm_smccc_res smc_res;
        struct arm_smccc_args smc = {0};
 
@@ -148,4 +149,5 @@ int scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
        }
 
        return (long)smc_res.a0 ? qcom_scm_remap_error(smc_res.a0) : 0;
+
 }
index 54ba2834e7630b458c21e4341d3317560ac225f4..a455c22bcdbdb83bd3a73a0f470acd12da74fb51 100644 (file)
@@ -113,11 +113,10 @@ static void qcom_scm_clk_disable(void)
        clk_disable_unprepare(__scm->bus_clk);
 }
 
-enum qcom_scm_convention qcom_scm_convention;
-static bool has_queried __read_mostly;
-static DEFINE_SPINLOCK(query_lock);
+enum qcom_scm_convention qcom_scm_convention = SMC_CONVENTION_UNKNOWN;
+static DEFINE_SPINLOCK(scm_query_lock);
 
-static void __query_convention(void)
+static enum qcom_scm_convention __get_convention(void)
 {
        unsigned long flags;
        struct qcom_scm_desc desc = {
@@ -130,36 +129,36 @@ static void __query_convention(void)
                .owner = ARM_SMCCC_OWNER_SIP,
        };
        struct qcom_scm_res res;
+       enum qcom_scm_convention probed_convention;
        int ret;
 
-       spin_lock_irqsave(&query_lock, flags);
-       if (has_queried)
-               goto out;
+       if (likely(qcom_scm_convention != SMC_CONVENTION_UNKNOWN))
+               return qcom_scm_convention;
 
-       qcom_scm_convention = SMC_CONVENTION_ARM_64;
-       // Device isn't required as there is only one argument - no device
-       // needed to dma_map_single to secure world
-       ret = scm_smc_call(NULL, &desc, &res, true);
+       /*
+        * Device isn't required as there is only one argument - no device
+        * needed to dma_map_single to secure world
+        */
+       probed_convention = SMC_CONVENTION_ARM_64;
+       ret = __scm_smc_call(NULL, &desc, probed_convention, &res, true);
        if (!ret && res.result[0] == 1)
-               goto out;
+               goto found;
 
-       qcom_scm_convention = SMC_CONVENTION_ARM_32;
-       ret = scm_smc_call(NULL, &desc, &res, true);
+       probed_convention = SMC_CONVENTION_ARM_32;
+       ret = __scm_smc_call(NULL, &desc, probed_convention, &res, true);
        if (!ret && res.result[0] == 1)
-               goto out;
-
-       qcom_scm_convention = SMC_CONVENTION_LEGACY;
-out:
-       has_queried = true;
-       spin_unlock_irqrestore(&query_lock, flags);
-       pr_info("qcom_scm: convention: %s\n",
-               qcom_scm_convention_names[qcom_scm_convention]);
-}
+               goto found;
+
+       probed_convention = SMC_CONVENTION_LEGACY;
+found:
+       spin_lock_irqsave(&scm_query_lock, flags);
+       if (probed_convention != qcom_scm_convention) {
+               qcom_scm_convention = probed_convention;
+               pr_info("qcom_scm: convention: %s\n",
+                       qcom_scm_convention_names[qcom_scm_convention]);
+       }
+       spin_unlock_irqrestore(&scm_query_lock, flags);
 
-static inline enum qcom_scm_convention __get_convention(void)
-{
-       if (unlikely(!has_queried))
-               __query_convention();
        return qcom_scm_convention;
 }
 
@@ -1233,7 +1232,7 @@ static int qcom_scm_probe(struct platform_device *pdev)
        __scm = scm;
        __scm->dev = &pdev->dev;
 
-       __query_convention();
+       __get_convention();
 
        /*
         * If requested enable "download mode", from this point on warmboot
index 95cd1ac30ab0bb23c57b212a84eb773ae1079330..632fe31424621a79f4793e8d0ee99c53320ae2ff 100644 (file)
@@ -61,8 +61,11 @@ struct qcom_scm_res {
 };
 
 #define SCM_SMC_FNID(s, c)     ((((s) & 0xFF) << 8) | ((c) & 0xFF))
-extern int scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
-                       struct qcom_scm_res *res, bool atomic);
+extern int __scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc,
+                         enum qcom_scm_convention qcom_convention,
+                         struct qcom_scm_res *res, bool atomic);
+#define scm_smc_call(dev, desc, res, atomic) \
+       __scm_smc_call((dev), (desc), qcom_scm_convention, (res), (atomic))
 
 #define SCM_LEGACY_FNID(s, c)  (((s) << 10) | ((c) & 0x3ff))
 extern int scm_legacy_call_atomic(struct device *dev,