3个线程
一个做视频解码一个做音频解码一个做复用推流
视频解码线程展示
#include "videodecodethread.h" VideodecodeThread::VideodecodeThread(QObject *parent) :QThread(parent) { avdevice_register_all(); avformat_network_init(); } void VideodecodeThread::run() { fmt = av_find_input_format("dshow"); av_dict_set(&options, "video_size", "640*480", 0); av_dict_set(&options, "framerate", "30", 0); ret = avformat_open_input(&pFormatCtx, "video=ov9734_azurewave_camera", fmt, &options); if (ret < 0) { qDebug() << "Couldn't open input stream." << ret; return; } ret = avformat_find_stream_info(pFormatCtx, &options); if(ret < 0) { qDebug()<< "Couldn't find stream information."; return; } videoIndex = av_find_best_stream(pFormatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &pAvCodec, 0); if(videoIndex < 0) { qDebug()<< "Couldn't av_find_best_stream."; return; } pAvCodec = avcodec_find_decoder(pFormatCtx->streams[videoIndex]->codecpar->codec_id); if(!pAvCodec) { qDebug()<< "Couldn't avcodec_find_decoder."; return; } qDebug()<<"avcodec_open2 pAVCodec->name:" << QString::fromStdString(pAvCodec->name); if(pFormatCtx->streams[videoIndex]->avg_frame_rate.den != 0) { float fps_ = pFormatCtx->streams[videoIndex]->avg_frame_rate.num / pFormatCtx->streams[videoIndex]->avg_frame_rate.den; qDebug() <<"fps:" << fps_; } int64_t video_length_sec_ = pFormatCtx->duration/AV_TIME_BASE; qDebug() <<"video_length_sec_:" << video_length_sec_; pAvCodecCtx = avcodec_alloc_context3(pAvCodec); if(!pAvCodecCtx) { qDebug()<< "Couldn't avcodec_alloc_context3."; return; } ret = avcodec_parameters_to_context(pAvCodecCtx, pFormatCtx->streams[videoIndex]->codecpar); if(ret < 0) { qDebug()<< "Couldn't avcodec_parameters_to_context."; return; } pAvFrame = av_frame_alloc(); pAvFrameYUV = av_frame_alloc(); // pSwsCtx = sws_getContext(pAvCodecCtx->width, pAvCodecCtx->height, pAvCodecCtx->pix_fmt, // pAvCodecCtx->width, pAvCodecCtx->height, AV_PIX_FMT_YUV420P9, // SWS_BICUBIC, NULL, NULL, NULL); //m_size = av_image_get_buffer_size(AVPixelFormat(AV_PIX_FMT_YUV420P9), pAvCodecCtx->width, pAvCodecCtx->height, 1); //为已经分配的空间的结构体AVPicture挂上一段用于保存数据的空间 //av_image_fill_arrays(pAvFrameYUV->data, pAvFrameYUV->linesize, buffer, AV_PIX_FMT_BGR32, pAvCodecCtx->width, pAvCodecCtx->height, 1); av_new_packet(packet, pAvCodecCtx->width * pAvCodecCtx->height); while(run_flag && !av_read_frame(pFormatCtx, packet)) { if (packet->stream_index == videoIndex) { //解码一帧视频数据 int iGotPic = avcodec_send_packet(pAvCodecCtx, packet); if(iGotPic != 0) { qDebug()<<"iVideoIndex avcodec_send_packet error"; continue; } iGotPic = avcodec_receive_frame(pAvCodecCtx, pAvFrame); if(iGotPic == 0){ //转换像素 // sws_scale(pSwsCtx, (uint8_t const * const *)pAvFrame->data, pAvFrame->linesize, 0, // pAvFrame->height, pAvFrameYUV->data, pAvFrameYUV->linesize); //buffer = (uint8_t*)av_malloc(pAvFrame->height * pAvFrame->linesize[0]); byte = QByteArray((char*)pAvFrame->data); videoQueue.push(byte); videoCount++; } } av_packet_unref(packet); } }
ffmpeg 函数解读
后面加上音频以及复用推流过程