diff mbox series

ALSA: pcm: Release paused streams before suspend if resume is not supported

Message ID 20250401133652.11617-1-peter.ujfalusi@linux.intel.com (mailing list archive)
State New
Headers show
Series ALSA: pcm: Release paused streams before suspend if resume is not supported | expand

Commit Message

Péter Ujfalusi April 1, 2025, 1:36 p.m. UTC
Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
does not support RESUME then on system resume the RESUME trigger is not
sent, the stream's state and suspended_state remains untouched.
When the user space releases the pause then the core will reject this
because the state of the stream is _not_ PAUSED, it is still SUSPENDED.

From this point user space will do the normal (hw_params) prepare and
START, PAUSE_RELEASE trigger will not be sent by the core after the
system has resumed.

To fix this issue, release the paused stream before handling the suspend
if the resume is not supported.
This will ensure that the stream is properly suspended for suspend entry.
On resume no attempt will be taken to resume the suspended stream as the
resume is not supported, userspace will eventually going restart the audio.

Link: https://github.com/thesofproject/linux/issues/5035
Link: https://github.com/thesofproject/linux/issues/5341
Suggested-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
---
 sound/core/pcm_native.c | 13 +++++++++++++
 1 file changed, 13 insertions(+)

Comments

Takashi Iwai April 1, 2025, 2:49 p.m. UTC | #1
On Tue, 01 Apr 2025 15:36:52 +0200,
Peter Ujfalusi wrote:
> 
> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
> does not support RESUME then on system resume the RESUME trigger is not
> sent, the stream's state and suspended_state remains untouched.
> When the user space releases the pause then the core will reject this
> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
> 
> From this point user space will do the normal (hw_params) prepare and
> START, PAUSE_RELEASE trigger will not be sent by the core after the
> system has resumed.
> 
> To fix this issue, release the paused stream before handling the suspend
> if the resume is not supported.
> This will ensure that the stream is properly suspended for suspend entry.
> On resume no attempt will be taken to resume the suspended stream as the
> resume is not supported, userspace will eventually going restart the audio.
> 
> Link: https://github.com/thesofproject/linux/issues/5035
> Link: https://github.com/thesofproject/linux/issues/5341
> Suggested-by: Takashi Iwai <tiwai@suse.de>
> Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>

Applied now.  Thanks.


Takashi
Jaroslav Kysela April 1, 2025, 4:58 p.m. UTC | #2
On 01. 04. 25 16:49, Takashi Iwai wrote:
> On Tue, 01 Apr 2025 15:36:52 +0200,
> Peter Ujfalusi wrote:
>>
>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
>> does not support RESUME then on system resume the RESUME trigger is not
>> sent, the stream's state and suspended_state remains untouched.
>> When the user space releases the pause then the core will reject this
>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
>>
>>  From this point user space will do the normal (hw_params) prepare and
>> START, PAUSE_RELEASE trigger will not be sent by the core after the
>> system has resumed.

Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger when resume 
is not supported ? I think that the driver has complete information:

PAUSE -> SUSPEND [-> no RESUME]

The driver should stay in the suspend state until a new stream 
re-initialization is invoked.

It looks like that other logic should be moved to the end driver (if some 
parts must be reinitialized in the suspend phase when the resume is not 
supported). I don't think that this is job for PCM core nor SoC PCM core routines.

The SUSPEND trigger is always invoked, isn't?

>> To fix this issue, release the paused stream before handling the suspend
>> if the resume is not supported.
>> This will ensure that the stream is properly suspended for suspend entry.
>> On resume no attempt will be taken to resume the suspended stream as the
>> resume is not supported, userspace will eventually going restart the audio.
>>
>> Link: https://github.com/thesofproject/linux/issues/5035
>> Link: https://github.com/thesofproject/linux/issues/5341
>> Suggested-by: Takashi Iwai <tiwai@suse.de>
>> Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
> 
> Applied now.  Thanks.

I would consider revert if my objection is appropriate.

				Jaroslav
Takashi Iwai April 1, 2025, 5:19 p.m. UTC | #3
On Tue, 01 Apr 2025 18:58:40 +0200,
Jaroslav Kysela wrote:
> 
> On 01. 04. 25 16:49, Takashi Iwai wrote:
> > On Tue, 01 Apr 2025 15:36:52 +0200,
> > Peter Ujfalusi wrote:
> >> 
> >> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
> >> does not support RESUME then on system resume the RESUME trigger is not
> >> sent, the stream's state and suspended_state remains untouched.
> >> When the user space releases the pause then the core will reject this
> >> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
> >> 
> >>  From this point user space will do the normal (hw_params) prepare and
> >> START, PAUSE_RELEASE trigger will not be sent by the core after the
> >> system has resumed.
> 
> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
> when resume is not supported ? I think that the driver has complete
> information:
> 
> PAUSE -> SUSPEND [-> no RESUME]
> 
> The driver should stay in the suspend state until a new stream
> re-initialization is invoked.
> 
> It looks like that other logic should be moved to the end driver (if
> some parts must be reinitialized in the suspend phase when the resume
> is not supported). I don't think that this is job for PCM core nor SoC
> PCM core routines.
> 
> The SUSPEND trigger is always invoked, isn't?

No, that's a part of the problems.  The ops->suspend is called only
for the running state, while at snd_pcm_post_suspend(), the state is
changed to SUSPENDED.  The original state is saved as
suspended_state, and this can be restored via snd_pcm_resume(), but
only if the driver supports the full resume.

We may allow calling ops->suspend also for PAUSED state, too.  But
then each driver would need to have to handle this state change, too.
Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
PAUSED state, the core automatically releases the pause beforehand.

> >> To fix this issue, release the paused stream before handling the suspend
> >> if the resume is not supported.
> >> This will ensure that the stream is properly suspended for suspend entry.
> >> On resume no attempt will be taken to resume the suspended stream as the
> >> resume is not supported, userspace will eventually going restart the audio.
> >> 
> >> Link: https://github.com/thesofproject/linux/issues/5035
> >> Link: https://github.com/thesofproject/linux/issues/5341
> >> Suggested-by: Takashi Iwai <tiwai@suse.de>
> >> Signed-off-by: Peter Ujfalusi <peter.ujfalusi@linux.intel.com>
> > 
> > Applied now.  Thanks.
> 
> I would consider revert if my objection is appropriate.

OK, I dropped the patch for now until we get agreement.


thanks,

Takashi
Jaroslav Kysela April 1, 2025, 6:46 p.m. UTC | #4
On 01. 04. 25 19:19, Takashi Iwai wrote:
> On Tue, 01 Apr 2025 18:58:40 +0200,
> Jaroslav Kysela wrote:
>>
>> On 01. 04. 25 16:49, Takashi Iwai wrote:
>>> On Tue, 01 Apr 2025 15:36:52 +0200,
>>> Peter Ujfalusi wrote:
>>>>
>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
>>>> does not support RESUME then on system resume the RESUME trigger is not
>>>> sent, the stream's state and suspended_state remains untouched.
>>>> When the user space releases the pause then the core will reject this
>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
>>>>
>>>>   From this point user space will do the normal (hw_params) prepare and
>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
>>>> system has resumed.
>>
>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
>> when resume is not supported ? I think that the driver has complete
>> information:
>>
>> PAUSE -> SUSPEND [-> no RESUME]
>>
>> The driver should stay in the suspend state until a new stream
>> re-initialization is invoked.
>>
>> It looks like that other logic should be moved to the end driver (if
>> some parts must be reinitialized in the suspend phase when the resume
>> is not supported). I don't think that this is job for PCM core nor SoC
>> PCM core routines.
>>
>> The SUSPEND trigger is always invoked, isn't?
> 
> No, that's a part of the problems.  The ops->suspend is called only
> for the running state, while at snd_pcm_post_suspend(), the state is
> changed to SUSPENDED.  The original state is saved as
> suspended_state, and this can be restored via snd_pcm_resume(), but
> only if the driver supports the full resume.
> 
> We may allow calling ops->suspend also for PAUSED state, too.  But
> then each driver would need to have to handle this state change, too.
> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
> PAUSED state, the core automatically releases the pause beforehand.

Thanks for those hints. I don't think that the pause release is sufficient
for this case - the stream must be stopped, too. Actually, all mentioned
prepare/drop/drain routines immediately changes state when the pause is
released.

Maybe a new trigger may be added to notify drivers like:

--- a/sound/core/pcm_native.c
+++ b/sound/core/pcm_native.c
@@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
         struct snd_pcm_runtime *runtime = substream->runtime;
         if (runtime->trigger_master != substream)
                 return 0;
-       if (! snd_pcm_running(substream))
+       if (! snd_pcm_running(substream)) {
+               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
                 return 0;
+       }
         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
         runtime->stop_operating = true;
         return 0; /* suspend unconditionally */

With this, the states won't be changed - the driver without resume support
can shut down all necessary things.

					Jaroslav
Takashi Iwai April 1, 2025, 7:27 p.m. UTC | #5
On Tue, 01 Apr 2025 20:46:15 +0200,
Jaroslav Kysela wrote:
> 
> On 01. 04. 25 19:19, Takashi Iwai wrote:
> > On Tue, 01 Apr 2025 18:58:40 +0200,
> > Jaroslav Kysela wrote:
> >> 
> >> On 01. 04. 25 16:49, Takashi Iwai wrote:
> >>> On Tue, 01 Apr 2025 15:36:52 +0200,
> >>> Peter Ujfalusi wrote:
> >>>> 
> >>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
> >>>> does not support RESUME then on system resume the RESUME trigger is not
> >>>> sent, the stream's state and suspended_state remains untouched.
> >>>> When the user space releases the pause then the core will reject this
> >>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
> >>>> 
> >>>>   From this point user space will do the normal (hw_params) prepare and
> >>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
> >>>> system has resumed.
> >> 
> >> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
> >> when resume is not supported ? I think that the driver has complete
> >> information:
> >> 
> >> PAUSE -> SUSPEND [-> no RESUME]
> >> 
> >> The driver should stay in the suspend state until a new stream
> >> re-initialization is invoked.
> >> 
> >> It looks like that other logic should be moved to the end driver (if
> >> some parts must be reinitialized in the suspend phase when the resume
> >> is not supported). I don't think that this is job for PCM core nor SoC
> >> PCM core routines.
> >> 
> >> The SUSPEND trigger is always invoked, isn't?
> > 
> > No, that's a part of the problems.  The ops->suspend is called only
> > for the running state, while at snd_pcm_post_suspend(), the state is
> > changed to SUSPENDED.  The original state is saved as
> > suspended_state, and this can be restored via snd_pcm_resume(), but
> > only if the driver supports the full resume.
> > 
> > We may allow calling ops->suspend also for PAUSED state, too.  But
> > then each driver would need to have to handle this state change, too.
> > Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
> > PAUSED state, the core automatically releases the pause beforehand.
> 
> Thanks for those hints. I don't think that the pause release is sufficient
> for this case - the stream must be stopped, too. Actually, all mentioned
> prepare/drop/drain routines immediately changes state when the pause is
> released.

Actually, this patch follows the same pattern, too.  It calls
snd_pcm_pause(false) to set the state to RUNNING again, then proceed
to the suspend action immediately that sets to SUSPENDED.

> Maybe a new trigger may be added to notify drivers like:
> 
> --- a/sound/core/pcm_native.c
> +++ b/sound/core/pcm_native.c
> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
>         struct snd_pcm_runtime *runtime = substream->runtime;
>         if (runtime->trigger_master != substream)
>                 return 0;
> -       if (! snd_pcm_running(substream))
> +       if (! snd_pcm_running(substream)) {
> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
>                 return 0;
> +       }
>         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>         runtime->stop_operating = true;
>         return 0; /* suspend unconditionally */
> 
> With this, the states won't be changed - the driver without resume support
> can shut down all necessary things.

That's another possibility, yes.  Though, IMO, it makes the
pause-handling a bit more cumbersome.  With Peter's proposal, the
pause state change is always paired, so the driver doesn't consider
about that too much; that's the biggest merit.

But I'm all ears about this issue.  Let's see.

(BTW, I thought the state change PAUSED -> XRUN were possible, but
actually it's not; this is another corner case to be addressed, I
guess.)


Takashi
Péter Ujfalusi April 2, 2025, 6:41 a.m. UTC | #6
Hi,

