Skip to content

Commit 544ae27

Browse files
authored
Adds an error log when a potential video codec stall is detected (#26)
We have a production issue where we get infinite fast state flips between buffering and playing after a while. So far, investigations have allowed us to understand that the issue seems that video frames do not get out of the codec. So after approximately 3 mins, the playhead gets to the end of the video buffer, so the player switches immediately to buffering. But at the same, the buffer is full (it's not being consumed anymore) so the buffering system flips it back to playing immediately. And that cycle is then repeated. This PR adds a way to detect the codec stall situation, and send an error through the logger. The app will then be able to handle the error properly (potentially by restarting the playback automatically). To do so, we just check if we've been waiting for a decoded buffer for a certain time without getting any.
1 parent 180537d commit 544ae27

File tree

4 files changed

+37
-6
lines changed

4 files changed

+37
-6
lines changed

libraries/common/src/main/java/androidx/media3/common/PlaybackException.java

+5
Original file line numberDiff line numberDiff line change
@@ -244,6 +244,9 @@ public class PlaybackException extends Exception implements Bundleable {
244244
/** MIREGO: Caused by a network error trying to get time from the NTP server */
245245
public static final int ERROR_CODE_NTP = 5905;
246246

247+
/** MIREGO: Caused by the video codec being stalled (issue under investigation) */
248+
public static final int ERROR_CODE_VIDEO_CODEC_STALLED = 5906;
249+
247250
// DRM errors (6xxx).
248251

249252
/** Caused by an unspecified error related to DRM protection. */
@@ -385,6 +388,8 @@ public static String getErrorCodeName(@ErrorCode int errorCode) {
385388
return "ERROR_CODE_AUDIO_WAITING_FOR_HEAD_POSITION_RESET";
386389
case ERROR_CODE_NTP:
387390
return "ERROR_CODE_NTP";
391+
case ERROR_CODE_VIDEO_CODEC_STALLED:
392+
return "ERROR_CODE_VIDEO_CODEC_STALLED";
388393

389394
default:
390395
if (errorCode >= CUSTOM_ERROR_CODE_BASE) {

libraries/common/src/main/java/androidx/media3/common/util/Util.java

+2-1
Original file line numberDiff line numberDiff line change
@@ -184,13 +184,14 @@ public final class Util {
184184
public static boolean useDifferentAudioSessionIdForTunneling = false; // MIREGO ADDED
185185
public static int ntpTimeoutMs = 5000; // MIREGO ADDED
186186

187-
// TEMP DEBUG STUFF
187+
// TEMP DEBUG STUFF (to investigate an infinite buffering issue caused by what seems to be a video codec stall)
188188
public static int videoLastFeedInputBufferStep = 0;
189189
public static int audioLastFeedInputBufferStep = 0;
190190
public static int videoLastDrainOutputBufferStep = 0;
191191
public static int audioLastDrainOutputBufferStep = 0;
192192
public static int currentQueuedInputBuffers = 0;
193193
public static int currentProcessedOutputBuffers = 0;
194+
public static long waitingForDecodedVideoBufferTimeMs = 0;
194195

195196
/** An empty long array. */
196197
@UnstableApi public static final long[] EMPTY_LONG_ARRAY = new long[0];

libraries/exoplayer/src/main/java/androidx/media3/exoplayer/mediacodec/MediaCodecRenderer.java

+27-3
Original file line numberDiff line numberDiff line change
@@ -414,11 +414,13 @@ private static String buildCustomDiagnosticInfo(int errorCode) {
414414
private boolean needToNotifyOutputFormatChangeAfterStreamChange;
415415

416416
// MIREGO for logging
417-
int dequeuedInputCount = 0;
418-
long lastLogMs = 0;
417+
private int dequeuedInputCount = 0;
418+
private long lastLogMs = 0;
419419

420-
// MIREGO: added the following 2 functions to log and debug a specific issue (rendering pipleine stall)
420+
// MIREGO: added the following field and 2 functions to log and debug a specific issue (rendering pipleine stall)
421421
// MIREGO: once the issue is solved, we should get rid of that code
422+
private boolean hasReportedRenderingStall = false;
423+
422424
void saveFeedInputBufferStep(int stepIndex) {
423425
if (getTrackType() == TRACK_TYPE_AUDIO) {
424426
Util.audioLastFeedInputBufferStep = stepIndex;
@@ -790,6 +792,7 @@ protected void onDisabled() {
790792
protected void onReset() {
791793
// MIREGO
792794
Log.v(Log.LOG_LEVEL_VERBOSE2, TAG, "onReset() this: %s", this);
795+
hasReportedRenderingStall = false;
793796

794797
try {
795798
disableBypass();
@@ -2139,6 +2142,22 @@ private boolean drainOutputBuffer(long positionUs, long elapsedRealtimeUs)
21392142
}
21402143

21412144
if (outputIndex < 0) {
2145+
2146+
// MIREGO video renderer stall detection
2147+
if (getTrackType() == TRACK_TYPE_VIDEO) {
2148+
2149+
if (Util.currentProcessedOutputBuffers < Util.currentQueuedInputBuffers) {
2150+
// waiting for a decoded buffer to be available from the codec
2151+
long currentTimeMs = System.currentTimeMillis();
2152+
if (Util.waitingForDecodedVideoBufferTimeMs == 0) {
2153+
Util.waitingForDecodedVideoBufferTimeMs = currentTimeMs; // starting to wait for the decoded buffer
2154+
} else if (!hasReportedRenderingStall && currentTimeMs > Util.waitingForDecodedVideoBufferTimeMs + 7000) { // been waiting for an arbitrary while, send an error to the app
2155+
Log.e(TAG, new PlaybackException("Video codec may be stalled error", new RuntimeException(), PlaybackException.ERROR_CODE_VIDEO_CODEC_STALLED));
2156+
hasReportedRenderingStall = true;
2157+
}
2158+
}
2159+
}
2160+
21422161
// MIREGO
21432162
Log.v(Log.LOG_LEVEL_VERBOSE2, TAG, "drainOutputBuffer(type:%d) Failed dequeueOutputBufferIndex res: %d (dequeuedOutputCount: %d)",
21442163
getTrackType(), outputIndex, dequeuedOutputCount);
@@ -2157,6 +2176,11 @@ private boolean drainOutputBuffer(long positionUs, long elapsedRealtimeUs)
21572176
return false;
21582177
}
21592178

2179+
// MIREGO video renderer stall detection
2180+
if (getTrackType() == TRACK_TYPE_VIDEO) {
2181+
Util.waitingForDecodedVideoBufferTimeMs = 0; // we got a decoded buffer, reset the wait time
2182+
}
2183+
21602184
// MIREGO START
21612185
dequeuedOutputCount++;
21622186
Log.v(Log.LOG_LEVEL_VERBOSE3, TAG, "drainOutputBuffer(type:%d) presentationTime: %d dequeuedOutputCount: %d outputIndex: %d",

libraries/exoplayer/src/main/java/androidx/media3/exoplayer/video/MediaCodecVideoRenderer.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -734,8 +734,6 @@ protected void onStarted() {
734734
hasNotifiedAvDesyncError = false;
735735
hasNotifiedAvDesyncSkippedFramesError = false;
736736
queuedFrames = 0;
737-
Util.currentQueuedInputBuffers = 0;
738-
Util.currentProcessedOutputBuffers = 0;
739737

740738
videoFrameReleaseControl.onStarted();
741739
}
@@ -777,6 +775,9 @@ protected void onReset() {
777775
if (placeholderSurface != null) {
778776
releasePlaceholderSurface();
779777
}
778+
Util.currentQueuedInputBuffers = 0;
779+
Util.currentProcessedOutputBuffers = 0;
780+
Util.waitingForDecodedVideoBufferTimeMs = 0;
780781
}
781782
}
782783

0 commit comments

Comments
 (0)