ASoC: Intel: Skylake: Support multi-core in Skylake
authorJayachandran B <jayachandran.b@intel.com>
Tue, 21 Jun 2016 04:47:42 +0000 (10:17 +0530)
committerMark Brown <broonie@kernel.org>
Wed, 22 Jun 2016 15:13:12 +0000 (16:13 +0100)
Add multicore DSP support in Skylake DSP operations.

Signed-off-by: Jayachandran B <jayachandran.b@intel.com>
Signed-off-by: Vinod Koul <vinod.koul@intel.com>
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/intel/skylake/skl-sst.c

index ecaca94d2a968c8af4eb0e3b1005d0b02863fd8a..588f899ceb652b5efcb2b87451a058f6dfc2ab0d 100644 (file)
@@ -150,7 +150,6 @@ static int skl_load_base_firmware(struct sst_dsp *ctx)
                }
 
                dev_dbg(ctx->dev, "Download firmware successful%d\n", ret);
-               skl_dsp_set_state_locked(ctx, SKL_DSP_RUNNING);
                skl->fw_loaded = true;
        }
        return 0;
@@ -166,14 +165,41 @@ skl_load_base_firmware_failed:
 static int skl_set_dsp_D0(struct sst_dsp *ctx, unsigned int core_id)
 {
        int ret;
+       struct skl_ipc_dxstate_info dx;
+       struct skl_sst *skl = ctx->thread_context;
+       unsigned int core_mask = SKL_DSP_CORE_MASK(core_id);
 
-       ret = skl_load_base_firmware(ctx);
-       if (ret < 0) {
-               dev_err(ctx->dev, "unable to load firmware\n");
-               return ret;
+       /* If core0 is being turned on, we need to load the FW */
+       if (core_id == SKL_DSP_CORE0_ID) {
+               ret = skl_load_base_firmware(ctx);
+               if (ret < 0) {
+                       dev_err(ctx->dev, "unable to load firmware\n");
+                       return ret;
+               }
+       }
+
+       /*
+        * If any core other than core 0 is being moved to D0, enable the
+        * core and send the set dx IPC for the core.
+        */
+       if (core_id != SKL_DSP_CORE0_ID) {
+               ret = skl_dsp_enable_core(ctx, core_mask);
+               if (ret < 0)
+                       return ret;
+
+               dx.core_mask = core_mask;
+               dx.dx_mask = core_mask;
+
+               ret = skl_ipc_set_dx(&skl->ipc, SKL_INSTANCE_ID,
+                                       SKL_BASE_FW_MODULE_ID, &dx);
+               if (ret < 0) {
+                       dev_err(ctx->dev, "Failed to set dsp to D0:core id= %d\n",
+                                       core_id);
+                       skl_dsp_disable_core(ctx, core_mask);
+               }
        }
 
-       skl_dsp_set_state_locked(ctx, SKL_DSP_RUNNING);
+       skl->cores.state[core_id] = SKL_DSP_RUNNING;
 
        return ret;
 }
@@ -183,35 +209,28 @@ static int skl_set_dsp_D3(struct sst_dsp *ctx, unsigned int core_id)
        int ret;
        struct skl_ipc_dxstate_info dx;
        struct skl_sst *skl = ctx->thread_context;
+       unsigned int core_mask = SKL_DSP_CORE_MASK(core_id);
 
-       dev_dbg(ctx->dev, "In %s:\n", __func__);
-       mutex_lock(&ctx->mutex);
-       if (!is_skl_dsp_running(ctx)) {
-               mutex_unlock(&ctx->mutex);
-               return 0;
-       }
-       mutex_unlock(&ctx->mutex);
-
-       dx.core_mask = SKL_DSP_CORE0_MASK;
+       dx.core_mask = core_mask;
        dx.dx_mask = SKL_IPC_D3_MASK;
+
        ret = skl_ipc_set_dx(&skl->ipc, SKL_INSTANCE_ID, SKL_BASE_FW_MODULE_ID, &dx);
        if (ret < 0)
-               dev_err(ctx->dev,
-                       "D3 request to FW failed, continuing reset: %d", ret);
-
-       /* disable Interrupt */
-       ctx->cl_dev.ops.cl_cleanup_controller(ctx);
-       skl_cldma_int_disable(ctx);
-       skl_ipc_op_int_disable(ctx);
-       skl_ipc_int_disable(ctx);
-
-       ret = skl_dsp_disable_core(ctx, core_id);
-       if (ret < 0) {
-               dev_err(ctx->dev, "disable dsp core failed ret: %d\n", ret);
-               ret = -EIO;
+               dev_err(ctx->dev, "set Dx core %d fail: %d\n", core_id, ret);
+
+       if (core_id == SKL_DSP_CORE0_ID) {
+               /* disable Interrupt */
+               ctx->cl_dev.ops.cl_cleanup_controller(ctx);
+               skl_cldma_int_disable(ctx);
+               skl_ipc_op_int_disable(ctx);
+               skl_ipc_int_disable(ctx);
        }
-       skl_dsp_set_state_locked(ctx, SKL_DSP_RESET);
 
+       ret = skl_dsp_disable_core(ctx, core_mask);
+       if (ret < 0)
+               return ret;
+
+       skl->cores.state[core_id] = SKL_DSP_RESET;
        return ret;
 }