On 01/04/2025 22:27, Takashi Iwai wrote:
> On Tue, 01 Apr 2025 20:46:15 +0200,
> Jaroslav Kysela wrote:
>>
>> On 01. 04. 25 19:19, Takashi Iwai wrote:
>>> On Tue, 01 Apr 2025 18:58:40 +0200,
>>> Jaroslav Kysela wrote:
>>>>
>>>> On 01. 04. 25 16:49, Takashi Iwai wrote:
>>>>> On Tue, 01 Apr 2025 15:36:52 +0200,
>>>>> Peter Ujfalusi wrote:
>>>>>>
>>>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
>>>>>> does not support RESUME then on system resume the RESUME trigger is not
>>>>>> sent, the stream's state and suspended_state remains untouched.
>>>>>> When the user space releases the pause then the core will reject this
>>>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
>>>>>>
>>>>>>   From this point user space will do the normal (hw_params) prepare and
>>>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
>>>>>> system has resumed.
>>>>
>>>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
>>>> when resume is not supported ? I think that the driver has complete
>>>> information:
>>>>
>>>> PAUSE -> SUSPEND [-> no RESUME]
>>>>
>>>> The driver should stay in the suspend state until a new stream
>>>> re-initialization is invoked.
>>>>
>>>> It looks like that other logic should be moved to the end driver (if
>>>> some parts must be reinitialized in the suspend phase when the resume
>>>> is not supported). I don't think that this is job for PCM core nor SoC
>>>> PCM core routines.
>>>>
>>>> The SUSPEND trigger is always invoked, isn't?
>>>
>>> No, that's a part of the problems.  The ops->suspend is called only
>>> for the running state, while at snd_pcm_post_suspend(), the state is
>>> changed to SUSPENDED.  The original state is saved as
>>> suspended_state, and this can be restored via snd_pcm_resume(), but
>>> only if the driver supports the full resume.
>>>
>>> We may allow calling ops->suspend also for PAUSED state, too.  But
>>> then each driver would need to have to handle this state change, too.
>>> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
>>> PAUSED state, the core automatically releases the pause beforehand.
>>
>> Thanks for those hints. I don't think that the pause release is sufficient
>> for this case - the stream must be stopped, too. Actually, all mentioned
>> prepare/drop/drain routines immediately changes state when the pause is
>> released.
> 
> Actually, this patch follows the same pattern, too.  It calls
> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
> to the suspend action immediately that sets to SUSPENDED.

My original RFC patch did the PAUSE_RELEASE followed by STOP in SOF code
to clear out the paused streams on suspend:
https://lore.kernel.org/linux-sound/20250331105631.7436-1-peter.ujfalusi@linux.intel.com/

Which if I convert it on top of the applied patch would look like this
in core (+ pending comment update):

diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
index a16d15ee98fa..c1bd993c650c 100644
--- a/sound/core/pcm_native.c
+++ b/sound/core/pcm_native.c
@@ -1741,8 +1741,10 @@ static int snd_pcm_suspend(struct
snd_pcm_substream *substream)
 	 * to handle them properly when the system resumes.
 	 */
 	if (runtime->state == SNDRV_PCM_STATE_PAUSED &&
-	    !(runtime->info & SNDRV_PCM_INFO_RESUME))
+	    !(runtime->info & SNDRV_PCM_INFO_RESUME)) {
 		snd_pcm_pause(substream, false);
+		return snd_pcm_stop(substream, SNDRV_PCM_STATE_SETUP);
+	}

 	return snd_pcm_action(&snd_pcm_action_suspend, substream,
 			      ACTION_ARG_IGNORE);


But what I like about the current way is that it is bringing the paused
stream handling in line with other triggers as Takashi-san mentioned.
Paused cannot be directly stopped for example, it needs to be released,
then stopped. Drivers don't need special state machines to handle all
permutations of state changes.

>> Maybe a new trigger may be added to notify drivers like:
>>
>> --- a/sound/core/pcm_native.c
>> +++ b/sound/core/pcm_native.c
>> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
>>         struct snd_pcm_runtime *runtime = substream->runtime;
>>         if (runtime->trigger_master != substream)
>>                 return 0;
>> -       if (! snd_pcm_running(substream))
>> +       if (! snd_pcm_running(substream)) {
>> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
>>                 return 0;
>> +       }
>>         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>>         runtime->stop_operating = true;
>>         return 0; /* suspend unconditionally */
>>
>> With this, the states won't be changed - the driver without resume support
>> can shut down all necessary things.
> 
> That's another possibility, yes.  Though, IMO, it makes the
> pause-handling a bit more cumbersome.  With Peter's proposal, the
> pause state change is always paired, so the driver doesn't consider
> about that too much; that's the biggest merit.

I agree with this sentiment. If we do something like this then all
drivers (which might have similar issue, a note on this later) needs to
be updated and have some magic done, and in case of SOF that would be:
substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_PAUSE_RELEASE);
substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_STOP/SUSPEND);

to handle the messy FE/BE/DSP sequencing.
> But I'm all ears about this issue.  Let's see.
> 
> (BTW, I thought the state change PAUSED -> XRUN were possible, but
> actually it's not; this is another corner case to be addressed, I
> guess.)

Overnight I was thinking of how the suspend while a stream is handled or
can be handled if the PCM supports RESUME.
The paused stream is not a stopped stream and it is not receiving
SUSPEND as it is not running either, but a paused stream might be only
having it's DMA trigger source disabled to stop the hw_ptr moving while
a stopped one have the DMA disabled along with all analog/digital
components.
I'm not sure if sending the SUSPEND trigger to a paused stream is a
solution either as that will surely need different handling as it is not
expected currently (and again SOF have a delicate dance around triggers
starting in ASoC level, but might work).

Note on the pause in general: for few years now pause is in reality a
niche feature and most drivers will never see it used. PA. PW, Chrome,
Android don't use it much or at all, so hitting a suspend while the
stream is paused is a CI thing mostly. mplayer/mpv (kodi might as well)
do pause directly when using ALSA and the PCM supports it, but they work
w/o SNDRV_PCM_INFO_PAUSE just fine.
Jaroslav Kysela April 2, 2025, 8:09 a.m. UTC | #7
On 01. 04. 25 21:27, Takashi Iwai wrote:
> On Tue, 01 Apr 2025 20:46:15 +0200,
> Jaroslav Kysela wrote:
>>
>> On 01. 04. 25 19:19, Takashi Iwai wrote:
>>> On Tue, 01 Apr 2025 18:58:40 +0200,
>>> Jaroslav Kysela wrote:
>>>>
>>>> On 01. 04. 25 16:49, Takashi Iwai wrote:
>>>>> On Tue, 01 Apr 2025 15:36:52 +0200,
>>>>> Peter Ujfalusi wrote:
>>>>>>
>>>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
>>>>>> does not support RESUME then on system resume the RESUME trigger is not
>>>>>> sent, the stream's state and suspended_state remains untouched.
>>>>>> When the user space releases the pause then the core will reject this
>>>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
>>>>>>
>>>>>>    From this point user space will do the normal (hw_params) prepare and
>>>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
>>>>>> system has resumed.
>>>>
>>>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
>>>> when resume is not supported ? I think that the driver has complete
>>>> information:
>>>>
>>>> PAUSE -> SUSPEND [-> no RESUME]
>>>>
>>>> The driver should stay in the suspend state until a new stream
>>>> re-initialization is invoked.
>>>>
>>>> It looks like that other logic should be moved to the end driver (if
>>>> some parts must be reinitialized in the suspend phase when the resume
>>>> is not supported). I don't think that this is job for PCM core nor SoC
>>>> PCM core routines.
>>>>
>>>> The SUSPEND trigger is always invoked, isn't?
>>>
>>> No, that's a part of the problems.  The ops->suspend is called only
>>> for the running state, while at snd_pcm_post_suspend(), the state is
>>> changed to SUSPENDED.  The original state is saved as
>>> suspended_state, and this can be restored via snd_pcm_resume(), but
>>> only if the driver supports the full resume.
>>>
>>> We may allow calling ops->suspend also for PAUSED state, too.  But
>>> then each driver would need to have to handle this state change, too.
>>> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
>>> PAUSED state, the core automatically releases the pause beforehand.
>>
>> Thanks for those hints. I don't think that the pause release is sufficient
>> for this case - the stream must be stopped, too. Actually, all mentioned
>> prepare/drop/drain routines immediately changes state when the pause is
>> released.
> 
> Actually, this patch follows the same pattern, too.  It calls
> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
> to the suspend action immediately that sets to SUSPENDED.

The previous state (suspended_state) will be confusing from the application 
with the proposed patch, because there will be RUNNING state instead PAUSED. 
This previous state is exported via API.

The pause release / stop patterns provide a short time window to process some 
samples which may cause unwanted pops (especially for playback). So perhaps,
we can address this in core (new triggers like PAUSE_RELEASE_AND_STOP), too.

>> Maybe a new trigger may be added to notify drivers like:
>>
>> --- a/sound/core/pcm_native.c
>> +++ b/sound/core/pcm_native.c
>> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
>>          struct snd_pcm_runtime *runtime = substream->runtime;
>>          if (runtime->trigger_master != substream)
>>                  return 0;
>> -       if (! snd_pcm_running(substream))
>> +       if (! snd_pcm_running(substream)) {
>> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
>>                  return 0;
>> +       }
>>          substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>>          runtime->stop_operating = true;
>>          return 0; /* suspend unconditionally */
>>
>> With this, the states won't be changed - the driver without resume support
>> can shut down all necessary things.
> 
> That's another possibility, yes.  Though, IMO, it makes the
> pause-handling a bit more cumbersome.  With Peter's proposal, the
> pause state change is always paired, so the driver doesn't consider
> about that too much; that's the biggest merit.

It depends, if we agree that releasing pause is a extra thing to do when we 
know that this step is just to minimize state changes for drivers.

For my view, the drivers may also receive those triggers:

SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE

This will allow to handle all states properly without any side effects (like 
short unwanted DMA transfers).

The drivers should probably activate those triggers conditionally to not break 
current state sequences.

> (BTW, I thought the state change PAUSED -> XRUN were possible, but
> actually it's not; this is another corner case to be addressed, I
 > guess.)

Which code (lines) do you mean?

					Jaroslav
Jaroslav Kysela April 2, 2025, 8:14 a.m. UTC | #8
On 02. 04. 25 8:41, Péter Ujfalusi wrote:

> I'm not sure if sending the SUSPEND trigger to a paused stream is a
> solution either as that will surely need different handling as it is not
> expected currently (and again SOF have a delicate dance around triggers
> starting in ASoC level, but might work).

It's unfortunate, but things should be resolved in the appropriate code levels 
IMHO. The core code should allow all mechanisms.

> Note on the pause in general: for few years now pause is in reality a
> niche feature and most drivers will never see it used. PA. PW, Chrome,
> Android don't use it much or at all, so hitting a suspend while the
> stream is paused is a CI thing mostly. mplayer/mpv (kodi might as well)
> do pause directly when using ALSA and the PCM supports it, but they work
> w/o SNDRV_PCM_INFO_PAUSE just fine.

We are providing API, so we should not judge, if things are widely used. I 
believe that some users use the perfect pause instead full stream restart.

					Jaroslav
Takashi Iwai April 2, 2025, 8:39 a.m. UTC | #9
On Wed, 02 Apr 2025 10:09:36 +0200,
Jaroslav Kysela wrote:
> 
> On 01. 04. 25 21:27, Takashi Iwai wrote:
> > On Tue, 01 Apr 2025 20:46:15 +0200,
> > Jaroslav Kysela wrote:
> >> 
> >> On 01. 04. 25 19:19, Takashi Iwai wrote:
> >>> On Tue, 01 Apr 2025 18:58:40 +0200,
> >>> Jaroslav Kysela wrote:
> >>>> 
> >>>> On 01. 04. 25 16:49, Takashi Iwai wrote:
> >>>>> On Tue, 01 Apr 2025 15:36:52 +0200,
> >>>>> Peter Ujfalusi wrote:
> >>>>>> 
> >>>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
> >>>>>> does not support RESUME then on system resume the RESUME trigger is not
> >>>>>> sent, the stream's state and suspended_state remains untouched.
> >>>>>> When the user space releases the pause then the core will reject this
> >>>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
> >>>>>> 
> >>>>>>    From this point user space will do the normal (hw_params) prepare and
> >>>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
> >>>>>> system has resumed.
> >>>> 
> >>>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
> >>>> when resume is not supported ? I think that the driver has complete
> >>>> information:
> >>>> 
> >>>> PAUSE -> SUSPEND [-> no RESUME]
> >>>> 
> >>>> The driver should stay in the suspend state until a new stream
> >>>> re-initialization is invoked.
> >>>> 
> >>>> It looks like that other logic should be moved to the end driver (if
> >>>> some parts must be reinitialized in the suspend phase when the resume
> >>>> is not supported). I don't think that this is job for PCM core nor SoC
> >>>> PCM core routines.
> >>>> 
> >>>> The SUSPEND trigger is always invoked, isn't?
> >>> 
> >>> No, that's a part of the problems.  The ops->suspend is called only
> >>> for the running state, while at snd_pcm_post_suspend(), the state is
> >>> changed to SUSPENDED.  The original state is saved as
> >>> suspended_state, and this can be restored via snd_pcm_resume(), but
> >>> only if the driver supports the full resume.
> >>> 
> >>> We may allow calling ops->suspend also for PAUSED state, too.  But
> >>> then each driver would need to have to handle this state change, too.
> >>> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
> >>> PAUSED state, the core automatically releases the pause beforehand.
> >> 
> >> Thanks for those hints. I don't think that the pause release is sufficient
> >> for this case - the stream must be stopped, too. Actually, all mentioned
> >> prepare/drop/drain routines immediately changes state when the pause is
> >> released.
> > 
> > Actually, this patch follows the same pattern, too.  It calls
> > snd_pcm_pause(false) to set the state to RUNNING again, then proceed
> > to the suspend action immediately that sets to SUSPENDED.
> 
> The previous state (suspended_state) will be confusing from the
> application with the proposed patch, because there will be RUNNING
> state instead PAUSED. This previous state is exported via API.

