Commit 3bcca378 authored by Mark Brown's avatar Mark Brown

ASoC: Intel: avs: PCM power management

Merge series from Cezary Rojewski <cezary.rojewski@intel.com>:

Goal of the series is implementation of suspend/resume operations for a
PCM stream along with all the collaterals connected to the subject.

Start with splitting avs_dai_fe_hw_free() as ideally we would like to
reuse as much of existing code as possible but snd_pcm_lib_free_pages()
is not desired part of the function when speaking of suspend operation.

The actual implementation of suspend/resume() for component drivers
follows. For most scenarios, the PM flow is similar to standard
streaming one, except for the part where the position register are being
saved and the lack of PCM pages freeing. To reduce code duplication, all
avs_dai_suspend_XXX() and avs_dai_resume_XXX() functions reuse their
non-PM equivalents.
Order of operations is affected by the fact that path binding/unbinding
happens only in FE part of the stream.

Above essentially unlocks SX+streaming scenarios i.e.: power transitions
with an ongoing stream.

As some streams are allowed to run in low power state, support is
provided for S0iX state. The handlers check ACPI capabilities and the
number of active low-power paths before deciding between SX and S0iX
flows.

The last portion of the patchset is addition of power/clock gating
overrides. There is no single set of registers that ensures AudioDSP
firmware loads 100% of time on every single configuration. By having
them exposed, user can have the loading procedure behavior adjusted for
their configuration without having to recompile the kernel.
parents 5e01ff7d 758ba92f
...@@ -597,6 +597,7 @@ int snd_hdac_stream_get_spbmaxfifo(struct hdac_bus *bus, ...@@ -597,6 +597,7 @@ int snd_hdac_stream_get_spbmaxfifo(struct hdac_bus *bus,
struct hdac_stream *azx_dev); struct hdac_stream *azx_dev);
void snd_hdac_stream_drsm_enable(struct hdac_bus *bus, void snd_hdac_stream_drsm_enable(struct hdac_bus *bus,
bool enable, int index); bool enable, int index);
int snd_hdac_stream_wait_drsm(struct hdac_stream *azx_dev);
int snd_hdac_stream_set_dpibr(struct hdac_bus *bus, int snd_hdac_stream_set_dpibr(struct hdac_bus *bus,
struct hdac_stream *azx_dev, u32 value); struct hdac_stream *azx_dev, u32 value);
int snd_hdac_stream_set_lpib(struct hdac_stream *azx_dev, u32 value); int snd_hdac_stream_set_lpib(struct hdac_stream *azx_dev, u32 value);
......
...@@ -51,6 +51,11 @@ struct hdac_ext_stream { ...@@ -51,6 +51,11 @@ struct hdac_ext_stream {
void __iomem *pphc_addr; void __iomem *pphc_addr;
void __iomem *pplc_addr; void __iomem *pplc_addr;
u32 pphcllpl;
u32 pphcllpu;
u32 pphcldpl;
u32 pphcldpu;
bool decoupled:1; bool decoupled:1;
bool link_locked:1; bool link_locked:1;
bool link_prepared; bool link_prepared;
......
...@@ -821,6 +821,28 @@ void snd_hdac_stream_drsm_enable(struct hdac_bus *bus, ...@@ -821,6 +821,28 @@ void snd_hdac_stream_drsm_enable(struct hdac_bus *bus,
} }
EXPORT_SYMBOL_GPL(snd_hdac_stream_drsm_enable); EXPORT_SYMBOL_GPL(snd_hdac_stream_drsm_enable);
/*
* snd_hdac_stream_wait_drsm - wait for HW to clear RSM for a stream
* @azx_dev: HD-audio core stream to await RSM for
*
* Returns 0 on success and -ETIMEDOUT upon a timeout.
*/
int snd_hdac_stream_wait_drsm(struct hdac_stream *azx_dev)
{
struct hdac_bus *bus = azx_dev->bus;
u32 mask, reg;
int ret;
mask = 1 << azx_dev->index;
ret = read_poll_timeout(snd_hdac_reg_readl, reg, !(reg & mask), 250, 2000, false, bus,
bus->drsmcap + AZX_REG_DRSM_CTL);
if (ret)
dev_dbg(bus->dev, "polling RSM 0x%08x failed: %d\n", mask, ret);
return ret;
}
EXPORT_SYMBOL_GPL(snd_hdac_stream_wait_drsm);
/** /**
* snd_hdac_stream_set_dpibr - sets the dpibr value of a stream * snd_hdac_stream_set_dpibr - sets the dpibr value of a stream
* @bus: HD-audio core bus * @bus: HD-audio core bus
......
...@@ -24,6 +24,13 @@ struct avs_tplg_library; ...@@ -24,6 +24,13 @@ struct avs_tplg_library;
struct avs_soc_component; struct avs_soc_component;
struct avs_ipc_msg; struct avs_ipc_msg;
#ifdef CONFIG_ACPI
#define AVS_S0IX_SUPPORTED \
(acpi_gbl_FADT.flags & ACPI_FADT_LOW_POWER_S0)
#else
#define AVS_S0IX_SUPPORTED false
#endif
/* /*
* struct avs_dsp_ops - Platform-specific DSP operations * struct avs_dsp_ops - Platform-specific DSP operations
* *
...@@ -127,6 +134,7 @@ struct avs_dev { ...@@ -127,6 +134,7 @@ struct avs_dev {
struct list_head fw_list; struct list_head fw_list;
int *core_refs; /* reference count per core */ int *core_refs; /* reference count per core */
char **lib_names; char **lib_names;
int num_lp_paths;
struct completion fw_ready; struct completion fw_ready;
struct work_struct probe_work; struct work_struct probe_work;
......
...@@ -27,6 +27,14 @@ ...@@ -27,6 +27,14 @@
#include "avs.h" #include "avs.h"
#include "cldma.h" #include "cldma.h"
static u32 pgctl_mask = AZX_PGCTL_LSRMD_MASK;
module_param(pgctl_mask, uint, 0444);
MODULE_PARM_DESC(pgctl_mask, "PCI PGCTL policy override");
static u32 cgctl_mask = AZX_CGCTL_MISCBDCGE_MASK;
module_param(cgctl_mask, uint, 0444);
MODULE_PARM_DESC(cgctl_mask, "PCI CGCTL policy override");
static void static void
avs_hda_update_config_dword(struct hdac_bus *bus, u32 reg, u32 mask, u32 value) avs_hda_update_config_dword(struct hdac_bus *bus, u32 reg, u32 mask, u32 value)
{ {
...@@ -41,19 +49,16 @@ avs_hda_update_config_dword(struct hdac_bus *bus, u32 reg, u32 mask, u32 value) ...@@ -41,19 +49,16 @@ avs_hda_update_config_dword(struct hdac_bus *bus, u32 reg, u32 mask, u32 value)
void avs_hda_power_gating_enable(struct avs_dev *adev, bool enable) void avs_hda_power_gating_enable(struct avs_dev *adev, bool enable)
{ {
u32 value; u32 value = enable ? 0 : pgctl_mask;
value = enable ? 0 : AZX_PGCTL_LSRMD_MASK; avs_hda_update_config_dword(&adev->base.core, AZX_PCIREG_PGCTL, pgctl_mask, value);
avs_hda_update_config_dword(&adev->base.core, AZX_PCIREG_PGCTL,
AZX_PGCTL_LSRMD_MASK, value);
} }
static void avs_hdac_clock_gating_enable(struct hdac_bus *bus, bool enable) static void avs_hdac_clock_gating_enable(struct hdac_bus *bus, bool enable)
{ {
u32 value; u32 value = enable ? cgctl_mask : 0;
value = enable ? AZX_CGCTL_MISCBDCGE_MASK : 0; avs_hda_update_config_dword(bus, AZX_PCIREG_CGCTL, cgctl_mask, value);
avs_hda_update_config_dword(bus, AZX_PCIREG_CGCTL, AZX_CGCTL_MISCBDCGE_MASK, value);
} }
void avs_hda_clock_gating_enable(struct avs_dev *adev, bool enable) void avs_hda_clock_gating_enable(struct avs_dev *adev, bool enable)
...@@ -63,9 +68,8 @@ void avs_hda_clock_gating_enable(struct avs_dev *adev, bool enable) ...@@ -63,9 +68,8 @@ void avs_hda_clock_gating_enable(struct avs_dev *adev, bool enable)
void avs_hda_l1sen_enable(struct avs_dev *adev, bool enable) void avs_hda_l1sen_enable(struct avs_dev *adev, bool enable)
{ {
u32 value; u32 value = enable ? AZX_VS_EM2_L1SEN : 0;
value = enable ? AZX_VS_EM2_L1SEN : 0;
snd_hdac_chip_updatel(&adev->base.core, VS_EM2, AZX_VS_EM2_L1SEN, value); snd_hdac_chip_updatel(&adev->base.core, VS_EM2, AZX_VS_EM2_L1SEN, value);
} }
...@@ -534,12 +538,30 @@ static void avs_pci_remove(struct pci_dev *pci) ...@@ -534,12 +538,30 @@ static void avs_pci_remove(struct pci_dev *pci)
pm_runtime_get_noresume(&pci->dev); pm_runtime_get_noresume(&pci->dev);
} }
static int __maybe_unused avs_suspend_common(struct avs_dev *adev) static int avs_suspend_standby(struct avs_dev *adev)
{
struct hdac_bus *bus = &adev->base.core;
struct pci_dev *pci = adev->base.pci;
if (bus->cmd_dma_state)
snd_hdac_bus_stop_cmd_io(bus);
snd_hdac_ext_bus_link_power_down_all(bus);
enable_irq_wake(pci->irq);
pci_save_state(pci);
return 0;
}
static int __maybe_unused avs_suspend_common(struct avs_dev *adev, bool low_power)
{ {
struct hdac_bus *bus = &adev->base.core; struct hdac_bus *bus = &adev->base.core;
int ret; int ret;
flush_work(&adev->probe_work); flush_work(&adev->probe_work);
if (low_power && adev->num_lp_paths)
return avs_suspend_standby(adev);
snd_hdac_ext_bus_link_power_down_all(bus); snd_hdac_ext_bus_link_power_down_all(bus);
...@@ -577,11 +599,30 @@ static int __maybe_unused avs_suspend_common(struct avs_dev *adev) ...@@ -577,11 +599,30 @@ static int __maybe_unused avs_suspend_common(struct avs_dev *adev)
return 0; return 0;
} }
static int __maybe_unused avs_resume_common(struct avs_dev *adev, bool purge) static int avs_resume_standby(struct avs_dev *adev)
{
struct hdac_bus *bus = &adev->base.core;
struct pci_dev *pci = adev->base.pci;
pci_restore_state(pci);
disable_irq_wake(pci->irq);
snd_hdac_ext_bus_link_power_up_all(bus);
if (bus->cmd_dma_state)
snd_hdac_bus_init_cmd_io(bus);
return 0;
}
static int __maybe_unused avs_resume_common(struct avs_dev *adev, bool low_power, bool purge)
{ {
struct hdac_bus *bus = &adev->base.core; struct hdac_bus *bus = &adev->base.core;
int ret; int ret;
if (low_power && adev->num_lp_paths)
return avs_resume_standby(adev);
snd_hdac_display_power(bus, HDA_CODEC_IDX_CONTROLLER, true); snd_hdac_display_power(bus, HDA_CODEC_IDX_CONTROLLER, true);
avs_hdac_bus_init_chip(bus, true); avs_hdac_bus_init_chip(bus, true);
...@@ -599,26 +640,50 @@ static int __maybe_unused avs_resume_common(struct avs_dev *adev, bool purge) ...@@ -599,26 +640,50 @@ static int __maybe_unused avs_resume_common(struct avs_dev *adev, bool purge)
static int __maybe_unused avs_suspend(struct device *dev) static int __maybe_unused avs_suspend(struct device *dev)
{ {
return avs_suspend_common(to_avs_dev(dev)); return avs_suspend_common(to_avs_dev(dev), true);
} }
static int __maybe_unused avs_resume(struct device *dev) static int __maybe_unused avs_resume(struct device *dev)
{ {
return avs_resume_common(to_avs_dev(dev), true); return avs_resume_common(to_avs_dev(dev), true, true);
} }
static int __maybe_unused avs_runtime_suspend(struct device *dev) static int __maybe_unused avs_runtime_suspend(struct device *dev)
{ {
return avs_suspend_common(to_avs_dev(dev)); return avs_suspend_common(to_avs_dev(dev), true);
} }
static int __maybe_unused avs_runtime_resume(struct device *dev) static int __maybe_unused avs_runtime_resume(struct device *dev)
{ {
return avs_resume_common(to_avs_dev(dev), true); return avs_resume_common(to_avs_dev(dev), true, false);
}
static int __maybe_unused avs_freeze(struct device *dev)
{
return avs_suspend_common(to_avs_dev(dev), false);
}
static int __maybe_unused avs_thaw(struct device *dev)
{
return avs_resume_common(to_avs_dev(dev), false, true);
}
static int __maybe_unused avs_poweroff(struct device *dev)
{
return avs_suspend_common(to_avs_dev(dev), false);
}
static int __maybe_unused avs_restore(struct device *dev)
{
return avs_resume_common(to_avs_dev(dev), false, true);
} }
static const struct dev_pm_ops avs_dev_pm = { static const struct dev_pm_ops avs_dev_pm = {
SET_SYSTEM_SLEEP_PM_OPS(avs_suspend, avs_resume) .suspend = avs_suspend,
.resume = avs_resume,
.freeze = avs_freeze,
.thaw = avs_thaw,
.poweroff = avs_poweroff,
.restore = avs_restore,
SET_RUNTIME_PM_OPS(avs_runtime_suspend, avs_runtime_resume, NULL) SET_RUNTIME_PM_OPS(avs_runtime_suspend, avs_runtime_resume, NULL)
}; };
......
...@@ -224,11 +224,19 @@ static int avs_cldma_load_module(struct avs_dev *adev, struct avs_module_entry * ...@@ -224,11 +224,19 @@ static int avs_cldma_load_module(struct avs_dev *adev, struct avs_module_entry *
if (ret < 0) if (ret < 0)
return ret; return ret;
avs_hda_power_gating_enable(adev, false);
avs_hda_clock_gating_enable(adev, false);
avs_hda_l1sen_enable(adev, false);
hda_cldma_set_data(cl, (void *)mod->data, mod->size); hda_cldma_set_data(cl, (void *)mod->data, mod->size);
hda_cldma_transfer(cl, msecs_to_jiffies(AVS_CLDMA_START_DELAY_MS)); hda_cldma_transfer(cl, msecs_to_jiffies(AVS_CLDMA_START_DELAY_MS));
ret = avs_ipc_load_modules(adev, &mentry->module_id, 1); ret = avs_ipc_load_modules(adev, &mentry->module_id, 1);
hda_cldma_stop(cl); hda_cldma_stop(cl);
avs_hda_l1sen_enable(adev, true);
avs_hda_clock_gating_enable(adev, true);
avs_hda_power_gating_enable(adev, true);
if (ret) { if (ret) {
dev_err(adev->dev, "load module %d failed: %d\n", mentry->module_id, ret); dev_err(adev->dev, "load module %d failed: %d\n", mentry->module_id, ret);
avs_release_last_firmware(adev); avs_release_last_firmware(adev);
...@@ -605,6 +613,7 @@ int avs_dsp_boot_firmware(struct avs_dev *adev, bool purge) ...@@ -605,6 +613,7 @@ int avs_dsp_boot_firmware(struct avs_dev *adev, bool purge)
for (i = 1; i < adev->fw_cfg.max_libs_count; i++) for (i = 1; i < adev->fw_cfg.max_libs_count; i++)
memset(adev->lib_names[i], 0, AVS_LIB_NAME_SIZE); memset(adev->lib_names[i], 0, AVS_LIB_NAME_SIZE);
avs_hda_power_gating_enable(adev, false);
avs_hda_clock_gating_enable(adev, false); avs_hda_clock_gating_enable(adev, false);
avs_hda_l1sen_enable(adev, false); avs_hda_l1sen_enable(adev, false);
...@@ -625,6 +634,7 @@ int avs_dsp_boot_firmware(struct avs_dev *adev, bool purge) ...@@ -625,6 +634,7 @@ int avs_dsp_boot_firmware(struct avs_dev *adev, bool purge)
reenable_gating: reenable_gating:
avs_hda_l1sen_enable(adev, true); avs_hda_l1sen_enable(adev, true);
avs_hda_clock_gating_enable(adev, true); avs_hda_clock_gating_enable(adev, true);
avs_hda_power_gating_enable(adev, true);
if (ret < 0) if (ret < 0)
return ret; return ret;
......
...@@ -28,6 +28,8 @@ struct avs_dma_data { ...@@ -28,6 +28,8 @@ struct avs_dma_data {
* host stream assigned * host stream assigned
*/ */
struct hdac_ext_stream *host_stream; struct hdac_ext_stream *host_stream;
struct snd_pcm_substream *substream;
}; };
static struct avs_tplg_path_template * static struct avs_tplg_path_template *
...@@ -55,8 +57,11 @@ avs_dai_find_path_template(struct snd_soc_dai *dai, bool is_fe, int direction) ...@@ -55,8 +57,11 @@ avs_dai_find_path_template(struct snd_soc_dai *dai, bool is_fe, int direction)
return dw->priv; return dw->priv;
} }
static int avs_dai_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai, bool is_fe) static int avs_dai_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai, bool is_fe,
const struct snd_soc_dai_ops *ops)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct avs_dev *adev = to_avs_dev(dai->dev);
struct avs_tplg_path_template *template; struct avs_tplg_path_template *template;
struct avs_dma_data *data; struct avs_dma_data *data;
...@@ -71,9 +76,13 @@ static int avs_dai_startup(struct snd_pcm_substream *substream, struct snd_soc_d ...@@ -71,9 +76,13 @@ static int avs_dai_startup(struct snd_pcm_substream *substream, struct snd_soc_d
if (!data) if (!data)
return -ENOMEM; return -ENOMEM;
data->substream = substream;
data->template = template; data->template = template;
snd_soc_dai_set_dma_data(dai, substream, data); snd_soc_dai_set_dma_data(dai, substream, data);
if (rtd->dai_link->ignore_suspend)
adev->num_lp_paths++;
return 0; return 0;
} }
...@@ -151,15 +160,22 @@ static int avs_dai_prepare(struct avs_dev *adev, struct snd_pcm_substream *subst ...@@ -151,15 +160,22 @@ static int avs_dai_prepare(struct avs_dev *adev, struct snd_pcm_substream *subst
return ret; return ret;
} }
static const struct snd_soc_dai_ops avs_dai_nonhda_be_ops;
static int avs_dai_nonhda_be_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static int avs_dai_nonhda_be_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
return avs_dai_startup(substream, dai, false); return avs_dai_startup(substream, dai, false, &avs_dai_nonhda_be_ops);
} }
static void avs_dai_nonhda_be_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static void avs_dai_nonhda_be_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct avs_dev *adev = to_avs_dev(dai->dev);
struct avs_dma_data *data; struct avs_dma_data *data;
if (rtd->dai_link->ignore_suspend)
adev->num_lp_paths--;
data = snd_soc_dai_get_dma_data(dai, substream); data = snd_soc_dai_get_dma_data(dai, substream);
snd_soc_dai_set_dma_data(dai, substream, NULL); snd_soc_dai_set_dma_data(dai, substream, NULL);
...@@ -202,30 +218,43 @@ static int avs_dai_nonhda_be_prepare(struct snd_pcm_substream *substream, struct ...@@ -202,30 +218,43 @@ static int avs_dai_nonhda_be_prepare(struct snd_pcm_substream *substream, struct
static int avs_dai_nonhda_be_trigger(struct snd_pcm_substream *substream, int cmd, static int avs_dai_nonhda_be_trigger(struct snd_pcm_substream *substream, int cmd,
struct snd_soc_dai *dai) struct snd_soc_dai *dai)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct avs_dma_data *data; struct avs_dma_data *data;
int ret = 0; int ret = 0;
data = snd_soc_dai_get_dma_data(dai, substream); data = snd_soc_dai_get_dma_data(dai, substream);
switch (cmd) { switch (cmd) {
case SNDRV_PCM_TRIGGER_RESUME:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_START:
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
ret = avs_path_pause(data->path);
if (ret < 0) {
dev_err(dai->dev, "pause BE path failed: %d\n", ret);
break;
}
ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO); ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO);
if (ret < 0) if (ret < 0)
dev_err(dai->dev, "run BE path failed: %d\n", ret); dev_err(dai->dev, "run BE path failed: %d\n", ret);
break; break;
case SNDRV_PCM_TRIGGER_SUSPEND:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_PAUSE_PUSH: case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_STOP:
ret = avs_path_pause(data->path); ret = avs_path_pause(data->path);
if (ret < 0) if (ret < 0)
dev_err(dai->dev, "pause BE path failed: %d\n", ret); dev_err(dai->dev, "pause BE path failed: %d\n", ret);
if (cmd == SNDRV_PCM_TRIGGER_STOP) { ret = avs_path_reset(data->path);
ret = avs_path_reset(data->path); if (ret < 0)
if (ret < 0) dev_err(dai->dev, "reset BE path failed: %d\n", ret);
dev_err(dai->dev, "reset BE path failed: %d\n", ret);
}
break; break;
default: default:
...@@ -245,9 +274,11 @@ static const struct snd_soc_dai_ops avs_dai_nonhda_be_ops = { ...@@ -245,9 +274,11 @@ static const struct snd_soc_dai_ops avs_dai_nonhda_be_ops = {
.trigger = avs_dai_nonhda_be_trigger, .trigger = avs_dai_nonhda_be_trigger,
}; };
static const struct snd_soc_dai_ops avs_dai_hda_be_ops;
static int avs_dai_hda_be_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static int avs_dai_hda_be_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
return avs_dai_startup(substream, dai, false); return avs_dai_startup(substream, dai, false, &avs_dai_hda_be_ops);
} }
static void avs_dai_hda_be_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static void avs_dai_hda_be_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
...@@ -343,6 +374,7 @@ static int avs_dai_hda_be_prepare(struct snd_pcm_substream *substream, struct sn ...@@ -343,6 +374,7 @@ static int avs_dai_hda_be_prepare(struct snd_pcm_substream *substream, struct sn
static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd, static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd,
struct snd_soc_dai *dai) struct snd_soc_dai *dai)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct hdac_ext_stream *link_stream; struct hdac_ext_stream *link_stream;
struct avs_dma_data *data; struct avs_dma_data *data;
int ret = 0; int ret = 0;
...@@ -353,15 +385,29 @@ static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd, ...@@ -353,15 +385,29 @@ static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd,
link_stream = substream->runtime->private_data; link_stream = substream->runtime->private_data;
switch (cmd) { switch (cmd) {
case SNDRV_PCM_TRIGGER_RESUME:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_START:
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
snd_hdac_ext_stream_start(link_stream); snd_hdac_ext_stream_start(link_stream);
ret = avs_path_pause(data->path);
if (ret < 0) {
dev_err(dai->dev, "pause BE path failed: %d\n", ret);
break;
}
ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO); ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO);
if (ret < 0) if (ret < 0)
dev_err(dai->dev, "run BE path failed: %d\n", ret); dev_err(dai->dev, "run BE path failed: %d\n", ret);
break; break;
case SNDRV_PCM_TRIGGER_SUSPEND:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_PAUSE_PUSH: case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_STOP:
ret = avs_path_pause(data->path); ret = avs_path_pause(data->path);
...@@ -370,11 +416,9 @@ static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd, ...@@ -370,11 +416,9 @@ static int avs_dai_hda_be_trigger(struct snd_pcm_substream *substream, int cmd,
snd_hdac_ext_stream_clear(link_stream); snd_hdac_ext_stream_clear(link_stream);
if (cmd == SNDRV_PCM_TRIGGER_STOP) { ret = avs_path_reset(data->path);
ret = avs_path_reset(data->path); if (ret < 0)
if (ret < 0) dev_err(dai->dev, "reset BE path failed: %d\n", ret);
dev_err(dai->dev, "reset BE path failed: %d\n", ret);
}
break; break;
default: default:
...@@ -407,6 +451,8 @@ static const struct snd_pcm_hw_constraint_list hw_rates = { ...@@ -407,6 +451,8 @@ static const struct snd_pcm_hw_constraint_list hw_rates = {
.mask = 0, .mask = 0,
}; };
const struct snd_soc_dai_ops avs_dai_fe_ops;
static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
struct snd_pcm_runtime *runtime = substream->runtime; struct snd_pcm_runtime *runtime = substream->runtime;
...@@ -416,7 +462,7 @@ static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_so ...@@ -416,7 +462,7 @@ static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_so
struct hdac_ext_stream *host_stream; struct hdac_ext_stream *host_stream;
int ret; int ret;
ret = avs_dai_startup(substream, dai, true); ret = avs_dai_startup(substream, dai, true, &avs_dai_fe_ops);
if (ret) if (ret)
return ret; return ret;
...@@ -443,8 +489,13 @@ static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_so ...@@ -443,8 +489,13 @@ static int avs_dai_fe_startup(struct snd_pcm_substream *substream, struct snd_so
static void avs_dai_fe_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static void avs_dai_fe_shutdown(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct avs_dev *adev = to_avs_dev(dai->dev);
struct avs_dma_data *data; struct avs_dma_data *data;
if (rtd->dai_link->ignore_suspend)
adev->num_lp_paths--;
data = snd_soc_dai_get_dma_data(dai, substream); data = snd_soc_dai_get_dma_data(dai, substream);
snd_soc_dai_set_dma_data(dai, substream, NULL); snd_soc_dai_set_dma_data(dai, substream, NULL);
...@@ -499,7 +550,7 @@ static int avs_dai_fe_hw_params(struct snd_pcm_substream *substream, ...@@ -499,7 +550,7 @@ static int avs_dai_fe_hw_params(struct snd_pcm_substream *substream,
return ret; return ret;
} }
static int avs_dai_fe_hw_free(struct snd_pcm_substream *substream, struct snd_soc_dai *dai) static int __avs_dai_fe_hw_free(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{ {
struct avs_dma_data *data; struct avs_dma_data *data;
struct hdac_ext_stream *host_stream; struct hdac_ext_stream *host_stream;
...@@ -523,9 +574,15 @@ static int avs_dai_fe_hw_free(struct snd_pcm_substream *substream, struct snd_so ...@@ -523,9 +574,15 @@ static int avs_dai_fe_hw_free(struct snd_pcm_substream *substream, struct snd_so
snd_hdac_stream_cleanup(hdac_stream(host_stream)); snd_hdac_stream_cleanup(hdac_stream(host_stream));
hdac_stream(host_stream)->prepared = false; hdac_stream(host_stream)->prepared = false;
ret = snd_pcm_lib_free_pages(substream); return ret;
if (ret < 0) }
dev_dbg(dai->dev, "Failed to free pages!\n");
static int avs_dai_fe_hw_free(struct snd_pcm_substream *substream, struct snd_soc_dai *dai)
{
int ret;
ret = __avs_dai_fe_hw_free(substream, dai);
snd_pcm_lib_free_pages(substream);
return ret; return ret;
} }
...@@ -571,6 +628,7 @@ static int avs_dai_fe_prepare(struct snd_pcm_substream *substream, struct snd_so ...@@ -571,6 +628,7 @@ static int avs_dai_fe_prepare(struct snd_pcm_substream *substream, struct snd_so
static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *dai) static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, struct snd_soc_dai *dai)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct avs_dma_data *data; struct avs_dma_data *data;
struct hdac_ext_stream *host_stream; struct hdac_ext_stream *host_stream;
struct hdac_bus *bus; struct hdac_bus *bus;
...@@ -582,17 +640,36 @@ static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, stru ...@@ -582,17 +640,36 @@ static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, stru
bus = hdac_stream(host_stream)->bus; bus = hdac_stream(host_stream)->bus;
switch (cmd) { switch (cmd) {
case SNDRV_PCM_TRIGGER_RESUME:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_START: case SNDRV_PCM_TRIGGER_START:
case SNDRV_PCM_TRIGGER_PAUSE_RELEASE: case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
spin_lock_irqsave(&bus->reg_lock, flags); spin_lock_irqsave(&bus->reg_lock, flags);
snd_hdac_stream_start(hdac_stream(host_stream), true); snd_hdac_stream_start(hdac_stream(host_stream), true);
spin_unlock_irqrestore(&bus->reg_lock, flags); spin_unlock_irqrestore(&bus->reg_lock, flags);
/* Timeout on DRSM poll shall not stop the resume so ignore the result. */
if (cmd == SNDRV_PCM_TRIGGER_RESUME)
snd_hdac_stream_wait_drsm(hdac_stream(host_stream));
ret = avs_path_pause(data->path);
if (ret < 0) {
dev_err(dai->dev, "pause FE path failed: %d\n", ret);
break;
}
ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO); ret = avs_path_run(data->path, AVS_TPLG_TRIGGER_AUTO);
if (ret < 0) if (ret < 0)
dev_err(dai->dev, "run FE path failed: %d\n", ret); dev_err(dai->dev, "run FE path failed: %d\n", ret);
break; break;
case SNDRV_PCM_TRIGGER_SUSPEND:
if (rtd->dai_link->ignore_suspend)
break;
fallthrough;
case SNDRV_PCM_TRIGGER_PAUSE_PUSH: case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
case SNDRV_PCM_TRIGGER_STOP: case SNDRV_PCM_TRIGGER_STOP:
ret = avs_path_pause(data->path); ret = avs_path_pause(data->path);
...@@ -603,11 +680,9 @@ static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, stru ...@@ -603,11 +680,9 @@ static int avs_dai_fe_trigger(struct snd_pcm_substream *substream, int cmd, stru
snd_hdac_stream_stop(hdac_stream(host_stream)); snd_hdac_stream_stop(hdac_stream(host_stream));
spin_unlock_irqrestore(&bus->reg_lock, flags); spin_unlock_irqrestore(&bus->reg_lock, flags);
if (cmd == SNDRV_PCM_TRIGGER_STOP) { ret = avs_path_reset(data->path);
ret = avs_path_reset(data->path); if (ret < 0)
if (ret < 0) dev_err(dai->dev, "reset FE path failed: %d\n", ret);
dev_err(dai->dev, "reset FE path failed: %d\n", ret);
}
break; break;
default: default:
...@@ -662,6 +737,7 @@ static int avs_component_load_libraries(struct avs_soc_component *acomp) ...@@ -662,6 +737,7 @@ static int avs_component_load_libraries(struct avs_soc_component *acomp)
if (ret < 0) if (ret < 0)
return ret; return ret;
avs_hda_power_gating_enable(adev, false);
avs_hda_clock_gating_enable(adev, false); avs_hda_clock_gating_enable(adev, false);
avs_hda_l1sen_enable(adev, false); avs_hda_l1sen_enable(adev, false);
...@@ -669,6 +745,7 @@ static int avs_component_load_libraries(struct avs_soc_component *acomp) ...@@ -669,6 +745,7 @@ static int avs_component_load_libraries(struct avs_soc_component *acomp)
avs_hda_l1sen_enable(adev, true); avs_hda_l1sen_enable(adev, true);
avs_hda_clock_gating_enable(adev, true); avs_hda_clock_gating_enable(adev, true);
avs_hda_power_gating_enable(adev, true);
if (!ret) if (!ret)
ret = avs_module_info_init(adev, false); ret = avs_module_info_init(adev, false);
...@@ -752,33 +829,226 @@ static void avs_component_remove(struct snd_soc_component *component) ...@@ -752,33 +829,226 @@ static void avs_component_remove(struct snd_soc_component *component)
} }
} }
static int avs_dai_resume_hw_params(struct snd_soc_dai *dai, struct avs_dma_data *data)
{
struct snd_pcm_substream *substream;
struct snd_soc_pcm_runtime *rtd;
int ret;
substream = data->substream;
rtd = snd_pcm_substream_chip(substream);
ret = dai->driver->ops->hw_params(substream, &rtd->dpcm[substream->stream].hw_params, dai);
if (ret)
dev_err(dai->dev, "hw_params on resume failed: %d\n", ret);
return ret;
}
static int avs_dai_resume_fe_prepare(struct snd_soc_dai *dai, struct avs_dma_data *data)
{
struct hdac_ext_stream *host_stream;
struct hdac_stream *hstream;
struct hdac_bus *bus;
int ret;
host_stream = data->host_stream;
hstream = hdac_stream(host_stream);
bus = hdac_stream(host_stream)->bus;
/* Set DRSM before programming stream and position registers. */
snd_hdac_stream_drsm_enable(bus, true, hstream->index);
ret = dai->driver->ops->prepare(data->substream, dai);
if (ret) {
dev_err(dai->dev, "prepare FE on resume failed: %d\n", ret);
return ret;
}
writel(host_stream->pphcllpl, host_stream->pphc_addr + AZX_REG_PPHCLLPL);
writel(host_stream->pphcllpu, host_stream->pphc_addr + AZX_REG_PPHCLLPU);
writel(host_stream->pphcldpl, host_stream->pphc_addr + AZX_REG_PPHCLDPL);
writel(host_stream->pphcldpu, host_stream->pphc_addr + AZX_REG_PPHCLDPU);
/* As per HW spec recommendation, program LPIB and DPIB to the same value. */
snd_hdac_stream_set_lpib(hstream, hstream->lpib);
snd_hdac_stream_set_dpibr(bus, hstream, hstream->lpib);
return 0;
}
static int avs_dai_resume_be_prepare(struct snd_soc_dai *dai, struct avs_dma_data *data)
{
int ret;
ret = dai->driver->ops->prepare(data->substream, dai);
if (ret)
dev_err(dai->dev, "prepare BE on resume failed: %d\n", ret);
return ret;
}
static int avs_dai_suspend_fe_hw_free(struct snd_soc_dai *dai, struct avs_dma_data *data)
{
struct hdac_ext_stream *host_stream;
int ret;
host_stream = data->host_stream;
/* Store position addresses so we can resume from them later on. */
hdac_stream(host_stream)->lpib = snd_hdac_stream_get_pos_lpib(hdac_stream(host_stream));
host_stream->pphcllpl = readl(host_stream->pphc_addr + AZX_REG_PPHCLLPL);
host_stream->pphcllpu = readl(host_stream->pphc_addr + AZX_REG_PPHCLLPU);
host_stream->pphcldpl = readl(host_stream->pphc_addr + AZX_REG_PPHCLDPL);
host_stream->pphcldpu = readl(host_stream->pphc_addr + AZX_REG_PPHCLDPU);
ret = __avs_dai_fe_hw_free(data->substream, dai);
if (ret < 0)
dev_err(dai->dev, "hw_free FE on suspend failed: %d\n", ret);
return ret;
}
static int avs_dai_suspend_be_hw_free(struct snd_soc_dai *dai, struct avs_dma_data *data)
{
int ret;
ret = dai->driver->ops->hw_free(data->substream, dai);
if (ret < 0)
dev_err(dai->dev, "hw_free BE on suspend failed: %d\n", ret);
return ret;
}
static int avs_component_pm_op(struct snd_soc_component *component, bool be,
int (*op)(struct snd_soc_dai *, struct avs_dma_data *))
{
struct snd_soc_pcm_runtime *rtd;
struct avs_dma_data *data;
struct snd_soc_dai *dai;
int ret;
for_each_component_dais(component, dai) {
data = dai->playback_dma_data;
if (data) {
rtd = snd_pcm_substream_chip(data->substream);
if (rtd->dai_link->no_pcm == be && !rtd->dai_link->ignore_suspend) {
ret = op(dai, data);
if (ret < 0)
return ret;
}
}
data = dai->capture_dma_data;
if (data) {
rtd = snd_pcm_substream_chip(data->substream);
if (rtd->dai_link->no_pcm == be && !rtd->dai_link->ignore_suspend) {
ret = op(dai, data);
if (ret < 0)
return ret;
}
}
}
return 0;
}
static int avs_component_resume_hw_params(struct snd_soc_component *component, bool be)
{
return avs_component_pm_op(component, be, &avs_dai_resume_hw_params);
}
static int avs_component_resume_prepare(struct snd_soc_component *component, bool be)
{
int (*prepare_cb)(struct snd_soc_dai *dai, struct avs_dma_data *data);
if (be)
prepare_cb = &avs_dai_resume_be_prepare;
else
prepare_cb = &avs_dai_resume_fe_prepare;
return avs_component_pm_op(component, be, prepare_cb);
}
static int avs_component_suspend_hw_free(struct snd_soc_component *component, bool be)
{
int (*hw_free_cb)(struct snd_soc_dai *dai, struct avs_dma_data *data);
if (be)
hw_free_cb = &avs_dai_suspend_be_hw_free;
else
hw_free_cb = &avs_dai_suspend_fe_hw_free;
return avs_component_pm_op(component, be, hw_free_cb);
}
static int avs_component_suspend(struct snd_soc_component *component)
{
int ret;
/*
* When freeing paths, FEs need to be first as they perform
* path unbinding.
*/
ret = avs_component_suspend_hw_free(component, false);
if (ret)
return ret;
return avs_component_suspend_hw_free(component, true);
}
static int avs_component_resume(struct snd_soc_component *component)
{
int ret;
/*
* When creating paths, FEs need to be last as they perform
* path binding.
*/
ret = avs_component_resume_hw_params(component, true);
if (ret)
return ret;
ret = avs_component_resume_hw_params(component, false);
if (ret)
return ret;
/* It is expected that the LINK stream is prepared first. */
ret = avs_component_resume_prepare(component, true);
if (ret)
return ret;
return avs_component_resume_prepare(component, false);
}
static const struct snd_pcm_hardware avs_pcm_hardware = {
.info = SNDRV_PCM_INFO_MMAP |
SNDRV_PCM_INFO_MMAP_VALID |
SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_PAUSE |
SNDRV_PCM_INFO_RESUME |
SNDRV_PCM_INFO_NO_PERIOD_WAKEUP,
.formats = SNDRV_PCM_FMTBIT_S16_LE |
SNDRV_PCM_FMTBIT_S24_LE |
SNDRV_PCM_FMTBIT_S32_LE,
.buffer_bytes_max = AZX_MAX_BUF_SIZE,
.period_bytes_min = 128,
.period_bytes_max = AZX_MAX_BUF_SIZE / 2,
.periods_min = 2,
.periods_max = AZX_MAX_FRAG,
.fifo_size = 0,
};
static int avs_component_open(struct snd_soc_component *component, static int avs_component_open(struct snd_soc_component *component,
struct snd_pcm_substream *substream) struct snd_pcm_substream *substream)
{ {
struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream); struct snd_soc_pcm_runtime *rtd = snd_pcm_substream_chip(substream);
struct snd_pcm_hardware hwparams;
/* only FE DAI links are handled here */ /* only FE DAI links are handled here */
if (rtd->dai_link->no_pcm) if (rtd->dai_link->no_pcm)
return 0; return 0;
hwparams.info = SNDRV_PCM_INFO_MMAP | return snd_soc_set_runtime_hwparams(substream, &avs_pcm_hardware);
SNDRV_PCM_INFO_MMAP_VALID |
SNDRV_PCM_INFO_INTERLEAVED |
SNDRV_PCM_INFO_PAUSE |
SNDRV_PCM_INFO_NO_PERIOD_WAKEUP;
hwparams.formats = SNDRV_PCM_FMTBIT_S16_LE |
SNDRV_PCM_FMTBIT_S24_LE |
SNDRV_PCM_FMTBIT_S32_LE;
hwparams.period_bytes_min = 128;
hwparams.period_bytes_max = AZX_MAX_BUF_SIZE / 2;
hwparams.periods_min = 2;
hwparams.periods_max = AZX_MAX_FRAG;
hwparams.buffer_bytes_max = AZX_MAX_BUF_SIZE;
hwparams.fifo_size = 0;
return snd_soc_set_runtime_hwparams(substream, &hwparams);
} }
static unsigned int avs_hda_stream_dpib_read(struct hdac_ext_stream *stream) static unsigned int avs_hda_stream_dpib_read(struct hdac_ext_stream *stream)
...@@ -840,6 +1110,8 @@ static const struct snd_soc_component_driver avs_component_driver = { ...@@ -840,6 +1110,8 @@ static const struct snd_soc_component_driver avs_component_driver = {
.name = "avs-pcm", .name = "avs-pcm",
.probe = avs_component_probe, .probe = avs_component_probe,
.remove = avs_component_remove, .remove = avs_component_remove,
.suspend = avs_component_suspend,
.resume = avs_component_resume,
.open = avs_component_open, .open = avs_component_open,
.pointer = avs_component_pointer, .pointer = avs_component_pointer,
.mmap = avs_component_mmap, .mmap = avs_component_mmap,
...@@ -1120,9 +1392,15 @@ static int avs_component_hda_open(struct snd_soc_component *component, ...@@ -1120,9 +1392,15 @@ static int avs_component_hda_open(struct snd_soc_component *component,
struct hdac_ext_stream *link_stream; struct hdac_ext_stream *link_stream;
struct hda_codec *codec; struct hda_codec *codec;
/* only BE DAI links are handled here */ if (!rtd->dai_link->no_pcm) {
if (!rtd->dai_link->no_pcm) struct snd_pcm_hardware hwparams = avs_pcm_hardware;
return avs_component_open(component, substream);
/* RESUME unsupported for de-coupled HD-Audio capture. */
if (substream->stream == SNDRV_PCM_STREAM_CAPTURE)
hwparams.info &= ~SNDRV_PCM_INFO_RESUME;
return snd_soc_set_runtime_hwparams(substream, &hwparams);
}
codec = dev_to_hda_codec(asoc_rtd_to_codec(rtd, 0)->dev); codec = dev_to_hda_codec(asoc_rtd_to_codec(rtd, 0)->dev);
link_stream = snd_hdac_ext_stream_assign(&codec->bus->core, substream, link_stream = snd_hdac_ext_stream_assign(&codec->bus->core, substream,
...@@ -1155,6 +1433,8 @@ static const struct snd_soc_component_driver avs_hda_component_driver = { ...@@ -1155,6 +1433,8 @@ static const struct snd_soc_component_driver avs_hda_component_driver = {
.name = "avs-hda-pcm", .name = "avs-hda-pcm",
.probe = avs_component_hda_probe, .probe = avs_component_hda_probe,
.remove = avs_component_hda_remove, .remove = avs_component_hda_remove,
.suspend = avs_component_suspend,
.resume = avs_component_resume,
.open = avs_component_hda_open, .open = avs_component_hda_open,
.close = avs_component_hda_close, .close = avs_component_hda_close,
.pointer = avs_component_pointer, .pointer = avs_component_pointer,
......
...@@ -1405,6 +1405,11 @@ static int avs_widget_load(struct snd_soc_component *comp, int index, ...@@ -1405,6 +1405,11 @@ static int avs_widget_load(struct snd_soc_component *comp, int index,
if (!le32_to_cpu(dw->priv.size)) if (!le32_to_cpu(dw->priv.size))
return 0; return 0;
if (w->ignore_suspend && !AVS_S0IX_SUPPORTED) {
dev_info_once(comp->dev, "Device does not support S0IX, check BIOS settings\n");
w->ignore_suspend = false;
}
tplg = acomp->tplg; tplg = acomp->tplg;
mach = dev_get_platdata(comp->card->dev); mach = dev_get_platdata(comp->card->dev);
...@@ -1442,6 +1447,11 @@ static int avs_dai_load(struct snd_soc_component *comp, int index, ...@@ -1442,6 +1447,11 @@ static int avs_dai_load(struct snd_soc_component *comp, int index,
static int avs_link_load(struct snd_soc_component *comp, int index, struct snd_soc_dai_link *link, static int avs_link_load(struct snd_soc_component *comp, int index, struct snd_soc_dai_link *link,
struct snd_soc_tplg_link_config *cfg) struct snd_soc_tplg_link_config *cfg)
{ {
if (link->ignore_suspend && !AVS_S0IX_SUPPORTED) {
dev_info_once(comp->dev, "Device does not support S0IX, check BIOS settings\n");
link->ignore_suspend = false;
}
if (!link->no_pcm) { if (!link->no_pcm) {
/* Stream control handled by IPCs. */ /* Stream control handled by IPCs. */
link->nonatomic = true; link->nonatomic = true;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment