项目地址源码https://github.com/liluojun/PlayVideo
之前弄了下Android硬解H264https://blog.csdn.net/hjt505694246/article/details/105046233,但是硬解多少还是受到硬件影响所有再折腾下ffmpeg软解。
关于ffmpeg解码的文章很多我也是业余接触下并且我对C语言也是个初学者,有不足的地方还请见谅。
先上代码
先初始化解码器
int FFmpegEncode::creatFFmpeg(AVCodecID id) { avcodec_register_all(); mAVCodec = avcodec_find_decoder(id); if (mAVCodec == NULL) { printf("mAVCodec==NULL\n"); return -1; } mAVCodecContext = avcodec_alloc_context3(mAVCodec); if (mAVCodecContext == NULL) { printf("mAVCodecContext==NULL\n"); return -1; } if (avcodec_open2(mAVCodecContext, mAVCodec, NULL) < 0) { printf("avcodec_open2 error\n"); return -1; } mAVFrame = av_frame_alloc(); if (!mAVFrame) { printf("av_frame_alloc error\n"); return -1; } mAVPacket = av_packet_alloc(); if (!mAVPacket) { printf("av_packet_alloc error\n"); return -1; } av_init_packet(mAVPacket); return 0; } 设置解码参数:宽高、帧率、数据类型 int FFmpegEncode::initAVCodecContext(int width, int heigth, int fps, AVPixelFormat pixFmt) { if (mAVCodecContext == NULL) { printf("initAVCodecContext NULL\n"); return -1; } if (width <= 0 || heigth <= 0 || fps <= 10) { printf("initAVCodecContext Parameters of the abnormal\n"); return -1; } mAVCodecContext->width = width; mAVCodecContext->height = heigth; mAVCodecContext->frame_bits = fps; mAVCodecContext->pix_fmt = pixFmt; return 0; } 喂数据解码:framedata是源数据,*outputY,*outputU,*outputV是解码后的yuv三个分量数据,pixfmt数据类型。 这里我有个拷贝的动作,据说有零拷贝的方式,有木有大佬知道的可以指点一下。 int FFmpegEncode::encodeFFmpeg(uint8_t *framedata, int framelen,//input uint8_t *outputY, uint8_t *outputU, uint8_t *outputV, int *width, int *height, AVPixelFormat pixfmt) { if (mAVPacket == NULL) { printf("encodeFFmpeg NULL\n"); return -1; } mAVPacket->data = framedata; mAVPacket->size = framelen; while (mAVPacket->size > 0) { if (avcodec_send_packet(mAVCodecContext, mAVPacket)) { printf("%s %d avcodec_send_packet fail\n", __func__, __LINE__); return -2; } if (avcodec_receive_frame(mAVCodecContext, mAVFrame)) { printf("%s %d avcodec_receive_frame fail\n", __func__, __LINE__); return -3; } switch (mAVCodecContext->pix_fmt) { case AV_PIX_FMT_YUV420P: { memcpy(outputY, mAVFrame->data[0], *width * *height); memcpy(outputU, mAVFrame->data[1], *width * *height / 4); memcpy(outputV, mAVFrame->data[2], *width * *height / 4); width = &(mAVCodecContext->width); height = &(mAVCodecContext->height); pixfmt = mAVCodecContext->pix_fmt; break; } case AV_PIX_FMT_NV21: { memcpy(outputY, mAVFrame->data[0], *width * *height); memcpy(outputU, mAVFrame->data[1], *width * *height / 4); memcpy(outputV, mAVFrame->data[2], *width * *height / 4); width = &(mAVCodecContext->width); height = &(mAVCodecContext->height); pixfmt = mAVCodecContext->pix_fmt; break; } default: { printf("mAVCodecContext->pix_fmt Failure to identify\n"); return -4; } } return 0; } } 释放解码器资源 void FFmpegEncode::unEncode() { avcodec_close(mAVCodecContext); av_free(mAVCodecContext); av_frame_free(&mAVFrame); }