No, it won't happen.  The condition of the new behavior is only when
SNDRV_PCM_INFO_RESUME isn't set -- that is, no resume is possible,
hence no state recovery happens after resume.  From the application
POV, it won't change.

> The pause release / stop patterns provide a short time window to
> process some samples which may cause unwanted pops (especially for
> playback). So perhaps,
> we can address this in core (new triggers like PAUSE_RELEASE_AND_STOP), too.

That's true.  To be noted, the very same thing is applied to other
operations like snd_pcm_drop() or snd_pcm_prepare(), so it's not only
about the case of suspend.

> >> Maybe a new trigger may be added to notify drivers like:
> >> 
> >> --- a/sound/core/pcm_native.c
> >> +++ b/sound/core/pcm_native.c
> >> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
> >>          struct snd_pcm_runtime *runtime = substream->runtime;
> >>          if (runtime->trigger_master != substream)
> >>                  return 0;
> >> -       if (! snd_pcm_running(substream))
> >> +       if (! snd_pcm_running(substream)) {
> >> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
> >>                  return 0;
> >> +       }
> >>          substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
> >>          runtime->stop_operating = true;
> >>          return 0; /* suspend unconditionally */
> >> 
> >> With this, the states won't be changed - the driver without resume support
> >> can shut down all necessary things.
> > 
> > That's another possibility, yes.  Though, IMO, it makes the
> > pause-handling a bit more cumbersome.  With Peter's proposal, the
> > pause state change is always paired, so the driver doesn't consider
> > about that too much; that's the biggest merit.
> 
> It depends, if we agree that releasing pause is a extra thing to do
> when we know that this step is just to minimize state changes for
> drivers.
> 
> For my view, the drivers may also receive those triggers:
> 
> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
> 
> This will allow to handle all states properly without any side effects
> (like short unwanted DMA transfers).
> 
> The drivers should probably activate those triggers conditionally to
> not break current state sequences.

Hmm, that looks too complex, IMO.

Another slightly more straightforward fix would be just to allow the
direct state transition from PAUSED to any state.  Then the remaining
piece is all about driver implementations.  And, this can be
conditionally operated, e.g. only when some extra flag is set.  When
no flag is set, we keep applying PAUSE_RELEASE before the transition
like now, including Peter's patch, that can work generically (but not
always ideally).

> > (BTW, I thought the state change PAUSED -> XRUN were possible, but
> > actually it's not; this is another corner case to be addressed, I
> > guess.)
> 
> Which code (lines) do you mean?

snd_pcm_xrun() and snd_pcm_stop_xrun().  As far as I read the code,
the state transition won't happen when it's in PAUSED state.


thanks,

Takashi
Jaroslav Kysela April 2, 2025, 8:52 a.m. UTC | #10
On 02. 04. 25 10:39, Takashi Iwai wrote:
> On Wed, 02 Apr 2025 10:09:36 +0200,
> Jaroslav Kysela wrote:
>>
>> On 01. 04. 25 21:27, Takashi Iwai wrote:
>>> On Tue, 01 Apr 2025 20:46:15 +0200,
>>> Jaroslav Kysela wrote:
>>>>
>>>> On 01. 04. 25 19:19, Takashi Iwai wrote:
>>>>> On Tue, 01 Apr 2025 18:58:40 +0200,
>>>>> Jaroslav Kysela wrote:
>>>>>>
>>>>>> On 01. 04. 25 16:49, Takashi Iwai wrote:
>>>>>>> On Tue, 01 Apr 2025 15:36:52 +0200,
>>>>>>> Peter Ujfalusi wrote:
>>>>>>>>
>>>>>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
>>>>>>>> does not support RESUME then on system resume the RESUME trigger is not
>>>>>>>> sent, the stream's state and suspended_state remains untouched.
>>>>>>>> When the user space releases the pause then the core will reject this
>>>>>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
>>>>>>>>
>>>>>>>>     From this point user space will do the normal (hw_params) prepare and
>>>>>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
>>>>>>>> system has resumed.
>>>>>>
>>>>>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
>>>>>> when resume is not supported ? I think that the driver has complete
>>>>>> information:
>>>>>>
>>>>>> PAUSE -> SUSPEND [-> no RESUME]
>>>>>>
>>>>>> The driver should stay in the suspend state until a new stream
>>>>>> re-initialization is invoked.
>>>>>>
>>>>>> It looks like that other logic should be moved to the end driver (if
>>>>>> some parts must be reinitialized in the suspend phase when the resume
>>>>>> is not supported). I don't think that this is job for PCM core nor SoC
>>>>>> PCM core routines.
>>>>>>
>>>>>> The SUSPEND trigger is always invoked, isn't?
>>>>>
>>>>> No, that's a part of the problems.  The ops->suspend is called only
>>>>> for the running state, while at snd_pcm_post_suspend(), the state is
>>>>> changed to SUSPENDED.  The original state is saved as
>>>>> suspended_state, and this can be restored via snd_pcm_resume(), but
>>>>> only if the driver supports the full resume.
>>>>>
>>>>> We may allow calling ops->suspend also for PAUSED state, too.  But
>>>>> then each driver would need to have to handle this state change, too.
>>>>> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
>>>>> PAUSED state, the core automatically releases the pause beforehand.
>>>>
>>>> Thanks for those hints. I don't think that the pause release is sufficient
>>>> for this case - the stream must be stopped, too. Actually, all mentioned
>>>> prepare/drop/drain routines immediately changes state when the pause is
>>>> released.
>>>
>>> Actually, this patch follows the same pattern, too.  It calls
>>> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
>>> to the suspend action immediately that sets to SUSPENDED.
>>
>> The previous state (suspended_state) will be confusing from the
>> application with the proposed patch, because there will be RUNNING
>> state instead PAUSED. This previous state is exported via API.
> 
> No, it won't happen.  The condition of the new behavior is only when
> SNDRV_PCM_INFO_RESUME isn't set -- that is, no resume is possible,
> hence no state recovery happens after resume.  From the application
> POV, it won't change.

The suspended_state can be obtained in snd_pcm_status64(). With Peter's patch, 
there will be RUNNING instead PAUSED, don't?

>> The pause release / stop patterns provide a short time window to
>> process some samples which may cause unwanted pops (especially for
>> playback). So perhaps,
>> we can address this in core (new triggers like PAUSE_RELEASE_AND_STOP), too.
> 
> That's true.  To be noted, the very same thing is applied to other
> operations like snd_pcm_drop() or snd_pcm_prepare(), so it's not only
> about the case of suspend.

Yes, I meant all places.

>>>> Maybe a new trigger may be added to notify drivers like:
>>>>
>>>> --- a/sound/core/pcm_native.c
>>>> +++ b/sound/core/pcm_native.c
>>>> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
>>>>           struct snd_pcm_runtime *runtime = substream->runtime;
>>>>           if (runtime->trigger_master != substream)
>>>>                   return 0;
>>>> -       if (! snd_pcm_running(substream))
>>>> +       if (! snd_pcm_running(substream)) {
>>>> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
>>>>                   return 0;
>>>> +       }
>>>>           substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>>>>           runtime->stop_operating = true;
>>>>           return 0; /* suspend unconditionally */
>>>>
>>>> With this, the states won't be changed - the driver without resume support
>>>> can shut down all necessary things.
>>>
>>> That's another possibility, yes.  Though, IMO, it makes the
>>> pause-handling a bit more cumbersome.  With Peter's proposal, the
>>> pause state change is always paired, so the driver doesn't consider
>>> about that too much; that's the biggest merit.
>>
>> It depends, if we agree that releasing pause is a extra thing to do
>> when we know that this step is just to minimize state changes for
>> drivers.
>>
>> For my view, the drivers may also receive those triggers:
>>
>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
>> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
>> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
>>
>> This will allow to handle all states properly without any side effects
>> (like short unwanted DMA transfers).
>>
>> The drivers should probably activate those triggers conditionally to
>> not break current state sequences.
> 
> Hmm, that looks too complex, IMO.
> 
> Another slightly more straightforward fix would be just to allow the
> direct state transition from PAUSED to any state.  Then the remaining
> piece is all about driver implementations.  And, this can be
> conditionally operated, e.g. only when some extra flag is set.  When
> no flag is set, we keep applying PAUSE_RELEASE before the transition
> like now, including Peter's patch, that can work generically (but not
> always ideally).

It looks also feasible. My proposal just allows the straight state 
identification from the trigger value in the drivers.

>>> (BTW, I thought the state change PAUSED -> XRUN were possible, but
>>> actually it's not; this is another corner case to be addressed, I
>>> guess.)
>>
>> Which code (lines) do you mean?
> 
> snd_pcm_xrun() and snd_pcm_stop_xrun().  As far as I read the code,
> the state transition won't happen when it's in PAUSED state.

Yes, paused state should be handled there, too. But the pause release 
operation looks really suspicious for xrun.

					Jaroslav
Takashi Iwai April 2, 2025, 9:16 a.m. UTC | #11
On Wed, 02 Apr 2025 10:52:16 +0200,
Jaroslav Kysela wrote:
> 
> On 02. 04. 25 10:39, Takashi Iwai wrote:
> > On Wed, 02 Apr 2025 10:09:36 +0200,
> > Jaroslav Kysela wrote:
> >> 
> >> On 01. 04. 25 21:27, Takashi Iwai wrote:
> >>> On Tue, 01 Apr 2025 20:46:15 +0200,
> >>> Jaroslav Kysela wrote:
> >>>> 
> >>>> On 01. 04. 25 19:19, Takashi Iwai wrote:
> >>>>> On Tue, 01 Apr 2025 18:58:40 +0200,
> >>>>> Jaroslav Kysela wrote:
> >>>>>> 
> >>>>>> On 01. 04. 25 16:49, Takashi Iwai wrote:
> >>>>>>> On Tue, 01 Apr 2025 15:36:52 +0200,
> >>>>>>> Peter Ujfalusi wrote:
> >>>>>>>> 
> >>>>>>>> Streams moved to SUSPENDED state from PAUSED without trigger. If a stream
> >>>>>>>> does not support RESUME then on system resume the RESUME trigger is not
> >>>>>>>> sent, the stream's state and suspended_state remains untouched.
> >>>>>>>> When the user space releases the pause then the core will reject this
> >>>>>>>> because the state of the stream is _not_ PAUSED, it is still SUSPENDED.
> >>>>>>>> 
> >>>>>>>>     From this point user space will do the normal (hw_params) prepare and
> >>>>>>>> START, PAUSE_RELEASE trigger will not be sent by the core after the
> >>>>>>>> system has resumed.
> >>>>>> 
> >>>>>> Why you expect to see PAUSE_RELEASE trigger before SUSPEND trigger
> >>>>>> when resume is not supported ? I think that the driver has complete
> >>>>>> information:
> >>>>>> 
> >>>>>> PAUSE -> SUSPEND [-> no RESUME]
> >>>>>> 
> >>>>>> The driver should stay in the suspend state until a new stream
> >>>>>> re-initialization is invoked.
> >>>>>> 
> >>>>>> It looks like that other logic should be moved to the end driver (if
> >>>>>> some parts must be reinitialized in the suspend phase when the resume
> >>>>>> is not supported). I don't think that this is job for PCM core nor SoC
> >>>>>> PCM core routines.
> >>>>>> 
> >>>>>> The SUSPEND trigger is always invoked, isn't?
> >>>>> 
> >>>>> No, that's a part of the problems.  The ops->suspend is called only
> >>>>> for the running state, while at snd_pcm_post_suspend(), the state is
> >>>>> changed to SUSPENDED.  The original state is saved as
> >>>>> suspended_state, and this can be restored via snd_pcm_resume(), but
> >>>>> only if the driver supports the full resume.
> >>>>> 
> >>>>> We may allow calling ops->suspend also for PAUSED state, too.  But
> >>>>> then each driver would need to have to handle this state change, too.
> >>>>> Currently, when snd_pcm_prepare(), *_drop() or *_drain() is called at
> >>>>> PAUSED state, the core automatically releases the pause beforehand.
> >>>> 
> >>>> Thanks for those hints. I don't think that the pause release is sufficient
> >>>> for this case - the stream must be stopped, too. Actually, all mentioned
> >>>> prepare/drop/drain routines immediately changes state when the pause is
> >>>> released.
> >>> 
> >>> Actually, this patch follows the same pattern, too.  It calls
> >>> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
> >>> to the suspend action immediately that sets to SUSPENDED.
> >> 
> >> The previous state (suspended_state) will be confusing from the
> >> application with the proposed patch, because there will be RUNNING
> >> state instead PAUSED. This previous state is exported via API.
> > 
> > No, it won't happen.  The condition of the new behavior is only when
> > SNDRV_PCM_INFO_RESUME isn't set -- that is, no resume is possible,
> > hence no state recovery happens after resume.  From the application
> > POV, it won't change.
> 
> The suspended_state can be obtained in snd_pcm_status64(). With
> Peter's patch, there will be RUNNING instead PAUSED, don't?

