<html>
<head>
<base href="https://bugs.freedesktop.org/">
</head>
<body>
<p>
<div>
<b><a class="bz_bug_link
bz_status_NEW "
title="NEW - Discontinuity in the interpolated delay after corking, flushing and uncorking."
href="https://bugs.freedesktop.org/show_bug.cgi?id=97799#c14">Comment # 14</a>
on <a class="bz_bug_link
bz_status_NEW "
title="NEW - Discontinuity in the interpolated delay after corking, flushing and uncorking."
href="https://bugs.freedesktop.org/show_bug.cgi?id=97799">bug 97799</a>
from <span class="vcard"><a class="email" href="mailto:bugs.freedesktop@haasn.xyz" title="Niklas Haas <bugs.freedesktop@haasn.xyz>"> <span class="fn">Niklas Haas</span></a>
</span></b>
<pre><span class="quote">> Did you interpret my message so that I was asking you questions about the
> ordering of something? I did no such thing.</span >
No, I was just pointing out that care needs to be taken when making assumptions
about what order various calls happen in. Although I now realize the point you
were making is independent of the order of events because it does not concern
the delay measurement, just the seek mode.
<span class="quote">> Reading the ao_pulse code, cork/flush/uncork clearly happen in that order. The
> only possible uncertainty is that is the RELATIVE_ON_READ write really the
> first write after the seek.</span >
Here is my code for reference:
diff --git a/audio/out/ao_pulse.c b/audio/out/ao_pulse.c
index 5a68553..a02ccb3 100644
--- a/audio/out/ao_pulse.c
+++ b/audio/out/ao_pulse.c
@@ -62,6 +62,9 @@ struct priv {
char *cfg_sink;
int cfg_buffer;
int cfg_latency_hacks;
+
+ // XXX testing
+ bool want_reset;
};
#define GENERIC_ERR_MSG(str) \
@@ -518,9 +521,15 @@ static void cork(struct ao *ao, bool pause)
static int play(struct ao *ao, void **data, int samples, int flags)
{
struct priv *priv = ao->priv;
+ enum pa_seek_mode seekmode = PA_SEEK_RELATIVE;
+ if (priv->want_reset) {
+ seekmode = PA_SEEK_RELATIVE_ON_READ;
+ priv->want_reset = false;
+ }
+
pa_threaded_mainloop_lock(priv->mainloop);
if (pa_stream_write(priv->stream, data[0], samples * ao->sstride, NULL, 0,
- PA_SEEK_RELATIVE) < 0) {
+ seekmode) < 0) {
GENERIC_ERR_MSG("pa_stream_write() failed");
samples = -1;
}
@@ -545,6 +554,7 @@ static void reset(struct ao *ao)
!priv->retval)
GENERIC_ERR_MSG("pa_stream_flush() failed");
cork(ao, false);
+ priv->want_reset = true;
}
// Pause the audio stream by corking it on the server
Based on the way I implemented it, the stream flushing could should affect the
seek mode of exactly the next pa_stream_write and no more. play() itself
will only ever be called by a single thread, so it can't be the case that
two threads enter this code at the exact same time. But I'll add some printf
debugging to this to be sure:
...
relative
relative
relative
relative
relative
want_reset = true
relative_on_read
relative
relative
relative
relative
...
So yes, it happens in exactly that order.
I double, triple and quadruple checked (running the test multiple times for
each
case) and confirmed that the result is definitely different: With this patch
applied, I get only small spikes in ao-delay which translate to small
discontinuities of about 10-20ms in ao-dev.
If I take the exact same source code, replace PA_SEEK_RELATIVE_ON_READ by
PA_SEEK_RELATIVE in the branch I added, recompile and rerun the test, I get the
same behavior as in my first post again, which is the same behavior as without
this diff. So it isn't just down to pure chance.</pre>
</div>
</p>
<hr>
<span>You are receiving this mail because:</span>
<ul>
<li>You are the assignee for the bug.</li>
<li>You are the QA Contact for the bug.</li>
</ul>
</body>
</html>