From: Takashi Iwai <tiwai@suse.de>
To: Sasha Levin <sashal@kernel.org>
Cc: alsa-devel@alsa-project.org, linux-kernel@vger.kernel.org,
stable@vger.kernel.org
Subject: Re: [PATCH AUTOSEL 5.6 12/38] ALSA: hda: Skip controller resume if not needed
Date: Fri, 24 Apr 2020 14:44:17 +0200 [thread overview]
Message-ID: <s5himhprr32.wl-tiwai@suse.de> (raw)
In-Reply-To: <20200424122237.9831-12-sashal@kernel.org>
On Fri, 24 Apr 2020 14:22:10 +0200,
Sasha Levin wrote:
>
> From: Takashi Iwai <tiwai@suse.de>
>
> [ Upstream commit c4c8dd6ef807663e42a5f04ea77cd62029eb99fa ]
>
> The HD-audio controller does system-suspend and resume operations by
> directly calling its helpers __azx_runtime_suspend() and
> __azx_runtime_resume(). However, in general, we don't have to resume
> always the device fully at the system resume; typically, if a device
> has been runtime-suspended, we can leave it to runtime resume.
>
> Usually for achieving this, the driver would call
> pm_runtime_force_suspend() and pm_runtime_force_resume() pairs in the
> system suspend and resume ops. Unfortunately, this doesn't work for
> the resume path in our case. For handling the jack detection at the
> system resume, a child codec device may need the (literally) forcibly
> resume even if it's been runtime-suspended, and for that, the
> controller device must be also resumed even if it's been suspended.
>
> This patch is an attempt to improve the situation. It replaces the
> direct __azx_runtime_suspend()/_resume() calls with with
> pm_runtime_force_suspend() and pm_runtime_force_resume() with a slight
> trick as we've done for the codec side. More exactly:
>
> - azx_has_pm_runtime() check is dropped from azx_runtime_suspend() and
> azx_runtime_resume(), so that it can be properly executed from the
> system-suspend/resume path
>
> - The WAKEEN handling depends on the card's power state now; it's set
> and cleared only for the runtime-suspend
>
> - azx_resume() checks whether any codec may need the forcible resume
> beforehand. If the forcible resume is required, it does temporary
> PM refcount up/down for actually triggering the runtime resume.
>
> - A new helper function, hda_codec_need_resume(), is introduced for
> checking whether the codec needs a forcible runtime-resume, and the
> existing code is rewritten with that.
>
> BugLink: https://bugzilla.kernel.org/show_bug.cgi?id=207043
> Link: https://lore.kernel.org/r/20200413082034.25166-6-tiwai@suse.de
> Signed-off-by: Takashi Iwai <tiwai@suse.de>
> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is known to cause a regression, and the fix patch is
included in today's pull request. If we apply this, better to wait
for the next batch including its fix.
thanks,
Takashi
> ---
> include/sound/hda_codec.h | 5 +++++
> sound/pci/hda/hda_codec.c | 2 +-
> sound/pci/hda/hda_intel.c | 38 +++++++++++++++++++++++++++-----------
> 3 files changed, 33 insertions(+), 12 deletions(-)
>
> diff --git a/include/sound/hda_codec.h b/include/sound/hda_codec.h
> index 3ee8036f5436d..225154a4f2ed0 100644
> --- a/include/sound/hda_codec.h
> +++ b/include/sound/hda_codec.h
> @@ -494,6 +494,11 @@ void snd_hda_update_power_acct(struct hda_codec *codec);
> static inline void snd_hda_set_power_save(struct hda_bus *bus, int delay) {}
> #endif
>
> +static inline bool hda_codec_need_resume(struct hda_codec *codec)
> +{
> + return !codec->relaxed_resume && codec->jacktbl.used;
> +}
> +
> #ifdef CONFIG_SND_HDA_PATCH_LOADER
> /*
> * patch firmware
> diff --git a/sound/pci/hda/hda_codec.c b/sound/pci/hda/hda_codec.c
> index 53e7732ef7520..aed1f8188e662 100644
> --- a/sound/pci/hda/hda_codec.c
> +++ b/sound/pci/hda/hda_codec.c
> @@ -2951,7 +2951,7 @@ static int hda_codec_runtime_resume(struct device *dev)
> static int hda_codec_force_resume(struct device *dev)
> {
> struct hda_codec *codec = dev_to_hda_codec(dev);
> - bool forced_resume = !codec->relaxed_resume && codec->jacktbl.used;
> + bool forced_resume = hda_codec_need_resume(codec);
> int ret;
>
> /* The get/put pair below enforces the runtime resume even if the
> diff --git a/sound/pci/hda/hda_intel.c b/sound/pci/hda/hda_intel.c
> index aa0be85614b6c..02c6308502b1e 100644
> --- a/sound/pci/hda/hda_intel.c
> +++ b/sound/pci/hda/hda_intel.c
> @@ -1027,7 +1027,7 @@ static int azx_suspend(struct device *dev)
> chip = card->private_data;
> bus = azx_bus(chip);
> snd_power_change_state(card, SNDRV_CTL_POWER_D3hot);
> - __azx_runtime_suspend(chip);
> + pm_runtime_force_suspend(dev);
> if (bus->irq >= 0) {
> free_irq(bus->irq, chip);
> bus->irq = -1;
> @@ -1044,7 +1044,9 @@ static int azx_suspend(struct device *dev)
> static int azx_resume(struct device *dev)
> {
> struct snd_card *card = dev_get_drvdata(dev);
> + struct hda_codec *codec;
> struct azx *chip;
> + bool forced_resume = false;
>
> if (!azx_is_pm_ready(card))
> return 0;
> @@ -1055,7 +1057,20 @@ static int azx_resume(struct device *dev)
> chip->msi = 0;
> if (azx_acquire_irq(chip, 1) < 0)
> return -EIO;
> - __azx_runtime_resume(chip, false);
> +
> + /* check for the forced resume */
> + list_for_each_codec(codec, &chip->bus) {
> + if (hda_codec_need_resume(codec)) {
> + forced_resume = true;
> + break;
> + }
> + }
> +
> + if (forced_resume)
> + pm_runtime_get_noresume(dev);
> + pm_runtime_force_resume(dev);
> + if (forced_resume)
> + pm_runtime_put(dev);
> snd_power_change_state(card, SNDRV_CTL_POWER_D0);
>
> trace_azx_resume(chip);
> @@ -1102,12 +1117,12 @@ static int azx_runtime_suspend(struct device *dev)
> if (!azx_is_pm_ready(card))
> return 0;
> chip = card->private_data;
> - if (!azx_has_pm_runtime(chip))
> - return 0;
>
> /* enable controller wake up event */
> - azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) |
> - STATESTS_INT_MASK);
> + if (snd_power_get_state(card) == SNDRV_CTL_POWER_D0) {
> + azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) |
> + STATESTS_INT_MASK);
> + }
>
> __azx_runtime_suspend(chip);
> trace_azx_runtime_suspend(chip);
> @@ -1118,17 +1133,18 @@ static int azx_runtime_resume(struct device *dev)
> {
> struct snd_card *card = dev_get_drvdata(dev);
> struct azx *chip;
> + bool from_rt = snd_power_get_state(card) == SNDRV_CTL_POWER_D0;
>
> if (!azx_is_pm_ready(card))
> return 0;
> chip = card->private_data;
> - if (!azx_has_pm_runtime(chip))
> - return 0;
> - __azx_runtime_resume(chip, true);
> + __azx_runtime_resume(chip, from_rt);
>
> /* disable controller Wake Up event*/
> - azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) &
> - ~STATESTS_INT_MASK);
> + if (from_rt) {
> + azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) &
> + ~STATESTS_INT_MASK);
> + }
>
> trace_azx_runtime_resume(chip);
> return 0;
> --
> 2.20.1
>
WARNING: multiple messages have this Message-ID (diff)
From: Takashi Iwai <tiwai@suse.de>
To: Sasha Levin <sashal@kernel.org>
Cc: linux-kernel@vger.kernel.org, stable@vger.kernel.org,
alsa-devel@alsa-project.org
Subject: Re: [PATCH AUTOSEL 5.6 12/38] ALSA: hda: Skip controller resume if not needed
Date: Fri, 24 Apr 2020 14:44:17 +0200 [thread overview]
Message-ID: <s5himhprr32.wl-tiwai@suse.de> (raw)
In-Reply-To: <20200424122237.9831-12-sashal@kernel.org>
On Fri, 24 Apr 2020 14:22:10 +0200,
Sasha Levin wrote:
>
> From: Takashi Iwai <tiwai@suse.de>
>
> [ Upstream commit c4c8dd6ef807663e42a5f04ea77cd62029eb99fa ]
>
> The HD-audio controller does system-suspend and resume operations by
> directly calling its helpers __azx_runtime_suspend() and
> __azx_runtime_resume(). However, in general, we don't have to resume
> always the device fully at the system resume; typically, if a device
> has been runtime-suspended, we can leave it to runtime resume.
>
> Usually for achieving this, the driver would call
> pm_runtime_force_suspend() and pm_runtime_force_resume() pairs in the
> system suspend and resume ops. Unfortunately, this doesn't work for
> the resume path in our case. For handling the jack detection at the
> system resume, a child codec device may need the (literally) forcibly
> resume even if it's been runtime-suspended, and for that, the
> controller device must be also resumed even if it's been suspended.
>
> This patch is an attempt to improve the situation. It replaces the
> direct __azx_runtime_suspend()/_resume() calls with with
> pm_runtime_force_suspend() and pm_runtime_force_resume() with a slight
> trick as we've done for the codec side. More exactly:
>
> - azx_has_pm_runtime() check is dropped from azx_runtime_suspend() and
> azx_runtime_resume(), so that it can be properly executed from the
> system-suspend/resume path
>
> - The WAKEEN handling depends on the card's power state now; it's set
> and cleared only for the runtime-suspend
>
> - azx_resume() checks whether any codec may need the forcible resume
> beforehand. If the forcible resume is required, it does temporary
> PM refcount up/down for actually triggering the runtime resume.
>
> - A new helper function, hda_codec_need_resume(), is introduced for
> checking whether the codec needs a forcible runtime-resume, and the
> existing code is rewritten with that.
>
> BugLink: https://bugzilla.kernel.org/show_bug.cgi?id=207043
> Link: https://lore.kernel.org/r/20200413082034.25166-6-tiwai@suse.de
> Signed-off-by: Takashi Iwai <tiwai@suse.de>
> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is known to cause a regression, and the fix patch is
included in today's pull request. If we apply this, better to wait
for the next batch including its fix.
thanks,
Takashi
> ---
> include/sound/hda_codec.h | 5 +++++
> sound/pci/hda/hda_codec.c | 2 +-
> sound/pci/hda/hda_intel.c | 38 +++++++++++++++++++++++++++-----------
> 3 files changed, 33 insertions(+), 12 deletions(-)
>
> diff --git a/include/sound/hda_codec.h b/include/sound/hda_codec.h
> index 3ee8036f5436d..225154a4f2ed0 100644
> --- a/include/sound/hda_codec.h
> +++ b/include/sound/hda_codec.h
> @@ -494,6 +494,11 @@ void snd_hda_update_power_acct(struct hda_codec *codec);
> static inline void snd_hda_set_power_save(struct hda_bus *bus, int delay) {}
> #endif
>
> +static inline bool hda_codec_need_resume(struct hda_codec *codec)
> +{
> + return !codec->relaxed_resume && codec->jacktbl.used;
> +}
> +
> #ifdef CONFIG_SND_HDA_PATCH_LOADER
> /*
> * patch firmware
> diff --git a/sound/pci/hda/hda_codec.c b/sound/pci/hda/hda_codec.c
> index 53e7732ef7520..aed1f8188e662 100644
> --- a/sound/pci/hda/hda_codec.c
> +++ b/sound/pci/hda/hda_codec.c
> @@ -2951,7 +2951,7 @@ static int hda_codec_runtime_resume(struct device *dev)
> static int hda_codec_force_resume(struct device *dev)
> {
> struct hda_codec *codec = dev_to_hda_codec(dev);
> - bool forced_resume = !codec->relaxed_resume && codec->jacktbl.used;
> + bool forced_resume = hda_codec_need_resume(codec);
> int ret;
>
> /* The get/put pair below enforces the runtime resume even if the
> diff --git a/sound/pci/hda/hda_intel.c b/sound/pci/hda/hda_intel.c
> index aa0be85614b6c..02c6308502b1e 100644
> --- a/sound/pci/hda/hda_intel.c
> +++ b/sound/pci/hda/hda_intel.c
> @@ -1027,7 +1027,7 @@ static int azx_suspend(struct device *dev)
> chip = card->private_data;
> bus = azx_bus(chip);
> snd_power_change_state(card, SNDRV_CTL_POWER_D3hot);
> - __azx_runtime_suspend(chip);
> + pm_runtime_force_suspend(dev);
> if (bus->irq >= 0) {
> free_irq(bus->irq, chip);
> bus->irq = -1;
> @@ -1044,7 +1044,9 @@ static int azx_suspend(struct device *dev)
> static int azx_resume(struct device *dev)
> {
> struct snd_card *card = dev_get_drvdata(dev);
> + struct hda_codec *codec;
> struct azx *chip;
> + bool forced_resume = false;
>
> if (!azx_is_pm_ready(card))
> return 0;
> @@ -1055,7 +1057,20 @@ static int azx_resume(struct device *dev)
> chip->msi = 0;
> if (azx_acquire_irq(chip, 1) < 0)
> return -EIO;
> - __azx_runtime_resume(chip, false);
> +
> + /* check for the forced resume */
> + list_for_each_codec(codec, &chip->bus) {
> + if (hda_codec_need_resume(codec)) {
> + forced_resume = true;
> + break;
> + }
> + }
> +
> + if (forced_resume)
> + pm_runtime_get_noresume(dev);
> + pm_runtime_force_resume(dev);
> + if (forced_resume)
> + pm_runtime_put(dev);
> snd_power_change_state(card, SNDRV_CTL_POWER_D0);
>
> trace_azx_resume(chip);
> @@ -1102,12 +1117,12 @@ static int azx_runtime_suspend(struct device *dev)
> if (!azx_is_pm_ready(card))
> return 0;
> chip = card->private_data;
> - if (!azx_has_pm_runtime(chip))
> - return 0;
>
> /* enable controller wake up event */
> - azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) |
> - STATESTS_INT_MASK);
> + if (snd_power_get_state(card) == SNDRV_CTL_POWER_D0) {
> + azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) |
> + STATESTS_INT_MASK);
> + }
>
> __azx_runtime_suspend(chip);
> trace_azx_runtime_suspend(chip);
> @@ -1118,17 +1133,18 @@ static int azx_runtime_resume(struct device *dev)
> {
> struct snd_card *card = dev_get_drvdata(dev);
> struct azx *chip;
> + bool from_rt = snd_power_get_state(card) == SNDRV_CTL_POWER_D0;
>
> if (!azx_is_pm_ready(card))
> return 0;
> chip = card->private_data;
> - if (!azx_has_pm_runtime(chip))
> - return 0;
> - __azx_runtime_resume(chip, true);
> + __azx_runtime_resume(chip, from_rt);
>
> /* disable controller Wake Up event*/
> - azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) &
> - ~STATESTS_INT_MASK);
> + if (from_rt) {
> + azx_writew(chip, WAKEEN, azx_readw(chip, WAKEEN) &
> + ~STATESTS_INT_MASK);
> + }
>
> trace_azx_runtime_resume(chip);
> return 0;
> --
> 2.20.1
>
next prev parent reply other threads:[~2020-04-24 12:45 UTC|newest]
Thread overview: 62+ messages / expand[flat|nested] mbox.gz Atom feed top
2020-04-24 12:21 [PATCH AUTOSEL 5.6 01/38] libbpf: Initialize *nl_pid so gcc 10 is happy Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 02/38] net: fec: set GPR bit on suspend by DT configuration Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 03/38] x86: hyperv: report value of misc_features Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 04/38] signal: check sig before setting info in kill_pid_usb_asyncio Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 05/38] hwmon: (drivetemp) Use drivetemp's true module name in Kconfig section Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 06/38] hwmon: (drivetemp) Return -ENODATA for invalid temperatures Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 07/38] afs: Fix length of dump of bad YFSFetchStatus record Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 08/38] xfs: acquire superblock freeze protection on eofblocks scans Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 09/38] xfs: fix partially uninitialized structure in xfs_reflink_remap_extent Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 10/38] ALSA: hda: Release resources at error in delayed probe Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 11/38] ALSA: hda: Keep the controller initialization even if no codecs found Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 12/38] ALSA: hda: Skip controller resume if not needed Sasha Levin
2020-04-24 12:44 ` Takashi Iwai [this message]
2020-04-24 12:44 ` Takashi Iwai
2020-04-30 22:36 ` Roy Spliet
2020-05-01 1:17 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 13/38] ALSA: hda: Explicitly permit using autosuspend if runtime PM is supported Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 14/38] drm/amdgpu: fix wrong vram lost counter increment V2 Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 15/38] scsi: target: fix PR IN / READ FULL STATUS for FC Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 16/38] scsi: target: tcmu: reset_ring should reset TCMU_DEV_BIT_BROKEN Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 17/38] clk: asm9260: fix __clk_hw_register_fixed_rate_with_accuracy typo Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 18/38] efi/x86: Don't remap text<->rodata gap read-only for mixed mode Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 19/38] objtool: Fix CONFIG_UBSAN_TRAP unreachable warnings Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 20/38] objtool: Support Clang non-section symbols in ORC dump Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 21/38] xen/xenbus: ensure xenbus_map_ring_valloc() returns proper grant status Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 22/38] ALSA: hda: call runtime_allow() for all hda controllers Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 23/38] net: stmmac: socfpga: Allow all RGMII modes Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 24/38] net/cxgb4: Check the return from t4_query_params properly Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 25/38] mac80211: fix channel switch trigger from unknown mesh peer Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 26/38] sched/isolation: Allow "isolcpus=" to skip unknown sub-parameters Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 27/38] sched/vtime: Work around an unitialized variable warning Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 28/38] i2c: remove i2c_new_probed_device API Sasha Levin
2020-04-24 13:36 ` Wolfram Sang
2020-04-25 1:51 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 29/38] arm64: Delete the space separator in __emit_inst Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 30/38] i2c: tegra: Better handle case where CPU0 is busy for a long time Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-27 7:22 ` Thierry Reding
2020-04-27 7:22 ` Thierry Reding
2020-05-01 0:59 ` Sasha Levin
2020-05-01 0:59 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 31/38] i2c: tegra: Synchronize DMA before termination Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 32/38] ext4: use matching invalidatepage in ext4_writepage Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 33/38] ext4: increase wait time needed before reuse of deleted inode numbers Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 34/38] ext4: convert BUG_ON's to WARN_ON's in mballoc.c Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 35/38] irqchip/gic-v4.1: Add support for VPENDBASER's Dirty+Valid signaling Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 36/38] blk-mq: Put driver tag in blk_mq_dispatch_rq_list() when no budget Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 37/38] irqchip/meson-gpio: Fix HARDIRQ-safe -> HARDIRQ-unsafe lock order Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` Sasha Levin
2020-04-24 12:22 ` [PATCH AUTOSEL 5.6 38/38] hwmon: (jc42) Fix name to have no illegal characters Sasha Levin
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=s5himhprr32.wl-tiwai@suse.de \
--to=tiwai@suse.de \
--cc=alsa-devel@alsa-project.org \
--cc=linux-kernel@vger.kernel.org \
--cc=sashal@kernel.org \
--cc=stable@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.