I see, that should be better fixed.

> >> The pause release / stop patterns provide a short time window to
> >> process some samples which may cause unwanted pops (especially for
> >> playback). So perhaps,
> >> we can address this in core (new triggers like PAUSE_RELEASE_AND_STOP), too.
> > 
> > That's true.  To be noted, the very same thing is applied to other
> > operations like snd_pcm_drop() or snd_pcm_prepare(), so it's not only
> > about the case of suspend.
> 
> Yes, I meant all places.
> 
> >>>> Maybe a new trigger may be added to notify drivers like:
> >>>> 
> >>>> --- a/sound/core/pcm_native.c
> >>>> +++ b/sound/core/pcm_native.c
> >>>> @@ -1694,8 +1694,10 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
> >>>>           struct snd_pcm_runtime *runtime = substream->runtime;
> >>>>           if (runtime->trigger_master != substream)
> >>>>                   return 0;
> >>>> -       if (! snd_pcm_running(substream))
> >>>> +       if (! snd_pcm_running(substream)) {
> >>>> +               substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND_WHEN_IDLE);
> >>>>                   return 0;
> >>>> +       }
> >>>>           substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
> >>>>           runtime->stop_operating = true;
> >>>>           return 0; /* suspend unconditionally */
> >>>> 
> >>>> With this, the states won't be changed - the driver without resume support
> >>>> can shut down all necessary things.
> >>> 
> >>> That's another possibility, yes.  Though, IMO, it makes the
> >>> pause-handling a bit more cumbersome.  With Peter's proposal, the
> >>> pause state change is always paired, so the driver doesn't consider
> >>> about that too much; that's the biggest merit.
> >> 
> >> It depends, if we agree that releasing pause is a extra thing to do
> >> when we know that this step is just to minimize state changes for
> >> drivers.
> >> 
> >> For my view, the drivers may also receive those triggers:
> >> 
> >> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
> >> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
> >> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
> >> 
> >> This will allow to handle all states properly without any side effects
> >> (like short unwanted DMA transfers).
> >> 
> >> The drivers should probably activate those triggers conditionally to
> >> not break current state sequences.
> > 
> > Hmm, that looks too complex, IMO.
> > 
> > Another slightly more straightforward fix would be just to allow the
> > direct state transition from PAUSED to any state.  Then the remaining
> > piece is all about driver implementations.  And, this can be
> > conditionally operated, e.g. only when some extra flag is set.  When
> > no flag is set, we keep applying PAUSE_RELEASE before the transition
> > like now, including Peter's patch, that can work generically (but not
> > always ideally).
> 
> It looks also feasible. My proposal just allows the straight state
> identification from the trigger value in the drivers.

On the second thought, yet another variant would be to introduce only
one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
PAUSED state as a step before stopping the stream.  This can be
applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
If the driver doesn't support this mode, the PCM core may fall back to
SNDRV_PCM_TRIGGER_PAUSE_RELEASE.

> >>> (BTW, I thought the state change PAUSED -> XRUN were possible, but
> >>> actually it's not; this is another corner case to be addressed, I
> >>> guess.)
> >> 
> >> Which code (lines) do you mean?
> > 
> > snd_pcm_xrun() and snd_pcm_stop_xrun().  As far as I read the code,
> > the state transition won't happen when it's in PAUSED state.
> 
> Yes, paused state should be handled there, too. But the pause release
> operation looks really suspicious for xrun.

Yeah, I noticed it after sending out my previous post, too.
The engine is stopped, so practically the XRUN must not happen.
I believe we can forget about that.


thanks,

Takashi
Péter Ujfalusi April 2, 2025, 9:28 a.m. UTC | #12
On 02/04/2025 11:52, Jaroslav Kysela wrote:
>>>> Actually, this patch follows the same pattern, too.  It calls
>>>> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
>>>> to the suspend action immediately that sets to SUSPENDED.
>>>
>>> The previous state (suspended_state) will be confusing from the
>>> application with the proposed patch, because there will be RUNNING
>>> state instead PAUSED. This previous state is exported via API.
>>
>> No, it won't happen.  The condition of the new behavior is only when
>> SNDRV_PCM_INFO_RESUME isn't set -- that is, no resume is possible,
>> hence no state recovery happens after resume.  From the application
>> POV, it won't change.
> 
> The suspended_state can be obtained in snd_pcm_status64(). With Peter's
> patch, there will be RUNNING instead PAUSED, don't?

Yes, that is true, but it does not matter.
if the SNDRV_PCM_INFO_RESUME is not set, on resume nothing is going to
be done, the state remains SUSPENDED and the suspended_substate also
retained, so:

The stream was in RUNNING, after suspend:
state = SUSPENDED, suspended_state = RUNNING
After resume:
state = SUSPENDED, suspended_state = RUNNING
The the stream goes to xrun and got restarted.

The stream was in RUNNING, after suspend:
Without my patch / with my patch:
state = SUSPENDED, suspended_state = PAUSED / RUNNING
After resume:
state = SUSPENDED, suspended_state = RUNNING / RUNNING

The stream remains "not running" as resume is not supported and
application knows that it had left the stream paused.

When it tries to PAUSE_RELEASE the stream, in both case the core will
look at the state, which is SUSPENDED and thus the pause release fails
(it is not in paused state), so in both cases we go under a restart
(suspended_state is reset, ignored).

>>>> That's another possibility, yes.  Though, IMO, it makes the
>>>> pause-handling a bit more cumbersome.  With Peter's proposal, the
>>>> pause state change is always paired, so the driver doesn't consider
>>>> about that too much; that's the biggest merit.
>>>
>>> It depends, if we agree that releasing pause is a extra thing to do
>>> when we know that this step is just to minimize state changes for
>>> drivers.
>>>
>>> For my view, the drivers may also receive those triggers:
>>>
>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
>>> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
>>> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
>>>
>>> This will allow to handle all states properly without any side effects
>>> (like short unwanted DMA transfers).
>>>
>>> The drivers should probably activate those triggers conditionally to
>>> not break current state sequences.
>>
>> Hmm, that looks too complex, IMO.
>>
>> Another slightly more straightforward fix would be just to allow the
>> direct state transition from PAUSED to any state.  Then the remaining
>> piece is all about driver implementations.  And, this can be
>> conditionally operated, e.g. only when some extra flag is set.  When
>> no flag is set, we keep applying PAUSE_RELEASE before the transition
>> like now, including Peter's patch, that can work generically (but not
>> always ideally).
> 
> It looks also feasible. My proposal just allows the straight state
> identification from the trigger value in the drivers.

We could also add a flag that the driver can say that 'I know what I'm
doing' and in that case send just simple SUSPEND and trust the driver
that it really knows what it is doing.
However if the driver does not support RESUME, then things get a bit
complicated. I'm not entirely sure what drivers do when they support
RESUME and they went to suspend with paused state...

But, this is not that simple in ASoC, as there we can have at least four
drivers involved (machine, platform, cpu dai, codec), all of them has to
support this new fine graded state dependent trigger, right?
Jaroslav Kysela April 2, 2025, 10:45 a.m. UTC | #13
On 02. 04. 25 11:16, Takashi Iwai wrote:

>>>> For my view, the drivers may also receive those triggers:
>>>>
>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
>>>> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
>>>> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
>>>>
>>>> This will allow to handle all states properly without any side effects
>>>> (like short unwanted DMA transfers).
>>>>
>>>> The drivers should probably activate those triggers conditionally to
>>>> not break current state sequences.
>>>
>>> Hmm, that looks too complex, IMO.
>>>
>>> Another slightly more straightforward fix would be just to allow the
>>> direct state transition from PAUSED to any state.  Then the remaining
>>> piece is all about driver implementations.  And, this can be
>>> conditionally operated, e.g. only when some extra flag is set.  When
>>> no flag is set, we keep applying PAUSE_RELEASE before the transition
>>> like now, including Peter's patch, that can work generically (but not
>>> always ideally).
>>
>> It looks also feasible. My proposal just allows the straight state
>> identification from the trigger value in the drivers.
> 
> On the second thought, yet another variant would be to introduce only
> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
> PAUSED state as a step before stopping the stream.  This can be
> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
> If the driver doesn't support this mode, the PCM core may fall back to
> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.

This looks identical to my proposed SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP 
which is more explanatory. It's not clear what PAUSE_RESET should do (probably 
the stream should be kept inactive).

					Jaroslav
Takashi Iwai April 2, 2025, 11:21 a.m. UTC | #14
On Wed, 02 Apr 2025 12:45:36 +0200,
Jaroslav Kysela wrote:
> 
> On 02. 04. 25 11:16, Takashi Iwai wrote:
> 
> >>>> For my view, the drivers may also receive those triggers:
> >>>> 
> >>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP
> >>>> SNDRV_PCM_TRIGGER_SUSPEND_IN_PAUSE
> >>>> SNDRV_PCM_TRIGGER_RELEASE_IN_PAUSE
> >>>> 
> >>>> This will allow to handle all states properly without any side effects
> >>>> (like short unwanted DMA transfers).
> >>>> 
> >>>> The drivers should probably activate those triggers conditionally to
> >>>> not break current state sequences.
> >>> 
> >>> Hmm, that looks too complex, IMO.
> >>> 
> >>> Another slightly more straightforward fix would be just to allow the
> >>> direct state transition from PAUSED to any state.  Then the remaining
> >>> piece is all about driver implementations.  And, this can be
> >>> conditionally operated, e.g. only when some extra flag is set.  When
> >>> no flag is set, we keep applying PAUSE_RELEASE before the transition
> >>> like now, including Peter's patch, that can work generically (but not
> >>> always ideally).
> >> 
> >> It looks also feasible. My proposal just allows the straight state
> >> identification from the trigger value in the drivers.
> > 
> > On the second thought, yet another variant would be to introduce only
> > one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
> > PAUSED state as a step before stopping the stream.  This can be
> > applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
> > If the driver doesn't support this mode, the PCM core may fall back to
> > SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
> 
> This looks identical to my proposed
> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
> explanatory. It's not clear what PAUSE_RESET should do (probably the
> stream should be kept inactive).

There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
doesn't have to stop the stream by itself, but it's only a preparation
for stopping.  That is, a proper TRIGGER_STOP always follows after
TRIGGER_PAUSE_RESET.


thanks,

Takashi
Takashi Iwai April 2, 2025, 11:27 a.m. UTC | #15
On Wed, 02 Apr 2025 11:28:55 +0200,
Péter Ujfalusi wrote:
> 
> 
> On 02/04/2025 11:52, Jaroslav Kysela wrote:
> >>>> Actually, this patch follows the same pattern, too.  It calls
> >>>> snd_pcm_pause(false) to set the state to RUNNING again, then proceed
> >>>> to the suspend action immediately that sets to SUSPENDED.
> >>>
> >>> The previous state (suspended_state) will be confusing from the
> >>> application with the proposed patch, because there will be RUNNING
> >>> state instead PAUSED. This previous state is exported via API.
> >>
> >> No, it won't happen.  The condition of the new behavior is only when
> >> SNDRV_PCM_INFO_RESUME isn't set -- that is, no resume is possible,
> >> hence no state recovery happens after resume.  From the application
> >> POV, it won't change.
> > 
> > The suspended_state can be obtained in snd_pcm_status64(). With Peter's
> > patch, there will be RUNNING instead PAUSED, don't?
> 
> Yes, that is true, but it does not matter.
> if the SNDRV_PCM_INFO_RESUME is not set, on resume nothing is going to
> be done, the state remains SUSPENDED and the suspended_substate also
> retained, so:
> 
> The stream was in RUNNING, after suspend:
> state = SUSPENDED, suspended_state = RUNNING
> After resume:
> state = SUSPENDED, suspended_state = RUNNING
> The the stream goes to xrun and got restarted.
> 
> The stream was in RUNNING, after suspend:
> Without my patch / with my patch:
> state = SUSPENDED, suspended_state = PAUSED / RUNNING
> After resume:
> state = SUSPENDED, suspended_state = RUNNING / RUNNING
> 
> The stream remains "not running" as resume is not supported and
> application knows that it had left the stream paused.
> 
> When it tries to PAUSE_RELEASE the stream, in both case the core will
> look at the state, which is SUSPENDED and thus the pause release fails
> (it is not in paused state), so in both cases we go under a restart
> (suspended_state is reset, ignored).

