videodecode.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572
  1. #include "videodecode.h"
  2. #include <QDebug>
  3. #include <QImage>
  4. #include <QMutex>
  5. #include <qdatetime.h>
  6. extern "C" { // 用C规则编译指定的代码
  7. #include "libavcodec/avcodec.h"
  8. #include "libavformat/avformat.h"
  9. #include "libavutil/avutil.h"
  10. #include "libswscale/swscale.h"
  11. #include "libavutil/imgutils.h"
  12. }
  13. #define ERROR_LEN 1024 // 异常信息数组长度
  14. #define PRINT_LOG 1
  15. VideoDecode::VideoDecode()
  16. {
  17. // initFFmpeg(); // 5.1.2版本不需要调用了
  18. m_error = new char[ERROR_LEN];
  19. /*************************************** 获取当前环境支持的硬件解码器 *********************************************/
  20. AVHWDeviceType type = AV_HWDEVICE_TYPE_NONE; // ffmpeg支持的硬件解码器
  21. QStringList strTypes;
  22. while ((type = av_hwdevice_iterate_types(type)) != AV_HWDEVICE_TYPE_NONE) // 遍历支持的设备类型。
  23. {
  24. m_HWDeviceTypes.append(type);
  25. const char* ctype = av_hwdevice_get_type_name(type); // 获取AVHWDeviceType的字符串名称。
  26. if(ctype)
  27. {
  28. strTypes.append(QString(ctype));
  29. }
  30. }
  31. qDebug() << "支持的硬件解码器:" << strTypes;
  32. /************************************************ END ******************************************************/
  33. }
  34. VideoDecode::~VideoDecode()
  35. {
  36. close();
  37. }
  38. /**
  39. * @brief 初始化ffmpeg库(整个程序中只需加载一次)
  40. * 旧版本的ffmpeg需要注册各种文件格式、解复用器、对网络库进行全局初始化。
  41. * 在新版本的ffmpeg中纷纷弃用了,不需要注册了
  42. */
  43. void VideoDecode::initFFmpeg()
  44. {
  45. static bool isFirst = true;
  46. static QMutex mutex;
  47. QMutexLocker locker(&mutex);
  48. if(isFirst)
  49. {
  50. // av_register_all(); // 已经从源码中删除
  51. /**
  52. * 初始化网络库,用于打开网络流媒体,此函数仅用于解决旧GnuTLS或OpenSSL库的线程安全问题。
  53. * 一旦删除对旧GnuTLS和OpenSSL库的支持,此函数将被弃用,并且此函数不再有任何用途。
  54. */
  55. avformat_network_init();
  56. isFirst = false;
  57. }
  58. }
  59. /*********************************** FFmpeg获取GPU硬件解码帧格式的回调函数 *****************************************/
  60. static enum AVPixelFormat g_pixelFormat;
  61. /**
  62. * @brief 回调函数,获取GPU硬件解码帧的格式
  63. * @param s
  64. * @param fmt
  65. * @return
  66. */
  67. AVPixelFormat get_hw_format(AVCodecContext* s, const enum AVPixelFormat* fmt)
  68. {
  69. Q_UNUSED(s)
  70. const enum AVPixelFormat* p;
  71. for (p = fmt; *p != -1; p++)
  72. {
  73. if(*p == g_pixelFormat)
  74. {
  75. return *p;
  76. }
  77. }
  78. qDebug() << "无法获取硬件表面格式."; // 当同时打开太多路视频时,如果超过了GPU的能力,可能会返回找不到解码帧格式
  79. return AV_PIX_FMT_NONE;
  80. }
  81. /************************************************ END ******************************************************/
  82. /**************************************** FFmpeg初始化硬件解码器 **********************************************/
  83. /**
  84. * @brief 初始化硬件解码器
  85. * @param codec
  86. */
  87. void VideoDecode::initHWDecoder(const AVCodec *codec)
  88. {
  89. if(!codec) return;
  90. for(int i = 0; ; i++)
  91. {
  92. const AVCodecHWConfig* config = avcodec_get_hw_config(codec, i); // 检索编解码器支持的硬件配置。
  93. if(!config)
  94. {
  95. qDebug() << "打开硬件解码器失败!";
  96. return; // 没有找到支持的硬件配置
  97. }
  98. if (config->methods & AV_CODEC_HW_CONFIG_METHOD_HW_DEVICE_CTX) // 判断是否是设备类型
  99. {
  100. for(auto i : m_HWDeviceTypes)
  101. {
  102. if(config->device_type == AVHWDeviceType(i)) // 判断设备类型是否是支持的硬件解码器
  103. {
  104. g_pixelFormat = config->pix_fmt;
  105. // 打开指定类型的设备,并为其创建AVHWDeviceContext。
  106. int ret = av_hwdevice_ctx_create(&hw_device_ctx, config->device_type, nullptr, nullptr, 0);
  107. if(ret < 0)
  108. {
  109. showError(ret);
  110. free();
  111. return ;
  112. }
  113. qDebug() << "打开硬件解码器:" << av_hwdevice_get_type_name(config->device_type);
  114. m_codecContext->hw_device_ctx = av_buffer_ref(hw_device_ctx); // 创建一个对AVBuffer的新引用。
  115. m_codecContext->get_format = get_hw_format; // 由一些解码器调用,以选择将用于输出帧的像素格式
  116. return;
  117. }
  118. }
  119. }
  120. }
  121. }
  122. /************************************************ END ******************************************************/
  123. /**
  124. * @brief 打开媒体文件,或者流媒体,例如rtmp、strp、http
  125. * @param url 视频地址
  126. * @return true:成功 false:失败
  127. */
  128. bool VideoDecode::open(const QString &url)
  129. {
  130. if(url.isNull()) return false;
  131. AVDictionary* dict = nullptr;
  132. av_dict_set(&dict, "rtsp_transport", "tcp", 0); // 设置rtsp流使用tcp打开,如果打开失败错误信息为【Error number -135 occurred】可以切换(UDP、tcp、udp_multicast、http),比如vlc推流就需要使用udp打开
  133. av_dict_set(&dict, "max_delay", "3", 0); // 设置最大复用或解复用延迟(以微秒为单位)。当通过【UDP】 接收数据时,解复用器尝试重新排序接收到的数据包(因为它们可能无序到达,或者数据包可能完全丢失)。这可以通过将最大解复用延迟设置为零(通过max_delayAVFormatContext 字段)来禁用。
  134. av_dict_set(&dict, "timeout", "1000000", 0); // 以微秒为单位设置套接字 TCP I/O 超时,如果等待时间过短,也可能会还没连接就返回了。
  135. // 打开输入流并返回解封装上下文
  136. int ret = avformat_open_input(&m_formatContext, // 返回解封装上下文
  137. url.toStdString().data(), // 打开视频地址
  138. nullptr, // 如果非null,此参数强制使用特定的输入格式。自动选择解封装器(文件格式)
  139. &dict); // 参数设置
  140. // 释放参数字典
  141. if(dict)
  142. {
  143. av_dict_free(&dict);
  144. }
  145. // 打开视频失败
  146. if(ret < 0)
  147. {
  148. showError(ret);
  149. free();
  150. return false;
  151. }
  152. // 读取媒体文件的数据包以获取流信息。
  153. ret = avformat_find_stream_info(m_formatContext, nullptr);
  154. if(ret < 0)
  155. {
  156. showError(ret);
  157. free();
  158. return false;
  159. }
  160. m_totalTime = m_formatContext->duration / (AV_TIME_BASE / 1000); // 计算视频总时长(毫秒)
  161. #if PRINT_LOG
  162. qDebug() << QString("视频总时长:%1 ms,[%2]").arg(m_totalTime).arg(QTime::fromMSecsSinceStartOfDay(int(m_totalTime)).toString("HH:mm:ss zzz"));
  163. #endif
  164. // 通过AVMediaType枚举查询视频流ID(也可以通过遍历查找),最后一个参数无用
  165. m_videoIndex = av_find_best_stream(m_formatContext, AVMEDIA_TYPE_VIDEO, -1, -1, nullptr, 0);
  166. if(m_videoIndex < 0)
  167. {
  168. showError(m_videoIndex);
  169. free();
  170. return false;
  171. }
  172. AVStream* videoStream = m_formatContext->streams[m_videoIndex]; // 通过查询到的索引获取视频流
  173. // 获取视频图像分辨率(AVStream中的AVCodecContext在新版本中弃用,改为使用AVCodecParameters)
  174. m_size.setWidth(videoStream->codecpar->width);
  175. m_size.setHeight(videoStream->codecpar->height);
  176. m_frameRate = rationalToDouble(&videoStream->avg_frame_rate); // 视频帧率
  177. // 通过解码器ID获取视频解码器(新版本返回值必须使用const)
  178. const AVCodec* codec = avcodec_find_decoder(videoStream->codecpar->codec_id);
  179. m_totalFrames = videoStream->nb_frames;
  180. #if PRINT_LOG
  181. qDebug() << QString("分辨率:[w:%1,h:%2] 帧率:%3 总帧数:%4 解码器:%5")
  182. .arg(m_size.width()).arg(m_size.height()).arg(m_frameRate).arg(m_totalFrames).arg(codec->name);
  183. #endif
  184. // 分配AVCodecContext并将其字段设置为默认值。
  185. m_codecContext = avcodec_alloc_context3(codec);
  186. if(!m_codecContext)
  187. {
  188. #if PRINT_LOG
  189. qWarning() << "创建视频解码器上下文失败!";
  190. #endif
  191. free();
  192. return false;
  193. }
  194. // 使用视频流的codecpar为解码器上下文赋值
  195. ret = avcodec_parameters_to_context(m_codecContext, videoStream->codecpar);
  196. if(ret < 0)
  197. {
  198. showError(ret);
  199. free();
  200. return false;
  201. }
  202. m_codecContext->flags2 |= AV_CODEC_FLAG2_FAST; // 允许不符合规范的加速技巧。
  203. m_codecContext->thread_count = 8; // 使用8线程解码
  204. if(m_HWDecoder)
  205. {
  206. initHWDecoder(codec); // 初始化硬件解码器(在avcodec_open2前调用)
  207. }
  208. // 初始化解码器上下文,如果之前avcodec_alloc_context3传入了解码器,这里设置NULL就可以
  209. ret = avcodec_open2(m_codecContext, nullptr, nullptr);
  210. if(ret < 0)
  211. {
  212. showError(ret);
  213. free();
  214. return false;
  215. }
  216. return initObject();
  217. }
  218. /**
  219. * @brief 初始化需要用到的对象
  220. * @return
  221. */
  222. bool VideoDecode::initObject()
  223. {
  224. // 分配AVPacket并将其字段设置为默认值。
  225. m_packet = av_packet_alloc();
  226. if(!m_packet)
  227. {
  228. #if PRINT_LOG
  229. qWarning() << "av_packet_alloc() Error!";
  230. #endif
  231. free();
  232. return false;
  233. }
  234. // 分配AVFrame并将其字段设置为默认值。
  235. m_frame = av_frame_alloc();
  236. if(!m_frame)
  237. {
  238. #if PRINT_LOG
  239. qWarning() << "av_frame_alloc() Error!";
  240. #endif
  241. free();
  242. return false;
  243. }
  244. m_frameHW = av_frame_alloc();
  245. if(!m_frameHW)
  246. {
  247. #if PRINT_LOG
  248. qWarning() << "av_frame_alloc() Error!";
  249. #endif
  250. free();
  251. return false;
  252. }
  253. // 由于传递时是浅拷贝,可能显示类还没处理完成,所以如果播放完成就释放可能会崩溃;
  254. if(m_buffer)
  255. {
  256. delete [] m_buffer;
  257. m_buffer = nullptr;
  258. }
  259. // 分配图像空间
  260. int size = av_image_get_buffer_size(AV_PIX_FMT_RGBA, m_size.width(), m_size.height(), 4);
  261. /**
  262. * 【注意:】这里可以多分配一些,否则如果只是安装size分配,大部分视频图像数据拷贝没有问题,
  263. * 但是少部分视频图像在使用sws_scale()拷贝时会超出数组长度,在使用使用msvc debug模式时delete[] m_buffer会报错(HEAP CORRUPTION DETECTED: after Normal block(#32215) at 0x000001AC442830370.CRT delected that the application wrote to memory after end of heap buffer)
  264. * 特别是这个视频流http://vfx.mtime.cn/Video/2019/02/04/mp4/190204084208765161.mp4
  265. */
  266. m_buffer = new uchar[size + 1000]; // 这里多分配1000个字节就基本不会出现拷贝超出的情况了,反正不缺这点内存
  267. // m_image = new QImage(m_buffer, m_size.width(), m_size.height(), QImage::Format_RGBA8888); // 这种方式分配内存大部分情况下也可以,但是因为存在拷贝超出数组的情况,delete时也会报错
  268. m_end = false;
  269. return true;
  270. }
  271. /**
  272. * @brief 读取并返回视频图像
  273. * @return
  274. */
  275. QImage VideoDecode::read()
  276. {
  277. // 如果没有打开则返回
  278. if(!m_formatContext)
  279. {
  280. return QImage();
  281. }
  282. // 读取下一帧数据
  283. int readRet = av_read_frame(m_formatContext, m_packet);
  284. if(readRet < 0)
  285. {
  286. avcodec_send_packet(m_codecContext, m_packet); // 读取完成后向解码器中传如空AVPacket,否则无法读取出最后几帧
  287. }
  288. else
  289. {
  290. if(m_packet->stream_index == m_videoIndex) // 如果是图像数据则进行解码
  291. {
  292. // 计算当前帧时间(毫秒)
  293. #if 1 // 方法一:适用于所有场景,但是存在一定误差
  294. m_packet->pts = qRound64(m_packet->pts * (1000 * rationalToDouble(&m_formatContext->streams[m_videoIndex]->time_base)));
  295. m_packet->dts = qRound64(m_packet->dts * (1000 * rationalToDouble(&m_formatContext->streams[m_videoIndex]->time_base)));
  296. #else // 方法二:适用于播放本地视频文件,计算每一帧时间较准,但是由于网络视频流无法获取总帧数,所以无法适用
  297. m_obtainFrames++;
  298. m_packet->pts = qRound64(m_obtainFrames * (qreal(m_totalTime) / m_totalFrames));
  299. #endif
  300. // 将读取到的原始数据包传入解码器
  301. int ret = avcodec_send_packet(m_codecContext, m_packet);
  302. if(ret < 0)
  303. {
  304. showError(ret);
  305. }
  306. }
  307. }
  308. av_packet_unref(m_packet); // 释放数据包,引用计数-1,为0时释放空间
  309. int ret = avcodec_receive_frame(m_codecContext, m_frame);
  310. if(ret < 0)
  311. {
  312. av_frame_unref(m_frame);
  313. if(readRet < 0)
  314. {
  315. m_end = true; // 当无法读取到AVPacket并且解码器中也没有数据时表示读取完成
  316. }
  317. return QImage();
  318. }
  319. // 这样写是为了兼容软解码或者硬件解码打开失败情况
  320. AVFrame* m_frameTemp = m_frame;
  321. if(!m_frame->data[0]) // 如果是硬件解码就进入
  322. {
  323. m_frameTemp = m_frameHW;
  324. // 将解码后的数据从GPU拷贝到CPU
  325. if(!dataCopy())
  326. {
  327. return QImage();
  328. }
  329. }
  330. m_pts = m_frameTemp->pts;
  331. // 为什么图像转换上下文要放在这里初始化呢,是因为m_frame->format,如果使用硬件解码,解码出来的图像格式和m_codecContext->pix_fmt的图像格式不一样,就会导致无法转换为QImage
  332. if(!m_swsContext)
  333. {
  334. // 获取缓存的图像转换上下文。首先校验参数是否一致,如果校验不通过就释放资源;然后判断上下文是否存在,如果存在直接复用,如不存在进行分配、初始化操作
  335. m_swsContext = sws_getCachedContext(m_swsContext,
  336. m_frameTemp->width, // 输入图像的宽度
  337. m_frameTemp->height, // 输入图像的高度
  338. (AVPixelFormat)m_frameTemp->format, // 输入图像的像素格式
  339. m_size.width(), // 输出图像的宽度
  340. m_size.height(), // 输出图像的高度
  341. AV_PIX_FMT_RGBA, // 输出图像的像素格式
  342. SWS_BILINEAR, // 选择缩放算法(只有当输入输出图像大小不同时有效),一般选择SWS_FAST_BILINEAR
  343. nullptr, // 输入图像的滤波器信息, 若不需要传NULL
  344. nullptr, // 输出图像的滤波器信息, 若不需要传NULL
  345. nullptr); // 特定缩放算法需要的参数(?),默认为NULL
  346. if(!m_swsContext)
  347. {
  348. #if PRINT_LOG
  349. qWarning() << "sws_getCachedContext() Error!";
  350. #endif
  351. free();
  352. return QImage();
  353. }
  354. }
  355. // AVFrame转QImage
  356. uchar* data[] = {m_buffer};
  357. int lines[4];
  358. av_image_fill_linesizes(lines, AV_PIX_FMT_RGBA, m_frame->width); // 使用像素格式pix_fmt和宽度填充图像的平面线条大小。
  359. ret = sws_scale(m_swsContext, // 缩放上下文
  360. m_frameTemp->data, // 原图像数组
  361. m_frameTemp->linesize, // 包含源图像每个平面步幅的数组
  362. 0, // 开始位置
  363. m_frameTemp->height, // 行数
  364. data, // 目标图像数组
  365. lines); // 包含目标图像每个平面的步幅的数组
  366. QImage image(m_buffer, m_frameTemp->width, m_frameTemp->height, QImage::Format_RGBA8888);
  367. av_frame_unref(m_frame);
  368. av_frame_unref(m_frameHW);
  369. return image;
  370. }
  371. /********************************* FFmpeg初始化硬件后将图像数据从GPU拷贝到CPU *************************************/
  372. /**
  373. * @brief 硬件解码完成需要将数据从GPU复制到CPU
  374. * @return
  375. */
  376. bool VideoDecode::dataCopy()
  377. {
  378. if(m_frame->format != g_pixelFormat)
  379. {
  380. av_frame_unref(m_frame);
  381. return false;
  382. }
  383. int ret = av_hwframe_transfer_data(m_frameHW, m_frame, 0); // 将解码后的数据从GPU复制到CPU(m_frameHW) 这一步比较耗时,在这一步之前硬解码速度比软解码快很多
  384. if(ret < 0)
  385. {
  386. showError(ret);
  387. av_frame_unref(m_frame);
  388. return false;
  389. }
  390. av_frame_copy_props(m_frameHW, m_frame); // 仅将“metadata”字段从src复制到dst。
  391. return true;
  392. }
  393. /************************************************ END ******************************************************/
  394. /**
  395. * @brief 关闭视频播放并释放内存
  396. */
  397. void VideoDecode::close()
  398. {
  399. clear();
  400. free();
  401. m_totalTime = 0;
  402. m_videoIndex = 0;
  403. m_totalFrames = 0;
  404. m_obtainFrames = 0;
  405. m_pts = 0;
  406. m_frameRate = 0;
  407. m_size = QSize(0, 0);
  408. }
  409. /**
  410. * @brief 视频是否读取完成
  411. * @return
  412. */
  413. bool VideoDecode::isEnd()
  414. {
  415. return m_end;
  416. }
  417. /**
  418. * @brief 返回当前帧图像播放时间
  419. * @return
  420. */
  421. const qint64 &VideoDecode::pts()
  422. {
  423. return m_pts;
  424. }
  425. /**
  426. * @brief 设置是否使用硬件解码
  427. * @param flag true:使用 false:不使用
  428. */
  429. void VideoDecode::setHWDecoder(bool flag)
  430. {
  431. m_HWDecoder = flag;
  432. }
  433. /**
  434. * @brief 返回当前是否使用硬件解码
  435. * @return
  436. */
  437. bool VideoDecode::isHWDecoder()
  438. {
  439. return m_HWDecoder;
  440. }
  441. /**
  442. * @brief 显示ffmpeg函数调用异常信息
  443. * @param err
  444. */
  445. void VideoDecode::showError(int err)
  446. {
  447. #if PRINT_LOG
  448. memset(m_error, 0, ERROR_LEN); // 将数组置零
  449. av_strerror(err, m_error, ERROR_LEN);
  450. qWarning() << "DecodeVideo Error:" << m_error;
  451. #else
  452. Q_UNUSED(err)
  453. #endif
  454. }
  455. /**
  456. * @brief 将AVRational转换为double,用于计算帧率
  457. * @param rational
  458. * @return
  459. */
  460. qreal VideoDecode::rationalToDouble(AVRational* rational)
  461. {
  462. qreal frameRate = (rational->den == 0) ? 0 : (qreal(rational->num) / rational->den);
  463. return frameRate;
  464. }
  465. /**
  466. * @brief 清空读取缓冲
  467. */
  468. void VideoDecode::clear()
  469. {
  470. // 因为avformat_flush不会刷新AVIOContext (s->pb)。如果有必要,在调用此函数之前调用avio_flush(s->pb)。
  471. if(m_formatContext && m_formatContext->pb)
  472. {
  473. avio_flush(m_formatContext->pb);
  474. }
  475. if(m_formatContext)
  476. {
  477. avformat_flush(m_formatContext); // 清理读取缓冲
  478. }
  479. }
  480. void VideoDecode::free()
  481. {
  482. // 释放上下文swsContext。
  483. if(m_swsContext)
  484. {
  485. sws_freeContext(m_swsContext);
  486. m_swsContext = nullptr; // sws_freeContext不会把上下文置NULL
  487. }
  488. // 释放编解码器上下文和与之相关的所有内容,并将NULL写入提供的指针
  489. if(m_codecContext)
  490. {
  491. avcodec_free_context(&m_codecContext);
  492. }
  493. // 关闭并失败m_formatContext,并将指针置为null
  494. if(m_formatContext)
  495. {
  496. avformat_close_input(&m_formatContext);
  497. }
  498. if(hw_device_ctx)
  499. {
  500. av_buffer_unref(&hw_device_ctx);
  501. }
  502. if(m_packet)
  503. {
  504. av_packet_free(&m_packet);
  505. }
  506. if(m_frame)
  507. {
  508. av_frame_free(&m_frame);
  509. }
  510. if(m_frameHW)
  511. {
  512. av_frame_free(&m_frameHW);
  513. }
  514. }