最近在搞 Android 硬解码。在 软解码 和 硬解码 的兼容实现上,进行了各种尝试。后来,忽然发现,无法正常解码了,竟然出现 Failed to open codec。
使用 ret = avcodec_open2(ctx, codec, nullptr)
查看到 返回值为 -1,获取到的错误信息是 “Operation not permitted”。这让我很头疼,对 Android 开发而言,权限这种东西,唉。
但是经过百度,发现,有网友称是因为 timebase.den 和 timebase.num 的设置不对。我忽然想到,最后一次尝试硬解的代码把 codec parameters 的拷贝忘掉了。因此,出现返回 -1 的错误的代码原文如下:
AVStream* st = pFormatCtx->streams[streamIdx];
codec_context_ = avcodec_alloc_context3(pCodec);
if (!codec_context_) {
AVLOGE("Failed to allocate the codec context.");
return -1;
}
if (avcodec_parameters_to_context(codec_context_, st->codecpar) < 0) {
AVLOGE("Failed to copy codec parameters to decoder context.");
return -1;
}
if (enable_hw_decode_) {
if (codec_context_->codec_id == AV_CODEC_ID_H264) {
pCodec = avcodec_find_decoder_by_name("h264_mediacodec");
if (pCodec) {
avcodec_free_context(&codec_context_);
codec_context_ = avcodec_alloc_context3(pCodec);
using_hw_decode_ = true;
}
}
}
int ret = 0;
if ((ret = avcodec_open2(codec_context_, pCodec, nullptr)) < 0) {
AVLOGE("Failed to open codec. ret = %d, %s", ret, PAVUtils::get_error_string(ret));
return -1;
}
由于是先拷贝过一次,但是后来因为如果是用硬解码,则重新选择 decoder ,导致重新申请的 avcodec context 未能拷贝 parameters from stream to context 这个操作。当我在 using_hw_decode_ = true
之前加上 avcodec_parameters_to_context(codec_context_, st->codecpar);
后,就能正常运行了。
很奇怪,为什么未拷贝相关参数,不是返回 INVALID_PAR 而是返回 NO_PERMIT 呢?