Well, the suspended_state itself is set no matter whether the resume
is supported or not.  The difference is that it's evaluated in the
driver only by the suspended state, but it's free for user-space to
read this out from the output of snd_pcm_status ioctl.
How seriously it needs to be a different question, but there is a
possibility of breakage.

Though, looking through the code, the suspended_state isn't updated at
all except for the suspend, and it means this can keep a bogus state,
too.  We should fix the code to clear the suspended_state at init
phase.  It's another corner case.


thanks,

Takashi
Péter Ujfalusi April 2, 2025, 11:43 a.m. UTC | #16
On 02/04/2025 14:21, Takashi Iwai wrote:
>>> On the second thought, yet another variant would be to introduce only
>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
>>> PAUSED state as a step before stopping the stream.  This can be
>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
>>> If the driver doesn't support this mode, the PCM core may fall back to
>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
>>
>> This looks identical to my proposed
>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
>> explanatory. It's not clear what PAUSE_RESET should do (probably the
>> stream should be kept inactive).
> 
> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
> doesn't have to stop the stream by itself, but it's only a preparation
> for stopping.  That is, a proper TRIGGER_STOP always follows after
> TRIGGER_PAUSE_RESET.

But what state that we move after we reset the PAUSE? is it RUNNING
(iow, equals to PAUSE_RELEASE) or something else?

In case of suspend while the state is paused we will:
TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?

Most drivers are doing the same thing for STOP and SUSPEND trigger,
however it is possible to do extra steps for SUSPEND to make sure that
the whole system is in a lowest power state.
Takashi Iwai April 2, 2025, 11:50 a.m. UTC | #17
On Wed, 02 Apr 2025 13:43:57 +0200,
Péter Ujfalusi wrote:
> 
> 
> 
> On 02/04/2025 14:21, Takashi Iwai wrote:
> >>> On the second thought, yet another variant would be to introduce only
> >>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
> >>> PAUSED state as a step before stopping the stream.  This can be
> >>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
> >>> If the driver doesn't support this mode, the PCM core may fall back to
> >>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
> >>
> >> This looks identical to my proposed
> >> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
> >> explanatory. It's not clear what PAUSE_RESET should do (probably the
> >> stream should be kept inactive).
> > 
> > There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
> > doesn't have to stop the stream by itself, but it's only a preparation
> > for stopping.  That is, a proper TRIGGER_STOP always follows after
> > TRIGGER_PAUSE_RESET.
> 
> But what state that we move after we reset the PAUSE? is it RUNNING
> (iow, equals to PAUSE_RELEASE) or something else?

It'll be temporarily set to RUNNING.

> In case of suspend while the state is paused we will:
> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?

For suspend, PAUSE_RESET, then SUSPEND.
For dropping, PAUSE_RESET, then STOP.

> Most drivers are doing the same thing for STOP and SUSPEND trigger,
> however it is possible to do extra steps for SUSPEND to make sure that
> the whole system is in a lowest power state.

Yes, that's the reason I proposed PAUSE_RESET.  Then the driver can
handle the following stop or suspend operation differently.

That said, PAUSE_RESET is just another variant of PAUSE_RELEASE.
But this shouldn't actually enable the audio output again, but prepare
for the next SUSPEND or STOP trigger.


Takashi
Takashi Iwai April 2, 2025, 11:53 a.m. UTC | #18
On Wed, 02 Apr 2025 13:27:09 +0200,
Takashi Iwai wrote:
> 
> ... The difference is that it's evaluated in the
> driver only by the suspended state,

Here I meant "only by the resume operation".

> but it's free for user-space to
> read this out from the output of snd_pcm_status ioctl.
> How seriously it needs to be a different question, but there is a
> possibility of breakage.


Takashi
Péter Ujfalusi April 2, 2025, 12:52 p.m. UTC | #19
On 02/04/2025 14:50, Takashi Iwai wrote:
> On Wed, 02 Apr 2025 13:43:57 +0200,
> Péter Ujfalusi wrote:
>>
>>
>>
>> On 02/04/2025 14:21, Takashi Iwai wrote:
>>>>> On the second thought, yet another variant would be to introduce only
>>>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
>>>>> PAUSED state as a step before stopping the stream.  This can be
>>>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
>>>>> If the driver doesn't support this mode, the PCM core may fall back to
>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
>>>>
>>>> This looks identical to my proposed
>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
>>>> explanatory. It's not clear what PAUSE_RESET should do (probably the
>>>> stream should be kept inactive).
>>>
>>> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
>>> doesn't have to stop the stream by itself, but it's only a preparation
>>> for stopping.  That is, a proper TRIGGER_STOP always follows after
>>> TRIGGER_PAUSE_RESET.
>>
>> But what state that we move after we reset the PAUSE? is it RUNNING
>> (iow, equals to PAUSE_RELEASE) or something else?
> 
> It'll be temporarily set to RUNNING.
> 
>> In case of suspend while the state is paused we will:
>> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?
> 
> For suspend, PAUSE_RESET, then SUSPEND.
> For dropping, PAUSE_RESET, then STOP.

OK, but this is still the same as PAUSE_RELEASE+SUSPEND when it comes to
suspended_state, no?

I mean you need to move to RUNNING to be suspend, or stop to skip the
suspend.
>> Most drivers are doing the same thing for STOP and SUSPEND trigger,
>> however it is possible to do extra steps for SUSPEND to make sure that
>> the whole system is in a lowest power state.
> 
> Yes, that's the reason I proposed PAUSE_RESET.  Then the driver can
> handle the following stop or suspend operation differently.
> 
> That said, PAUSE_RESET is just another variant of PAUSE_RELEASE.
> But this shouldn't actually enable the audio output again, but prepare
> for the next SUSPEND or STOP trigger.

I'm not sure how this would be seen in ASoC level. Surely we need to
have flag per components that it is handling PAUSE_RESET as codecs would
like don't want to do it or they migh, hm.
But ASoC maintains another set of state for the dpcm:
SND_SOC_DPCM_STATE_* and it has this as well among other state-machine
things:
case SNDRV_PCM_TRIGGER_SUSPEND:
	if (be->dpcm[stream].state != SND_SOC_DPCM_STATE_START)
		goto next;

So, this is dpcm level, but one component can support PAUSE_RESET while
the other does not and the transitions are not too clear to me.

But the idea of resurrecting a paused stream with mute held is not bad.

I'm still uncertain how this actually works now on systems where RESUME
is supported and we suspend while a PCM is in paused...
Jaroslav Kysela April 2, 2025, 12:55 p.m. UTC | #20
On 02. 04. 25 13:50, Takashi Iwai wrote:
> On Wed, 02 Apr 2025 13:43:57 +0200,
> Péter Ujfalusi wrote:
>>
>>
>>
>> On 02/04/2025 14:21, Takashi Iwai wrote:
>>>>> On the second thought, yet another variant would be to introduce only
>>>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
>>>>> PAUSED state as a step before stopping the stream.  This can be
>>>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
>>>>> If the driver doesn't support this mode, the PCM core may fall back to
>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
>>>>
>>>> This looks identical to my proposed
>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
>>>> explanatory. It's not clear what PAUSE_RESET should do (probably the
>>>> stream should be kept inactive).
>>>
>>> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
>>> doesn't have to stop the stream by itself, but it's only a preparation
>>> for stopping.  That is, a proper TRIGGER_STOP always follows after
>>> TRIGGER_PAUSE_RESET.
>>
>> But what state that we move after we reset the PAUSE? is it RUNNING
>> (iow, equals to PAUSE_RELEASE) or something else?
> 
> It'll be temporarily set to RUNNING.

I don't think that there's requirement to change state at this point. It would 
be better to keep PAUSED state and add this state to the running state checks. 
Something like:

diff --git a/include/sound/pcm.h b/include/sound/pcm.h
index ac8f3aef9205..166ef0376293 100644
--- a/include/sound/pcm.h
+++ b/include/sound/pcm.h
@@ -708,6 +708,12 @@ static inline int snd_pcm_running(struct 
snd_pcm_substream *substream)
                  substream->stream == SNDRV_PCM_STREAM_PLAYBACK));
  }

