VCarContainer/engine/DecodeEngine/HardDecode.cpp

170 lines
4.6 KiB
C++

#include "HardDecode.h"
using namespace std;
HardDecode::HardDecode()
{
;
}
HardDecode::~HardDecode()
{
;
}
int HardDecode::hardDecoderInit(unsigned int uiWidth, unsigned int uiHeight, unsigned int uiFrameRate, AVCodecParameters *pCodecParameters)
{
uiWidth_ = uiWidth; uiHeight_ = uiHeight;
uiFrameRate_ = uiFrameRate;
this->pCodecParameters_ = pCodecParameters;
av_log_set_level(AV_LOG_ERROR);
// AVCodecID codec_id = AV_CODEC_ID_H264; //解码H264
// pCodec_ = avcodec_find_decoder(codec_id); //获取解码器
pCodec_ = avcodec_find_decoder(this->pCodecParameters_->codec_id);
if (pCodec_ == nullptr) {
LogError << "cannot find decoder !";
return 0;
}
//创建上下文
pCodecCtx_ = avcodec_alloc_context3(pCodec_);
if (!pCodecCtx_){
LogError << "Could not allocate video codec context";
return 0;
}
// 将AVCodecParameters结构体中码流参数拷贝到AVCodecContext结构体中,format, width, height, codec_type等
avcodec_parameters_to_context(pCodecCtx_, this->pCodecParameters_);
if (!pCodecCtx_) {
LogError << "Cannot alloc context.";
return 0;
}
//打开解码器
int iRet = 1;
if ((iRet = avcodec_open2(pCodecCtx_, pCodec_, nullptr)) < 0) {
LogError << "cannot open decoder, Ret:" << iRet;
return 0;
}
//分配packet
pPacket_ = av_packet_alloc();
if (!pPacket_){
LogError << "Could not allocate video packet";
return 0;
}
// av_init_packet(pPacket_);
//分配frame
pSrcFrame_ = av_frame_alloc();
if (!pSrcFrame_) {
LogError << "Could not allocate video src pFrame";
return 0;
}
pDstFrame_ = av_frame_alloc();
if (!pDstFrame_) {
LogError << "Could not allocate video dst pFrame";
return 0;
}
// printf("after align down video_width: %d, video_height: %d\n", uiWidth_, uiHeight_);
int bufferSize = av_image_get_buffer_size(AV_PIX_FMT_YUV420P,
pCodecCtx_->width,
pCodecCtx_->height, 1);
outBuffer_ = (unsigned char *) av_malloc(bufferSize);
av_image_fill_arrays(pDstFrame_->data,
pDstFrame_->linesize,
outBuffer_,
AV_PIX_FMT_YUV420P,
pCodecCtx_->width,
pCodecCtx_->height, 1);
pSwsContext_ = sws_getContext(pCodecCtx_->width, pCodecCtx_->height, pCodecCtx_->pix_fmt,
pCodecCtx_->width, pCodecCtx_->height, AV_PIX_FMT_YUV420P, SWS_BICUBIC, nullptr, nullptr, nullptr);
return 1;
}
int HardDecode::hardDecoderDeInit()
{
if (pCodecParameters_){
av_free(pCodecParameters_);
pCodecParameters_ = nullptr;
}
if(outBuffer_){
av_free(outBuffer_);
outBuffer_ = nullptr;
}
if(pSrcFrame_){
av_frame_free(&pSrcFrame_);
pSrcFrame_ = nullptr;
}
if(pDstFrame_){
av_frame_free(&pDstFrame_);
pDstFrame_ = nullptr;
}
if(pPacket_){
av_packet_free(&pPacket_);
pPacket_ = nullptr;
}
if(pCodecParserCtx_){
av_parser_close(pCodecParserCtx_);
pCodecParserCtx_ = nullptr;
}
if(pCodecCtx_){
avcodec_close(pCodecCtx_);
av_free(pCodecCtx_);
pCodecCtx_ = nullptr;
}
if(pSwsContext_){
sws_freeContext(pSwsContext_);
pSwsContext_ = nullptr;
}
}
int HardDecode::hardDecoder(void* pOutputData,unsigned int* puiOutputDataSize)
{
int ret;
ret = avcodec_send_packet(this->pCodecCtx_, this->pPacket_); //接收packet解码
if (ret < 0) {
LogError << "Error sending a packet for decoding";
return 0;
}
while (ret >= 0) {
ret = avcodec_receive_frame(this->pCodecCtx_, this->pSrcFrame_); //解码
if (ret == AVERROR(EAGAIN) || ret == AVERROR_EOF){
LogWarn << "During decoding eof";
return 0;
}
else if (ret < 0) {
LogError << "Error during decoding";
return 0;
}
sws_scale(this->pSwsContext_,
(const uint8_t *const *)this->pSrcFrame_->data,
this->pSrcFrame_->linesize,
0,
this->pCodecCtx_->height,
this->pDstFrame_->data,
this->pDstFrame_->linesize);
fflush(stdout);
int iSize = this->pCodecCtx_->width * this->pCodecCtx_->height;
memcpy(pOutputData, this->pDstFrame_->data[0], iSize); //Y
memcpy(pOutputData+iSize, this->pDstFrame_->data[1], iSize/4); //U
memcpy(pOutputData+iSize+iSize/4, this->pDstFrame_->data[2], iSize/4); //V
*puiOutputDataSize = iSize*3/2;
return iSize*3/2;
}
return 0;
}