I am trying to decode a video onto a surface in Android using MediaCodec
and MediaExtractor
.
To start with, I use the setDataSource(context, uri, null)
method on the MediaExtractor
object (extractor
below) to set the data source uri. The uri comes from the Android file picker for videos. This call doesn't seem to throw any exceptions.
In the onInputBufferAvailable()
call from the decoder, I then read a new sample from the extractor but the extractor.readSampleData(..)
call randomly fails to read the correct data and starts returning buffers of size -1. Ideally, the size should only be -1 when the whole file has been read. Moreover, once the extractor starts returning -1, it never returns the correct sample however many times I call extractor.advance()
.
I also observe this warning message in the logs whenever this issue happens:
W/NuMediaExtractor: read on track 0 failed with error -2147483646
Code:
object : MediaCodec.Callback() {
override fun onInputBufferAvailable(codec: MediaCodec, index: Int) {
val buffer = codec.getInputBuffer(index)!!
try {
val size = extractor.readSampleData(buffer, 0)
if (size > 0) {
decoder.queueInputBuffer(
index,
0,
size,
extractor.sampleTime,
sampleFlags
)
extractor.advance()
} else if (size == 0) {
Timber.d("Size 0 sample received from extractor")
} else if (size == -1) {
// Size is -1 when no more samples are available
} catch (exception: Exception) {
Timber.e(exception)
}
}
This doesn't happen all the time but frequent enough. So far, I have only observed this on the Android 9 OS.
Edit 1: To prevent a race condition, I save the free input buffers into a blocking queue. A while loop then picks up these buffers from the queue.
private val inputBuffersQueue: BlockingQueue<InputBufferData> = LinkedBlockingQueue()
private fun feedSamplesToDecoder(extractor: MediaExtractor, trimStartUs: Long, trimEndUs: Long) {
inputHandler.post {
while (!wasEOSInputBufferFed) {
val inputBuffer = inputBuffersQueue.take()
val decoder = inputBuffer.codec
val index = inputBuffer.index
val sampleSize = extractor.readSampleData(inputBuffer.codec.getInputBuffer(inputBuffer.index)!!, 0)
if (sampleSize > 0) {
val sampleTime = extractor.sampleTime
lastSampleTimestampUs = sampleTime
val sampleFlags = extractor.sampleFlags
decoder.queueInputBuffer(
index,
0,
sampleSize,
sampleTime - trimStartUs,
sampleFlags
)
extractor.advance()
} else if (sampleSize == 0) {
Timber.d("Size 0 sample received from extractor")
} else {
decoder.queueInputBuffer(
index,
0,
0,
0,
BUFFER_FLAG_END_OF_STREAM
)
}
}
}
}
I think the root problem of yours is the uri. My suggestion is to reparse it into something more common that widely accepted by many apps. For example is your uri looks like this (starts with content://)
content://com.android.providers.downloads.documents/document/3356
, try to change it intofile:///storage/emulated/0/document/3356
.com.android.providers.downloads.documents
is a package name anddocument/3356
is a file path. You can get/storage/emulated/0
viaEnvironment.getExternalStorageDirectory()
method. Then useUri.parse("string")
to get a uri instance.