+static inline int snd_pcm_active(struct snd_pcm_substream *substream)
+{
+       return snd_pcm_running(substream) ||
+               substream->runtime->state == SNDRV_PCM_STATE_PAUSED;
+}
+
  /**
   * __snd_pcm_set_state - Change the current PCM state
   * @runtime: PCM runtime to set

But...

>> In case of suspend while the state is paused we will:
>> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?
> 
> For suspend, PAUSE_RESET, then SUSPEND.
> For dropping, PAUSE_RESET, then STOP.

I'm not convinced if the functionality/information split is a good idea in 
this case. The preparation (code) for suspend and stop may be different. With 
separate triggers, drivers know exactly about the requested behavior (and we 
know it at trigger call time already).

Perhaps, more universal triggers like SNDRV_PCM_TRIGGER_STOP_PREPARE and
SNDRV_PCM_TRIGGER_SUSPEND_PREPARE may be invoked (always) before STOP/PREPARE 
triggers  Drivers should handle also the pause state there on it's own to keep 
the DMA transfers in idle for those cases.

						Jaroslav
Takashi Iwai April 2, 2025, 1:23 p.m. UTC | #21
On Wed, 02 Apr 2025 14:52:48 +0200,
Péter Ujfalusi wrote:
> 
> 
> 
> On 02/04/2025 14:50, Takashi Iwai wrote:
> > On Wed, 02 Apr 2025 13:43:57 +0200,
> > Péter Ujfalusi wrote:
> >>
> >>
> >>
> >> On 02/04/2025 14:21, Takashi Iwai wrote:
> >>>>> On the second thought, yet another variant would be to introduce only
> >>>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
> >>>>> PAUSED state as a step before stopping the stream.  This can be
> >>>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
> >>>>> If the driver doesn't support this mode, the PCM core may fall back to
> >>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
> >>>>
> >>>> This looks identical to my proposed
> >>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
> >>>> explanatory. It's not clear what PAUSE_RESET should do (probably the
> >>>> stream should be kept inactive).
> >>>
> >>> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
> >>> doesn't have to stop the stream by itself, but it's only a preparation
> >>> for stopping.  That is, a proper TRIGGER_STOP always follows after
> >>> TRIGGER_PAUSE_RESET.
> >>
> >> But what state that we move after we reset the PAUSE? is it RUNNING
> >> (iow, equals to PAUSE_RELEASE) or something else?
> > 
> > It'll be temporarily set to RUNNING.
> > 
> >> In case of suspend while the state is paused we will:
> >> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?
> > 
> > For suspend, PAUSE_RESET, then SUSPEND.
> > For dropping, PAUSE_RESET, then STOP.
> 
> OK, but this is still the same as PAUSE_RELEASE+SUSPEND when it comes to
> suspended_state, no?
> I mean you need to move to RUNNING to be suspend, or stop to skip the
> suspend.

Yes, and this needs to be addressed.  e.g. save the suspended_state
beforehand.

> >> Most drivers are doing the same thing for STOP and SUSPEND trigger,
> >> however it is possible to do extra steps for SUSPEND to make sure that
> >> the whole system is in a lowest power state.
> > 
> > Yes, that's the reason I proposed PAUSE_RESET.  Then the driver can
> > handle the following stop or suspend operation differently.
> > 
> > That said, PAUSE_RESET is just another variant of PAUSE_RELEASE.
> > But this shouldn't actually enable the audio output again, but prepare
> > for the next SUSPEND or STOP trigger.
> 
> I'm not sure how this would be seen in ASoC level. Surely we need to
> have flag per components that it is handling PAUSE_RESET as codecs would
> like don't want to do it or they migh, hm.
> But ASoC maintains another set of state for the dpcm:
> SND_SOC_DPCM_STATE_* and it has this as well among other state-machine
> things:
> case SNDRV_PCM_TRIGGER_SUSPEND:
> 	if (be->dpcm[stream].state != SND_SOC_DPCM_STATE_START)
> 		goto next;
> 
> So, this is dpcm level, but one component can support PAUSE_RESET while
> the other does not and the transitions are not too clear to me.

It's a good question.  ASoC may have the own intermediate state
(SND_SOC_DPCM_STATE_PAUSED_RESET) or just use
SND_SOC_DPCM_STATE_START but leave all to BE trigger (by passing
SNDRV_PCM_TRIGGER_PAUSE_RESET).

> But the idea of resurrecting a paused stream with mute held is not bad.
>
> I'm still uncertain how this actually works now on systems where RESUME
> is supported and we suspend while a PCM is in paused...

Is it a general question about suspend+resume during pause, or the
behavior change after PAUSE_RESET?  PAUSE_RESET won't be applied to
the drivers that support RESUME.

BTW, I'm not pushing to sell PAUSE_RESET so much for now at all.
Currently rather in a brain storming phase.  Let's try to pick up the
more appropriate one.


thanks,

Takashi
Péter Ujfalusi April 3, 2025, 10:13 a.m. UTC | #22
On 02/04/2025 16:23, Takashi Iwai wrote:
> On Wed, 02 Apr 2025 14:52:48 +0200,
> Péter Ujfalusi wrote:
>>
>>
>>
>> On 02/04/2025 14:50, Takashi Iwai wrote:
>>> On Wed, 02 Apr 2025 13:43:57 +0200,
>>> Péter Ujfalusi wrote:
>>>>
>>>>
>>>>
>>>> On 02/04/2025 14:21, Takashi Iwai wrote:
>>>>>>> On the second thought, yet another variant would be to introduce only
>>>>>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
>>>>>>> PAUSED state as a step before stopping the stream.  This can be
>>>>>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
>>>>>>> If the driver doesn't support this mode, the PCM core may fall back to
>>>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
>>>>>>
>>>>>> This looks identical to my proposed
>>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
>>>>>> explanatory. It's not clear what PAUSE_RESET should do (probably the
>>>>>> stream should be kept inactive).
>>>>>
>>>>> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
>>>>> doesn't have to stop the stream by itself, but it's only a preparation
>>>>> for stopping.  That is, a proper TRIGGER_STOP always follows after
>>>>> TRIGGER_PAUSE_RESET.
>>>>
>>>> But what state that we move after we reset the PAUSE? is it RUNNING
>>>> (iow, equals to PAUSE_RELEASE) or something else?
>>>
>>> It'll be temporarily set to RUNNING.
>>>
>>>> In case of suspend while the state is paused we will:
>>>> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?
>>>
>>> For suspend, PAUSE_RESET, then SUSPEND.
>>> For dropping, PAUSE_RESET, then STOP.
>>
>> OK, but this is still the same as PAUSE_RELEASE+SUSPEND when it comes to
>> suspended_state, no?
>> I mean you need to move to RUNNING to be suspend, or stop to skip the
>> suspend.
> 
> Yes, and this needs to be addressed.  e.g. save the suspended_state
> beforehand.
> 
>>>> Most drivers are doing the same thing for STOP and SUSPEND trigger,
>>>> however it is possible to do extra steps for SUSPEND to make sure that
>>>> the whole system is in a lowest power state.
>>>
>>> Yes, that's the reason I proposed PAUSE_RESET.  Then the driver can
>>> handle the following stop or suspend operation differently.
>>>
>>> That said, PAUSE_RESET is just another variant of PAUSE_RELEASE.
>>> But this shouldn't actually enable the audio output again, but prepare
>>> for the next SUSPEND or STOP trigger.
>>
>> I'm not sure how this would be seen in ASoC level. Surely we need to
>> have flag per components that it is handling PAUSE_RESET as codecs would
>> like don't want to do it or they migh, hm.
>> But ASoC maintains another set of state for the dpcm:
>> SND_SOC_DPCM_STATE_* and it has this as well among other state-machine
>> things:
>> case SNDRV_PCM_TRIGGER_SUSPEND:
>> 	if (be->dpcm[stream].state != SND_SOC_DPCM_STATE_START)
>> 		goto next;
>>
>> So, this is dpcm level, but one component can support PAUSE_RESET while
>> the other does not and the transitions are not too clear to me.
> 
> It's a good question.  ASoC may have the own intermediate state
> (SND_SOC_DPCM_STATE_PAUSED_RESET) or just use
> SND_SOC_DPCM_STATE_START but leave all to BE trigger (by passing
> SNDRV_PCM_TRIGGER_PAUSE_RESET).
> 
>> But the idea of resurrecting a paused stream with mute held is not bad.
>>
>> I'm still uncertain how this actually works now on systems where RESUME
>> is supported and we suspend while a PCM is in paused...
> 
> Is it a general question about suspend+resume during pause, or the
> behavior change after PAUSE_RESET?  PAUSE_RESET won't be applied to
> the drivers that support RESUME.
> 
> BTW, I'm not pushing to sell PAUSE_RESET so much for now at all.
> Currently rather in a brain storming phase.  Let's try to pick up the
> more appropriate one.

While I think the patch is a good start to handle the current situation
with a minimal change (and a possibility of dropping it when the more
elaborate and likely laborious one is available), this is how it was
before the patch in regards to suspend while paused:

RESUME is suported:
- Stream is PAUSED
- No SUSPEND trigger is sent (stream is not running), state changes to
SUSPENDED, suspended_state is PAUSED
- On resume, no RESUME trigger is sent (suspended state is not RUNNING
or DRAIN), but the state is changed to PAUSED.
- On PAUSE_RELEASE the trigger is sent (how it is handled in driver is
another question as we had suspended the system in between).

RESUME is not suported:
- Stream is PAUSED
- No SUSPEND trigger is sent (stream is not running), state changes to
SUSPENDED, suspended_state is PAUSED
- On resume, no RESUME trigger is sent (resume is not supported), the
state remains to SUSPENDED, suspended_state is PAUSED
On PAUSE_RELEASE the core will raturn error as the state is not PAUSED,
it is SUSPENDED and we go to xrun recovery
- driver received PAUSE_PUSH but never receives PAUSE_RELEASE

I think that the SUSPEND trigger (or a new similar trigger) should be
sent to a PAUSED stream regardless of RESUME support. Imho it is not a
correct assumption that a PAUSED stream is ready for system suspend, it
might not be, it might needs to be prepared for that.
If we use the same SUSPEND trigger than the drivers need to track their
paused state as well to know what to do - we rely on the enforced state
changes in core to do this implicitly atm.

If there is no RESUME support, then it is a bit more complicated as
after the system suspend the audio hardware's state does not matter,
whenever it was RUNNMING or PAUSED we need to go to xrun recovery to
restart, resume is not supported.

But if RESUME is not supported we do need to send the PAUSE_RELEASE to
balance the PUSH.
Should we set XRUN as suspended_state in case of RESUME is not
supported? Or some other state which indicates that whatever we had
before, we are not going to be able to pick up where we were?

The fact that we discuss this shows how much this corner case have been
excercised (we eneded upm disabling this in CI testing for now in SOF).
Takashi Iwai April 3, 2025, 2:01 p.m. UTC | #23
On Thu, 03 Apr 2025 12:13:39 +0200,
Péter Ujfalusi wrote:
> 
> 
> 
> On 02/04/2025 16:23, Takashi Iwai wrote:
> > On Wed, 02 Apr 2025 14:52:48 +0200,
> > Péter Ujfalusi wrote:
> >>
> >>
> >>
> >> On 02/04/2025 14:50, Takashi Iwai wrote:
> >>> On Wed, 02 Apr 2025 13:43:57 +0200,
> >>> Péter Ujfalusi wrote:
> >>>>
> >>>>
> >>>>
> >>>> On 02/04/2025 14:21, Takashi Iwai wrote:
> >>>>>>> On the second thought, yet another variant would be to introduce only
> >>>>>>> one new trigger, SNDRV_PCM_TRIGGER_PAUSE_RESET, for resetting the
> >>>>>>> PAUSED state as a step before stopping the stream.  This can be
> >>>>>>> applied to all cases for snd_pcm_drop() and snd_pcm_suspend(), too.
> >>>>>>> If the driver doesn't support this mode, the PCM core may fall back to
> >>>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE.
> >>>>>>
> >>>>>> This looks identical to my proposed
> >>>>>> SNDRV_PCM_TRIGGER_PAUSE_RELEASE_AND_STOP which is more
> >>>>>> explanatory. It's not clear what PAUSE_RESET should do (probably the
> >>>>>> stream should be kept inactive).
> >>>>>
> >>>>> There is a slight difference.  My proposal, *_TRIGGER_PAUSE_RESET,
> >>>>> doesn't have to stop the stream by itself, but it's only a preparation
> >>>>> for stopping.  That is, a proper TRIGGER_STOP always follows after
> >>>>> TRIGGER_PAUSE_RESET.
> >>>>
> >>>> But what state that we move after we reset the PAUSE? is it RUNNING
> >>>> (iow, equals to PAUSE_RELEASE) or something else?
> >>>
> >>> It'll be temporarily set to RUNNING.
> >>>
> >>>> In case of suspend while the state is paused we will:
> >>>> TRIGGER_PAUSE_RESET followed by STOP or SUSPEND?
> >>>
> >>> For suspend, PAUSE_RESET, then SUSPEND.
> >>> For dropping, PAUSE_RESET, then STOP.
> >>
> >> OK, but this is still the same as PAUSE_RELEASE+SUSPEND when it comes to
> >> suspended_state, no?
> >> I mean you need to move to RUNNING to be suspend, or stop to skip the
> >> suspend.
> > 
> > Yes, and this needs to be addressed.  e.g. save the suspended_state
> > beforehand.
> > 
> >>>> Most drivers are doing the same thing for STOP and SUSPEND trigger,
> >>>> however it is possible to do extra steps for SUSPEND to make sure that
> >>>> the whole system is in a lowest power state.
> >>>
> >>> Yes, that's the reason I proposed PAUSE_RESET.  Then the driver can
> >>> handle the following stop or suspend operation differently.
> >>>
> >>> That said, PAUSE_RESET is just another variant of PAUSE_RELEASE.
> >>> But this shouldn't actually enable the audio output again, but prepare
> >>> for the next SUSPEND or STOP trigger.
> >>
> >> I'm not sure how this would be seen in ASoC level. Surely we need to
> >> have flag per components that it is handling PAUSE_RESET as codecs would
> >> like don't want to do it or they migh, hm.
> >> But ASoC maintains another set of state for the dpcm:
> >> SND_SOC_DPCM_STATE_* and it has this as well among other state-machine
> >> things:
> >> case SNDRV_PCM_TRIGGER_SUSPEND:
> >> 	if (be->dpcm[stream].state != SND_SOC_DPCM_STATE_START)
> >> 		goto next;
> >>
> >> So, this is dpcm level, but one component can support PAUSE_RESET while
> >> the other does not and the transitions are not too clear to me.
> > 
> > It's a good question.  ASoC may have the own intermediate state
> > (SND_SOC_DPCM_STATE_PAUSED_RESET) or just use
> > SND_SOC_DPCM_STATE_START but leave all to BE trigger (by passing
> > SNDRV_PCM_TRIGGER_PAUSE_RESET).
> > 
> >> But the idea of resurrecting a paused stream with mute held is not bad.
> >>
> >> I'm still uncertain how this actually works now on systems where RESUME
> >> is supported and we suspend while a PCM is in paused...
> > 
> > Is it a general question about suspend+resume during pause, or the
> > behavior change after PAUSE_RESET?  PAUSE_RESET won't be applied to
> > the drivers that support RESUME.
> > 
> > BTW, I'm not pushing to sell PAUSE_RESET so much for now at all.
> > Currently rather in a brain storming phase.  Let's try to pick up the
> > more appropriate one.
> 
> While I think the patch is a good start to handle the current situation
> with a minimal change (and a possibility of dropping it when the more
> elaborate and likely laborious one is available), this is how it was
> before the patch in regards to suspend while paused:
> 
> RESUME is suported:
> - Stream is PAUSED
> - No SUSPEND trigger is sent (stream is not running), state changes to
> SUSPENDED, suspended_state is PAUSED
> - On resume, no RESUME trigger is sent (suspended state is not RUNNING
> or DRAIN), but the state is changed to PAUSED.
> - On PAUSE_RELEASE the trigger is sent (how it is handled in driver is
> another question as we had suspended the system in between).
> 
> RESUME is not suported:
> - Stream is PAUSED
> - No SUSPEND trigger is sent (stream is not running), state changes to
> SUSPENDED, suspended_state is PAUSED
> - On resume, no RESUME trigger is sent (resume is not supported), the
> state remains to SUSPENDED, suspended_state is PAUSED
> On PAUSE_RELEASE the core will raturn error as the state is not PAUSED,
> it is SUSPENDED and we go to xrun recovery
> - driver received PAUSE_PUSH but never receives PAUSE_RELEASE
> 
> I think that the SUSPEND trigger (or a new similar trigger) should be
> sent to a PAUSED stream regardless of RESUME support. Imho it is not a
> correct assumption that a PAUSED stream is ready for system suspend, it
> might not be, it might needs to be prepared for that.
> If we use the same SUSPEND trigger than the drivers need to track their
> paused state as well to know what to do - we rely on the enforced state
> changes in core to do this implicitly atm.
> 
> If there is no RESUME support, then it is a bit more complicated as
> after the system suspend the audio hardware's state does not matter,
> whenever it was RUNNMING or PAUSED we need to go to xrun recovery to
> restart, resume is not supported.
> 
> But if RESUME is not supported we do need to send the PAUSE_RELEASE to
> balance the PUSH.
> Should we set XRUN as suspended_state in case of RESUME is not
> supported? Or some other state which indicates that whatever we had
> before, we are not going to be able to pick up where we were?
> 
> The fact that we discuss this shows how much this corner case have been
> excercised (we eneded upm disabling this in CI testing for now in SOF).

I agree that your patch can be a good start, at least, it addresses
the existing issue with the minimal change.  There are still rough
edges and we'll need to address, but I believe the patch (or modified
/ fixed one) can be applied to 6.15 kernel, while keeping the
development for 6.16.

I did some quick work on it, and now implemented
SNDRV_PCM_PAUSE_RELEASE_STOP trigger.  I guess it's more or less
aligned with what Jaroslav suggested.

The patches are pushed out to topic/pcm-pause-rework branch of my
sound.git tree.  Please take a look.


thanks,

Takashi
Jaroslav Kysela April 3, 2025, 3:24 p.m. UTC | #24
On 03. 04. 25 16:01, Takashi Iwai wrote:

> I agree that your patch can be a good start, at least, it addresses
> the existing issue with the minimal change.  There are still rough
> edges and we'll need to address, but I believe the patch (or modified
> / fixed one) can be applied to 6.15 kernel, while keeping the
> development for 6.16.

I did a quick research how is RESUME/SUSPEND implemented in current drivers 
and I found that it may be considered to enable SUSPEND trigger also in the 
paused state:

find sound/ -name "*.[ch]" -exec grep -A 2 -B 2 -H -E \
     "SNDRV_PCM_TRIGGER_(RESUME|SUSPEND)" {} \;

My comments - seems almost all drivers handles SUSPEND as STOP:

sound/pci/ymfpci/ymfpci_main.c
  - STOP/SUSPEND different, should not be a problem
sound/pci/nm256/nm256.c
sound/pci/intel8x0.c
  - only saves suspend flag additionaly to STOP
sound/soc
  - some drivers ignores suspend/resume (ignore_suspend flag)
sound/soc/pxa/pxa-ssp.c
  - should not be a problem (enable/disable hw parts)
sound/soc/sti/uniperif_player.c
  - only resume trigger is supported ? but no resume info is set
sound/soc/sof/intel/hda-dai.c
  - only suspend support, but resume is not supported
sound/xen/xen_snd_front_alsa.c
  - it's just rpc
sound/virtio/virtio_pcm_ops.c
  - only suspend support, no resume

It would be probably nice if more eyes verifies this, but the only suspicious 
place is the xen front end. And some drivers may be broken already (no changes 
for them).

I believe that this patch should be enough to resolve the issue instead the 
non-elegant pause release and may be applied also to 6.15:

diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
index 4057f9f10aee..63a1b37cc098 100644
--- a/sound/core/pcm_native.c
+++ b/sound/core/pcm_native.c
@@ -1694,8 +1694,17 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream 
*substream,
         struct snd_pcm_runtime *runtime = substream->runtime;
         if (runtime->trigger_master != substream)
                 return 0;
-       if (! snd_pcm_running(substream))
-               return 0;
+       if (! snd_pcm_running(substream)) {
+               if (runtime->info & SNDRV_PCM_INFO_RESUME)
+                       return 0;
+               if (runtime->state != SNDRV_PCM_STATE_PAUSED)
+                       return 0;
+               /*
+                * When resume is not supported, SUSPEND should STOP stream.
+                * For futher operation, the stream must be fully restarted
+                * to leave the permanent SUSPEND state.
+                */
+       }
         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
         runtime->stop_operating = true;
         return 0; /* suspend unconditionally */

				Jaroslav

