mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-08-31 22:23:05 +00:00

Define handlers specific to ACE platforms, that Frisco Lake (FCL), a PantherLake (PTL)-based platform, is founded upon. Most operations are still inherited from their predecessors with the major difference being AudioDSP cores management - replaced by DSP-domain power management. Software has to ensure the DSP domain is both powered on and its power-gating disabled before it can be utilized for streaming. Reviewed-by: Amadeusz Sławiński <amadeuszx.slawinski@linux.intel.com> Signed-off-by: Cezary Rojewski <cezary.rojewski@intel.com> Acked-by: Liam Girdwood <liam.r.girdwood@linux.intel.com> Link: https://patch.msgid.link/20250407112352.3720779-6-cezary.rojewski@intel.com Signed-off-by: Mark Brown <broonie@kernel.org>
201 lines
6.1 KiB
C
201 lines
6.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright(c) 2021-2025 Intel Corporation
|
|
*
|
|
* Authors: Cezary Rojewski <cezary.rojewski@intel.com>
|
|
* Amadeusz Slawinski <amadeuszx.slawinski@linux.intel.com>
|
|
*/
|
|
|
|
#include <sound/hdaudio_ext.h>
|
|
#include "avs.h"
|
|
#include "registers.h"
|
|
#include "trace.h"
|
|
|
|
#define MTL_HfDSSGBL_BASE 0x1000
|
|
#define MTL_REG_HfDSSCS (MTL_HfDSSGBL_BASE + 0x0)
|
|
#define MTL_HfDSSCS_SPA BIT(16)
|
|
#define MTL_HfDSSCS_CPA BIT(24)
|
|
|
|
#define MTL_DSPCS_BASE 0x178D00
|
|
#define MTL_REG_DSPCCTL (MTL_DSPCS_BASE + 0x4)
|
|
#define MTL_DSPCCTL_SPA BIT(0)
|
|
#define MTL_DSPCCTL_CPA BIT(8)
|
|
#define MTL_DSPCCTL_OSEL GENMASK(25, 24)
|
|
#define MTL_DSPCCTL_OSEL_HOST BIT(25)
|
|
|
|
#define MTL_HfINT_BASE 0x1100
|
|
#define MTL_REG_HfINTIPPTR (MTL_HfINT_BASE + 0x8)
|
|
#define MTL_REG_HfHIPCIE (MTL_HfINT_BASE + 0x40)
|
|
#define MTL_HfINTIPPTR_PTR GENMASK(20, 0)
|
|
#define MTL_HfHIPCIE_IE BIT(0)
|
|
|
|
#define MTL_DWICTL_INTENL_IE BIT(0)
|
|
#define MTL_DWICTL_FINALSTATUSL_IPC BIT(0) /* same as ADSPIS_IPC */
|
|
|
|
static int avs_mtl_core_power_on(struct avs_dev *adev)
|
|
{
|
|
u32 reg;
|
|
int ret;
|
|
|
|
/* Power up DSP domain. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfDSSCS, MTL_HfDSSCS_SPA, MTL_HfDSSCS_SPA);
|
|
trace_avs_dsp_core_op(1, AVS_MAIN_CORE_MASK, "power dsp", true);
|
|
|
|
ret = snd_hdac_adsp_readl_poll(adev, MTL_REG_HfDSSCS, reg,
|
|
(reg & MTL_HfDSSCS_CPA) == MTL_HfDSSCS_CPA,
|
|
AVS_ADSPCS_INTERVAL_US, AVS_ADSPCS_TIMEOUT_US);
|
|
if (ret) {
|
|
dev_err(adev->dev, "power on domain dsp failed: %d\n", ret);
|
|
return ret;
|
|
}
|
|
|
|
/* Prevent power gating of DSP domain. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfPWRCTL, MTL_HfPWRCTL_WPDSPHPxPG,
|
|
MTL_HfPWRCTL_WPDSPHPxPG);
|
|
trace_avs_dsp_core_op(1, AVS_MAIN_CORE_MASK, "prevent dsp PG", true);
|
|
|
|
ret = snd_hdac_adsp_readl_poll(adev, MTL_REG_HfPWRSTS, reg,
|
|
(reg & MTL_HfPWRSTS_DSPHPxPGS) == MTL_HfPWRSTS_DSPHPxPGS,
|
|
AVS_ADSPCS_INTERVAL_US, AVS_ADSPCS_TIMEOUT_US);
|
|
|
|
/* Set ownership to HOST. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_DSPCCTL, MTL_DSPCCTL_OSEL, MTL_DSPCCTL_OSEL_HOST);
|
|
return ret;
|
|
}
|
|
|
|
static int avs_mtl_core_power_off(struct avs_dev *adev)
|
|
{
|
|
u32 reg;
|
|
|
|
/* Allow power gating of DSP domain. No STS polling as HOST is only one of its users. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfPWRCTL, MTL_HfPWRCTL_WPDSPHPxPG, 0);
|
|
trace_avs_dsp_core_op(0, AVS_MAIN_CORE_MASK, "allow dsp pg", false);
|
|
|
|
/* Power down DSP domain. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfDSSCS, MTL_HfDSSCS_SPA, 0);
|
|
trace_avs_dsp_core_op(0, AVS_MAIN_CORE_MASK, "power dsp", false);
|
|
|
|
return snd_hdac_adsp_readl_poll(adev, MTL_REG_HfDSSCS, reg,
|
|
(reg & MTL_HfDSSCS_CPA) == 0,
|
|
AVS_ADSPCS_INTERVAL_US, AVS_ADSPCS_TIMEOUT_US);
|
|
}
|
|
|
|
int avs_mtl_core_power(struct avs_dev *adev, u32 core_mask, bool power)
|
|
{
|
|
core_mask &= AVS_MAIN_CORE_MASK;
|
|
if (!core_mask)
|
|
return 0;
|
|
|
|
if (power)
|
|
return avs_mtl_core_power_on(adev);
|
|
return avs_mtl_core_power_off(adev);
|
|
}
|
|
|
|
int avs_mtl_core_reset(struct avs_dev *adev, u32 core_mask, bool power)
|
|
{
|
|
/* No logical equivalent on ACE 1.x. */
|
|
return 0;
|
|
}
|
|
|
|
int avs_mtl_core_stall(struct avs_dev *adev, u32 core_mask, bool stall)
|
|
{
|
|
u32 value, reg;
|
|
int ret;
|
|
|
|
core_mask &= AVS_MAIN_CORE_MASK;
|
|
if (!core_mask)
|
|
return 0;
|
|
|
|
value = snd_hdac_adsp_readl(adev, MTL_REG_DSPCCTL);
|
|
trace_avs_dsp_core_op(value, core_mask, "stall", stall);
|
|
if (value == UINT_MAX)
|
|
return 0;
|
|
|
|
value = stall ? 0 : MTL_DSPCCTL_SPA;
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_DSPCCTL, MTL_DSPCCTL_SPA, value);
|
|
|
|
value = stall ? 0 : MTL_DSPCCTL_CPA;
|
|
ret = snd_hdac_adsp_readl_poll(adev, MTL_REG_DSPCCTL,
|
|
reg, (reg & MTL_DSPCCTL_CPA) == value,
|
|
AVS_ADSPCS_INTERVAL_US, AVS_ADSPCS_TIMEOUT_US);
|
|
if (ret)
|
|
dev_err(adev->dev, "core_mask %d %sstall failed: %d\n",
|
|
core_mask, stall ? "" : "un", ret);
|
|
return ret;
|
|
}
|
|
|
|
static void avs_mtl_ipc_interrupt(struct avs_dev *adev)
|
|
{
|
|
const struct avs_spec *spec = adev->spec;
|
|
u32 hipc_ack, hipc_rsp;
|
|
|
|
snd_hdac_adsp_updatel(adev, spec->hipc->ctl_offset,
|
|
AVS_ADSP_HIPCCTL_DONE | AVS_ADSP_HIPCCTL_BUSY, 0);
|
|
|
|
hipc_ack = snd_hdac_adsp_readl(adev, spec->hipc->ack_offset);
|
|
hipc_rsp = snd_hdac_adsp_readl(adev, spec->hipc->rsp_offset);
|
|
|
|
/* DSP acked host's request. */
|
|
if (hipc_ack & spec->hipc->ack_done_mask) {
|
|
complete(&adev->ipc->done_completion);
|
|
|
|
/* Tell DSP it has our attention. */
|
|
snd_hdac_adsp_updatel(adev, spec->hipc->ack_offset, spec->hipc->ack_done_mask,
|
|
spec->hipc->ack_done_mask);
|
|
}
|
|
|
|
/* DSP sent new response to process. */
|
|
if (hipc_rsp & spec->hipc->rsp_busy_mask) {
|
|
union avs_reply_msg msg;
|
|
|
|
msg.primary = snd_hdac_adsp_readl(adev, MTL_REG_HfIPCxTDR);
|
|
msg.ext.val = snd_hdac_adsp_readl(adev, MTL_REG_HfIPCxTDD);
|
|
|
|
avs_dsp_process_response(adev, msg.val);
|
|
|
|
/* Tell DSP we accepted its message. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxTDR,
|
|
MTL_HfIPCxTDR_BUSY, MTL_HfIPCxTDR_BUSY);
|
|
/* Ack this response. */
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxTDA, MTL_HfIPCxTDA_BUSY, 0);
|
|
}
|
|
|
|
snd_hdac_adsp_updatel(adev, spec->hipc->ctl_offset,
|
|
AVS_ADSP_HIPCCTL_DONE | AVS_ADSP_HIPCCTL_BUSY,
|
|
AVS_ADSP_HIPCCTL_DONE | AVS_ADSP_HIPCCTL_BUSY);
|
|
}
|
|
|
|
irqreturn_t avs_mtl_dsp_interrupt(struct avs_dev *adev)
|
|
{
|
|
u32 adspis = snd_hdac_adsp_readl(adev, MTL_DWICTL_REG_FINALSTATUSL);
|
|
irqreturn_t ret = IRQ_NONE;
|
|
|
|
if (adspis == UINT_MAX)
|
|
return ret;
|
|
|
|
if (adspis & MTL_DWICTL_FINALSTATUSL_IPC) {
|
|
avs_mtl_ipc_interrupt(adev);
|
|
ret = IRQ_HANDLED;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
void avs_mtl_interrupt_control(struct avs_dev *adev, bool enable)
|
|
{
|
|
if (enable) {
|
|
snd_hdac_adsp_updatel(adev, MTL_DWICTL_REG_INTENL, MTL_DWICTL_INTENL_IE,
|
|
MTL_DWICTL_INTENL_IE);
|
|
snd_hdac_adsp_updatew(adev, MTL_REG_HfHIPCIE, MTL_HfHIPCIE_IE, MTL_HfHIPCIE_IE);
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxCTL, AVS_ADSP_HIPCCTL_DONE,
|
|
AVS_ADSP_HIPCCTL_DONE);
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxCTL, AVS_ADSP_HIPCCTL_BUSY,
|
|
AVS_ADSP_HIPCCTL_BUSY);
|
|
} else {
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxCTL, AVS_ADSP_HIPCCTL_BUSY, 0);
|
|
snd_hdac_adsp_updatel(adev, MTL_REG_HfIPCxCTL, AVS_ADSP_HIPCCTL_DONE, 0);
|
|
snd_hdac_adsp_updatew(adev, MTL_REG_HfHIPCIE, MTL_HfHIPCIE_IE, 0);
|
|
snd_hdac_adsp_updatel(adev, MTL_DWICTL_REG_INTENL, MTL_DWICTL_INTENL_IE, 0);
|
|
}
|
|
}
|