Mic*_*ght 5 video android h.264 android-mediacodec
我在尝试使用 Android 软件解码器 (OMX.google.h264.decoder) 提取和解码 H264 视频时遇到问题。该问题似乎涉及多个设备。
该视频使用 Nexus 5 硬件解码器 (OMX.qcom.video.decoder.avc) 可以正常播放。
下面的示例代码展示了该问题,并且是使用 android MediaCodec 和 MediaExtractor 类的相当标准的示例。
当第一个缓冲区传递给解码器时,我得到的异常是非法状态异常。
该视频采用 Baseline 配置文件编码为 720x480 20fps,因此应符合兼容性准则。
我非常感谢任何有关使软件视频解码器正确处理 H264 视频的指导。
public void doMp4Test()
{
try
{
//String filename = "webserver_h264.mp4";
String filename = "toodee-720p.mp4";
MediaExtractor extractor = new MediaExtractor();
extractor.setDataSource(Constants.RootDirectory + File.separator + filename);
MediaCodec decoder = null;
for (int i = 0; i < extractor.getTrackCount(); i++)
{
MediaFormat format = extractor.getTrackFormat(i);
String mime = format.getString(MediaFormat.KEY_MIME);
if (mime.startsWith("video/"))
{
extractor.selectTrack(i);
decoder = MediaCodec.createByCodecName("OMX.google.h264.decoder");
// decoder = MediaCodec.createDecoderByType("OMX.qcom.video.decoder.avc"); // working decoder type
decoder.configure(format, m_surface, null, 0);
break;
}
}
if (decoder == null)
{
Log.e("DecodeActivity", "Can't find video info!");
return;
}
decoder.start();
ByteBuffer[] inputBuffers = decoder.getInputBuffers();
ByteBuffer[] outputBuffers = decoder.getOutputBuffers();
BufferInfo info = new BufferInfo();
boolean isEOS = false;
long startMs = System.currentTimeMillis();
while (!Thread.interrupted())
{
if (!isEOS)
{
int inIndex = decoder.dequeueInputBuffer(10000);
if (inIndex >= 0)
{
ByteBuffer buffer = inputBuffers[inIndex];
int sampleSize = extractor.readSampleData(buffer, 0);
if (sampleSize < 0)
{
Log.d("DecodeActivity", "InputBuffer BUFFER_FLAG_END_OF_STREAM");
decoder.queueInputBuffer(inIndex, 0, 0, 0, MediaCodec.BUFFER_FLAG_END_OF_STREAM);
isEOS = true;
}
else
{
int flags = 0;// extractor.getSampleFlags();
decoder.queueInputBuffer(inIndex, 0, sampleSize, extractor.getSampleTime(), flags);
extractor.advance();
}
}
}
int outIndex = decoder.dequeueOutputBuffer(info, 10000);
switch (outIndex)
{
case MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED:
Log.d("DecodeActivity", "INFO_OUTPUT_BUFFERS_CHANGED");
outputBuffers = decoder.getOutputBuffers();
break;
case MediaCodec.INFO_OUTPUT_FORMAT_CHANGED:
Log.d("DecodeActivity", "New format " + decoder.getOutputFormat());
break;
case MediaCodec.INFO_TRY_AGAIN_LATER:
Log.d("DecodeActivity", "dequeueOutputBuffer timed out!");
break;
default:
ByteBuffer buffer = outputBuffers[outIndex];
Log.v("DecodeActivity", "We can't use this buffer but render it due to the API limit, " + buffer);
// We use a very simple clock to keep the video FPS, or the
// video
// playback will be too fast
while (info.presentationTimeUs / 1000 > System.currentTimeMillis() - startMs)
{
try
{
Thread.sleep(10);
}
catch (InterruptedException e)
{
e.printStackTrace();
break;
}
}
decoder.releaseOutputBuffer(outIndex, true);
break;
}
if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0)
{
Log.d("DecodeActivity", "OutputBuffer BUFFER_FLAG_END_OF_STREAM");
break;
}
}
decoder.stop();
decoder.release();
extractor.release();
}
catch (Exception e)
{
e.printStackTrace();
}
}
Run Code Online (Sandbox Code Playgroud)
使用 ffprobe 查看视频,看起来它实际上是高配置文件而不是基本配置文件:
流 #0:0(eng):视频:h264(高)(avc1 / 0x31637661)、yuv420p、720x480 [SAR 1:1 DAR 3:2]、515 kb/s、24.66 fps、1000k tbr、1000k tbn、2000k待定(默认)
顺便说一句,当我在笔记本电脑上尝试时,该视频似乎可以播放,但在几个浏览器上播放时仅显示黑色空白屏幕。
| 归档时间: |
|
| 查看次数: |
25938 次 |
| 最近记录: |