> I did some quick work on it, and now implemented
> SNDRV_PCM_PAUSE_RELEASE_STOP trigger.  I guess it's more or less
> aligned with what Jaroslav suggested.

Nice cleanups. The "ALSA: pcm: Save the proper suspended_state for 
non-resumable case, too" should be recoded. The state should be handled in 
action callbacks for all streams separetely.

Also, I would consider to call suspend/resume triggers (depending on a flag 
like can_pause_release_stop) when the stream is paused, too. Some drivers may 
want this scheme.

					Jaroslav
Takashi Iwai April 3, 2025, 4:09 p.m. UTC | #25
On Thu, 03 Apr 2025 17:24:44 +0200,
Jaroslav Kysela wrote:
> 
> On 03. 04. 25 16:01, Takashi Iwai wrote:
> 
> > I agree that your patch can be a good start, at least, it addresses
> > the existing issue with the minimal change.  There are still rough
> > edges and we'll need to address, but I believe the patch (or modified
> > / fixed one) can be applied to 6.15 kernel, while keeping the
> > development for 6.16.
> 
> I did a quick research how is RESUME/SUSPEND implemented in current
> drivers and I found that it may be considered to enable SUSPEND
> trigger also in the paused state:
> 
> find sound/ -name "*.[ch]" -exec grep -A 2 -B 2 -H -E \
>     "SNDRV_PCM_TRIGGER_(RESUME|SUSPEND)" {} \;
> 
> My comments - seems almost all drivers handles SUSPEND as STOP:
> 
> sound/pci/ymfpci/ymfpci_main.c
>  - STOP/SUSPEND different, should not be a problem
> sound/pci/nm256/nm256.c
> sound/pci/intel8x0.c
>  - only saves suspend flag additionaly to STOP
> sound/soc
>  - some drivers ignores suspend/resume (ignore_suspend flag)
> sound/soc/pxa/pxa-ssp.c
>  - should not be a problem (enable/disable hw parts)
> sound/soc/sti/uniperif_player.c
>  - only resume trigger is supported ? but no resume info is set
> sound/soc/sof/intel/hda-dai.c
>  - only suspend support, but resume is not supported
> sound/xen/xen_snd_front_alsa.c
>  - it's just rpc
> sound/virtio/virtio_pcm_ops.c
>  - only suspend support, no resume
> 
> It would be probably nice if more eyes verifies this, but the only
> suspicious place is the xen front end. And some drivers may be broken
> already (no changes for them).

The xen driver provides SNDRV_PCM_INFO_PAUSE, but it doesn't handle
SNDRV_PCM_TRIGGER_PAUSE_* at all -- so the pause handling is broken
there, as it seems.

> I believe that this patch should be enough to resolve the issue
> instead the non-elegant pause release and may be applied also to 6.15:
> 
> diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
> index 4057f9f10aee..63a1b37cc098 100644
> --- a/sound/core/pcm_native.c
> +++ b/sound/core/pcm_native.c
> @@ -1694,8 +1694,17 @@ static int snd_pcm_do_suspend(struct
> snd_pcm_substream *substream,
>         struct snd_pcm_runtime *runtime = substream->runtime;
>         if (runtime->trigger_master != substream)
>                 return 0;
> -       if (! snd_pcm_running(substream))
> -               return 0;
> +       if (! snd_pcm_running(substream)) {
> +               if (runtime->info & SNDRV_PCM_INFO_RESUME)
> +                       return 0;
> +               if (runtime->state != SNDRV_PCM_STATE_PAUSED)
> +                       return 0;
> +               /*
> +                * When resume is not supported, SUSPEND should STOP stream.
> +                * For futher operation, the stream must be fully restarted
> +                * to leave the permanent SUSPEND state.
> +                */
> +       }
>         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>         runtime->stop_operating = true;
>         return 0; /* suspend unconditionally */

I'm afraid that it's not that straightforward, unfortunately.
There are some stuff that require the pause release.

For example, pcm_dmaengine.c uses dmaengine_pause() as the suspend,
and if we apply this change, it'll end up with dmaengine_pause() calls
twice.  As far as I see, a similar pattern is seen in ASoC Intel catpt
driver (and maybe some others), too.

And, in drivers/soundwire/intel.c, there is already a workaround for
suspend-at-pause, and this might conflict.  We need to verify it.

Last but not least, the ASoC PCM core has its own DPCM state, and the
trigger-SUSPEND skips the operation unless the stream has been
running.  I believe that's the very starting point of the problem
Peter's patch tries to address.


> > I did some quick work on it, and now implemented
> > SNDRV_PCM_PAUSE_RELEASE_STOP trigger.  I guess it's more or less
> > aligned with what Jaroslav suggested.
> 
> Nice cleanups. The "ALSA: pcm: Save the proper suspended_state for
> non-resumable case, too" should be recoded. The state should be
> handled in action callbacks for all streams separetely.

OK, I'll rewrite it.  I wrote in the current way, as it'll be a bit
too complex otherwise, but let me try another shot.

> Also, I would consider to call suspend/resume triggers (depending on a
> flag like can_pause_release_stop) when the stream is paused, too. Some
> drivers may want this scheme.

Do you mean drivers that do share the same operation for
suspend/resume and pause?


thanks,

Takashi
Jaroslav Kysela April 3, 2025, 7:05 p.m. UTC | #26
On 03. 04. 25 18:09, Takashi Iwai wrote:

> Last but not least, the ASoC PCM core has its own DPCM state, and the
> trigger-SUSPEND skips the operation unless the stream has been
> running.  I believe that's the very starting point of the problem
> Peter's patch tries to address.

Unfortunately, after all discussions, I'm not convinced about Peter's patch. It smells like an workaround rather than a fix for forever. The problem is there for years so we should not concentrate on a quick fix.

>> Also, I would consider to call suspend/resume triggers (depending on a
>> flag like can_pause_release_stop) when the stream is paused, too. Some
>> drivers may want this scheme.
> 
> Do you mean drivers that do share the same operation for
> suspend/resume and pause?

Many drivers supporting pause may use it when we allow PAUSE -> SUSPEND and SUSPEND -> PAUSE transitions.

Something like:

diff --git a/include/sound/pcm.h b/include/sound/pcm.h
index ac8f3aef9205..6d6882973883 100644
--- a/include/sound/pcm.h
+++ b/include/sound/pcm.h
@@ -541,6 +541,8 @@ struct snd_pcm {
         bool internal; /* pcm is for internal use only */
         bool nonatomic; /* whole PCM operations are in non-atomic context */
         bool no_device_suspend; /* don't invoke device PM suspend */
+       bool do_suspend_and_resume_in_pause; /* allow direct PAUSED -> SUSPENDED and
+                                               SUSPENDED -> PAUSED transitions */
  #if IS_ENABLED(CONFIG_SND_PCM_OSS)
         struct snd_pcm_oss oss;
  #endif
diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
index 4057f9f10aee..3ec4ef7809ea 100644
--- a/sound/core/pcm_native.c
+++ b/sound/core/pcm_native.c
@@ -1694,8 +1694,11 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
         struct snd_pcm_runtime *runtime = substream->runtime;
         if (runtime->trigger_master != substream)
                 return 0;
-       if (! snd_pcm_running(substream))
-               return 0;
+       if (! snd_pcm_running(substream)) {
+               if (!runtime->do_suspend_and_resume_in_pause ||
+                   runtime->state != SNDRV_PCM_STATE_PAUSED)
+                       return 0;
+       }
         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
         runtime->stop_operating = true;
         return 0; /* suspend unconditionally */
@@ -1798,8 +1801,11 @@ static int snd_pcm_do_resume(struct snd_pcm_substream *substream,
         /* DMA not running previously? */
         if (runtime->suspended_state != SNDRV_PCM_STATE_RUNNING &&
             (runtime->suspended_state != SNDRV_PCM_STATE_DRAINING ||
-            substream->stream != SNDRV_PCM_STREAM_PLAYBACK))
-               return 0;
+            substream->stream != SNDRV_PCM_STREAM_PLAYBACK)) {
+               if (!runtime->do_suspend_and_resume_in_pause ||
+                   runtime->suspend_state != SNDRV_PCM_STATE_PAUSED)
+                       return 0;
+       }
         return substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_RESUME);
  }

This will allow to address this problem correctly without any pause release just to put all parts to the running state and immediately stop the stream. Ideally, all drivers should be migrated to this state scheme and the condition may be removed at the end.

					Jaroslav
Péter Ujfalusi April 4, 2025, 8:58 a.m. UTC | #27
On 03/04/2025 18:24, Jaroslav Kysela wrote:
> On 03. 04. 25 16:01, Takashi Iwai wrote:
> 
>> I agree that your patch can be a good start, at least, it addresses
>> the existing issue with the minimal change.  There are still rough
>> edges and we'll need to address, but I believe the patch (or modified
>> / fixed one) can be applied to 6.15 kernel, while keeping the
>> development for 6.16.
> 
> I did a quick research how is RESUME/SUSPEND implemented in current
> drivers and I found that it may be considered to enable SUSPEND trigger
> also in the paused state:
> 
> find sound/ -name "*.[ch]" -exec grep -A 2 -B 2 -H -E \
>     "SNDRV_PCM_TRIGGER_(RESUME|SUSPEND)" {} \;
> 
> My comments - seems almost all drivers handles SUSPEND as STOP:
> 
> sound/pci/ymfpci/ymfpci_main.c
>  - STOP/SUSPEND different, should not be a problem
> sound/pci/nm256/nm256.c
> sound/pci/intel8x0.c
>  - only saves suspend flag additionaly to STOP
> sound/soc
>  - some drivers ignores suspend/resume (ignore_suspend flag)
> sound/soc/pxa/pxa-ssp.c
>  - should not be a problem (enable/disable hw parts)
> sound/soc/sti/uniperif_player.c
>  - only resume trigger is supported ? but no resume info is set
> sound/soc/sof/intel/hda-dai.c
>  - only suspend support, but resume is not supported
> sound/xen/xen_snd_front_alsa.c
>  - it's just rpc
> sound/virtio/virtio_pcm_ops.c
>  - only suspend support, no resume
> 
> It would be probably nice if more eyes verifies this, but the only
> suspicious place is the xen front end. And some drivers may be broken
> already (no changes for them).
> 
> I believe that this patch should be enough to resolve the issue instead
> the non-elegant pause release and may be applied also to 6.15:
> 
> diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
> index 4057f9f10aee..63a1b37cc098 100644
> --- a/sound/core/pcm_native.c
> +++ b/sound/core/pcm_native.c
> @@ -1694,8 +1694,17 @@ static int snd_pcm_do_suspend(struct
> snd_pcm_substream *substream,
>         struct snd_pcm_runtime *runtime = substream->runtime;
>         if (runtime->trigger_master != substream)
>                 return 0;
> -       if (! snd_pcm_running(substream))
> -               return 0;
> +       if (! snd_pcm_running(substream)) {
> +               if (runtime->info & SNDRV_PCM_INFO_RESUME)
> +                       return 0;
> +               if (runtime->state != SNDRV_PCM_STATE_PAUSED)
> +                       return 0;
> +               /*
> +                * When resume is not supported, SUSPEND should STOP
> stream.
> +                * For futher operation, the stream must be fully restarted
> +                * to leave the permanent SUSPEND state.
> +                */
> +       }
>         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>         runtime->stop_operating = true;
>         return 0; /* suspend unconditionally */

I have tried similar thing and the original problem remains that we are
missing the PAUSE_RELEASE trigger for the PUSH and ASoC core stops the
SUSPEND trigger to reach the drivers.

> 
>                 Jaroslav
> 
>> I did some quick work on it, and now implemented
>> SNDRV_PCM_PAUSE_RELEASE_STOP trigger.  I guess it's more or less
>> aligned with what Jaroslav suggested.
> 
> Nice cleanups. The "ALSA: pcm: Save the proper suspended_state for non-
> resumable case, too" should be recoded. The state should be handled in
> action callbacks for all streams separetely.
> 
> Also, I would consider to call suspend/resume triggers (depending on a
> flag like can_pause_release_stop) when the stream is paused, too. Some
> drivers may want this scheme.
> 
>                     Jaroslav
>
Jaroslav Kysela April 4, 2025, 9:08 a.m. UTC | #28
On 04. 04. 25 10:58, Péter Ujfalusi wrote:
> 
> 
> On 03/04/2025 18:24, Jaroslav Kysela wrote:
>> On 03. 04. 25 16:01, Takashi Iwai wrote:
>>
>>> I agree that your patch can be a good start, at least, it addresses
>>> the existing issue with the minimal change.  There are still rough
>>> edges and we'll need to address, but I believe the patch (or modified
>>> / fixed one) can be applied to 6.15 kernel, while keeping the
>>> development for 6.16.
>>
>> I did a quick research how is RESUME/SUSPEND implemented in current
>> drivers and I found that it may be considered to enable SUSPEND trigger
>> also in the paused state:
>>
>> find sound/ -name "*.[ch]" -exec grep -A 2 -B 2 -H -E \
>>      "SNDRV_PCM_TRIGGER_(RESUME|SUSPEND)" {} \;
>>
>> My comments - seems almost all drivers handles SUSPEND as STOP:
>>
>> sound/pci/ymfpci/ymfpci_main.c
>>   - STOP/SUSPEND different, should not be a problem
>> sound/pci/nm256/nm256.c
>> sound/pci/intel8x0.c
>>   - only saves suspend flag additionaly to STOP
>> sound/soc
>>   - some drivers ignores suspend/resume (ignore_suspend flag)
>> sound/soc/pxa/pxa-ssp.c
>>   - should not be a problem (enable/disable hw parts)
>> sound/soc/sti/uniperif_player.c
>>   - only resume trigger is supported ? but no resume info is set
>> sound/soc/sof/intel/hda-dai.c
>>   - only suspend support, but resume is not supported
>> sound/xen/xen_snd_front_alsa.c
>>   - it's just rpc
>> sound/virtio/virtio_pcm_ops.c
>>   - only suspend support, no resume
>>
>> It would be probably nice if more eyes verifies this, but the only
>> suspicious place is the xen front end. And some drivers may be broken
>> already (no changes for them).
>>
>> I believe that this patch should be enough to resolve the issue instead
>> the non-elegant pause release and may be applied also to 6.15:
>>
>> diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
>> index 4057f9f10aee..63a1b37cc098 100644
>> --- a/sound/core/pcm_native.c
>> +++ b/sound/core/pcm_native.c
>> @@ -1694,8 +1694,17 @@ static int snd_pcm_do_suspend(struct
>> snd_pcm_substream *substream,
>>          struct snd_pcm_runtime *runtime = substream->runtime;
>>          if (runtime->trigger_master != substream)
>>                  return 0;
>> -       if (! snd_pcm_running(substream))
>> -               return 0;
>> +       if (! snd_pcm_running(substream)) {
>> +               if (runtime->info & SNDRV_PCM_INFO_RESUME)
>> +                       return 0;
>> +               if (runtime->state != SNDRV_PCM_STATE_PAUSED)
>> +                       return 0;
>> +               /*
>> +                * When resume is not supported, SUSPEND should STOP
>> stream.
>> +                * For futher operation, the stream must be fully restarted
>> +                * to leave the permanent SUSPEND state.
>> +                */
>> +       }
>>          substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>>          runtime->stop_operating = true;
>>          return 0; /* suspend unconditionally */
> 
> I have tried similar thing and the original problem remains that we are
> missing the PAUSE_RELEASE trigger for the PUSH and ASoC core stops the
> SUSPEND trigger to reach the drivers.

All drivers must be adopted for this scheme including ASoC core, of course.

					Jaroslav
Takashi Iwai April 4, 2025, 10:44 a.m. UTC | #29
On Thu, 03 Apr 2025 21:05:51 +0200,
Jaroslav Kysela wrote:
> 
> On 03. 04. 25 18:09, Takashi Iwai wrote:
> 
> > Last but not least, the ASoC PCM core has its own DPCM state, and the
> > trigger-SUSPEND skips the operation unless the stream has been
> > running.  I believe that's the very starting point of the problem
> > Peter's patch tries to address.
> 
> Unfortunately, after all discussions, I'm not convinced about Peter's patch. It smells like an workaround rather than a fix for forever. The problem is there for years so we should not concentrate on a quick fix.
> 
> >> Also, I would consider to call suspend/resume triggers (depending on a
> >> flag like can_pause_release_stop) when the stream is paused, too. Some
> >> drivers may want this scheme.
> > 
> > Do you mean drivers that do share the same operation for
> > suspend/resume and pause?
> 
> Many drivers supporting pause may use it when we allow PAUSE -> SUSPEND and SUSPEND -> PAUSE transitions.
> 
> Something like:
> 
> diff --git a/include/sound/pcm.h b/include/sound/pcm.h
> index ac8f3aef9205..6d6882973883 100644
> --- a/include/sound/pcm.h
> +++ b/include/sound/pcm.h
> @@ -541,6 +541,8 @@ struct snd_pcm {
>         bool internal; /* pcm is for internal use only */
>         bool nonatomic; /* whole PCM operations are in non-atomic context */
>         bool no_device_suspend; /* don't invoke device PM suspend */
> +       bool do_suspend_and_resume_in_pause; /* allow direct PAUSED -> SUSPENDED and
> +                                               SUSPENDED -> PAUSED transitions */
>  #if IS_ENABLED(CONFIG_SND_PCM_OSS)
>         struct snd_pcm_oss oss;
>  #endif
> diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
> index 4057f9f10aee..3ec4ef7809ea 100644
> --- a/sound/core/pcm_native.c
> +++ b/sound/core/pcm_native.c
> @@ -1694,8 +1694,11 @@ static int snd_pcm_do_suspend(struct snd_pcm_substream *substream,
>         struct snd_pcm_runtime *runtime = substream->runtime;
>         if (runtime->trigger_master != substream)
>                 return 0;
> -       if (! snd_pcm_running(substream))
> -               return 0;
> +       if (! snd_pcm_running(substream)) {
> +               if (!runtime->do_suspend_and_resume_in_pause ||
> +                   runtime->state != SNDRV_PCM_STATE_PAUSED)
> +                       return 0;
> +       }
>         substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_SUSPEND);
>         runtime->stop_operating = true;
>         return 0; /* suspend unconditionally */
> @@ -1798,8 +1801,11 @@ static int snd_pcm_do_resume(struct snd_pcm_substream *substream,
>         /* DMA not running previously? */
>         if (runtime->suspended_state != SNDRV_PCM_STATE_RUNNING &&
>             (runtime->suspended_state != SNDRV_PCM_STATE_DRAINING ||
> -            substream->stream != SNDRV_PCM_STREAM_PLAYBACK))
> -               return 0;
> +            substream->stream != SNDRV_PCM_STREAM_PLAYBACK)) {
> +               if (!runtime->do_suspend_and_resume_in_pause ||
> +                   runtime->suspend_state != SNDRV_PCM_STATE_PAUSED)
> +                       return 0;
> +       }
>         return substream->ops->trigger(substream, SNDRV_PCM_TRIGGER_RESUME);
>  }
> 
> This will allow to address this problem correctly without any pause release just to put all parts to the running state and immediately stop the stream. Ideally, all drivers should be migrated to this state scheme and the condition may be removed at the end.

OK, this should work, too.
If we deal with another corner case such as PAUSED -> STOP, a new flag
can be added, too. (Or make one flag indicating both meanings.)

Or, there can be a driver that wants to receive trigger SUSPEND even
for the non-running substreams, too.  We need to check the details
before moving forward...

And, if there can be multiple flags, I'd rather introduce a new field
to snd_pcm_hardware, e.g. extra_flags field to keep the internal bit
flags that aren't exposed to user-space.  Otherwise the setup can be
cumbersome for each driver.

Unfortunately, the existing flags in snd_pcm like nonatomic and
no_device_suspend can't be put there, though, as those are referred
for the closed substreams, too.


thanks,

Takashi
diff mbox series

Patch

diff --git a/sound/core/pcm_native.c b/sound/core/pcm_native.c
index 6c2b6a62d9d2..a16d15ee98fa 100644
--- a/sound/core/pcm_native.c
+++ b/sound/core/pcm_native.c
@@ -1730,7 +1730,20 @@  static const struct action_ops snd_pcm_action_suspend = {
  */
 static int snd_pcm_suspend(struct snd_pcm_substream *substream)
 {
+	struct snd_pcm_runtime *runtime = substream->runtime;
+
 	guard(pcm_stream_lock_irqsave)(substream);
+	/*
+	 * Release the paused stream before suspending if resume is not
+	 * supported, because:
+	 * PAUSED streams will not receive the suspend trigger and since the
+	 * driver does not support resuming streams, it is not going to be able
+	 * to handle them properly when the system resumes.
+	 */
+	if (runtime->state == SNDRV_PCM_STATE_PAUSED &&
+	    !(runtime->info & SNDRV_PCM_INFO_RESUME))
+		snd_pcm_pause(substream, false);
+
 	return snd_pcm_action(&snd_pcm_action_suspend, substream,
 			      ACTION_ARG_IGNORE);
 }