DecodeVedio.h 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. #ifndef DECODEVEDIO_H
  2. #define DECODEVEDIO_H
  3. #include <QObject>
  4. #include <QQueue>
  5. #include <QTimer>
  6. #include <QMutex>
  7. #include <QWaitCondition>
  8. #include <QImage>
  9. #include "RingQueue/RingQueue.hpp"
  10. // #include "threadcontroller.h"
  11. extern "C"
  12. {
  13. // #include <libavcodec/avcodec.h>
  14. #include <libavformat/avformat.h>
  15. // #include <libswscale/swscale.h>
  16. // #include <libavutil/imgutils.h>
  17. }
  18. /**
  19. * 使用方式:
  20. * 1. 初始化FFmpeg:initFFmpeg()
  21. * 2. 开启解码线程:startDecodeVedio()
  22. * 3. 获取一帧图像:getOneImage()
  23. * 4. 停止解码线程:stopDecodeVedio()
  24. * 5. 在初始化完成后,未进入第二步的之前,可以获取视频的宽高信息,也可以设置宽高信息
  25. *
  26. */
  27. class DecodeVedio : public QObject
  28. {
  29. Q_OBJECT
  30. enum class DecodeState
  31. {
  32. NONE = 0,
  33. DecodeRun, /* 解码运行中 */
  34. DecodePause, /* 暂停解码 */
  35. DecodeSeek, /* 跳转中 */
  36. DecodeStop, /* 停止解码,但是并没有退出解码线程 */
  37. DecodeExit /* 退出解码 */
  38. };
  39. public:
  40. explicit DecodeVedio(QThread* thread, QObject* parent = nullptr);
  41. ~DecodeVedio();
  42. /* 打开视频,同时初始化解码器) */
  43. void openVedio(const QString& fileName);
  44. /* 开始解码视频,开始前先打开视频文件 */
  45. void startDecodeVedio();
  46. /* 停止解码视频,也是停止线程 */
  47. void stopDecodeVedio();
  48. /* 获取解码状态 */
  49. bool isDecoding() { return m_threadRuning; }
  50. /* 设置当前播放位置,单位ms */
  51. void setCurrentPos(qint64 pos);
  52. /* 获取当前播放位置,单位ms */
  53. qint64 getCurrentPos();
  54. /* 获取视频时长 */
  55. qint64 getDuration();
  56. qint64 getTotalFrame() { return m_totalFrame; }
  57. /* 获取一帧图像 */
  58. QImage* getOneImage();
  59. /* 获取一帧图像,直到有图像为止,可以设置超时时间 */
  60. QImage* getOneImageUntilHave(int timeOut = -1);
  61. /* 获取帧数 */
  62. int getFPS() const { return m_fps; }
  63. /* 设置帧数 */
  64. void setFPS(int fps) { m_fps = fps; }
  65. /* 获取图像宽度 */
  66. QSize getSrcVideoSize() const {return m_srcSize; }
  67. /* 获取解码器名称(编码格式) */
  68. QString getDecoderName() const { return m_decoderName; }
  69. /* 获取硬件解码器名称列表 */
  70. QStringList getHWDecoderList() const { return m_listDecoderName; }
  71. /* 查找硬件解码器 */
  72. static void findHWDecoder(QStringList& listDecoderName);
  73. signals:
  74. void signal_oneImage(); /* 一帧图像信号 */
  75. void signal_playCompleted(); /* 播放完成信号 */
  76. void signal_startDecode(); /* 开始解码信号 */
  77. private:
  78. /* 查找硬件解码器 */
  79. void findHWDecoder();
  80. /* 初始化硬件解码器 */
  81. void initHWDecoder(const AVCodec* codec);
  82. /* 拷贝数据,从GPU显存拷贝到内存中 */
  83. bool copyDataFromGPU(AVFrame* pFrameHW, AVFrame* pFrameSRC);
  84. /* 软解码线程 */
  85. void threadDecodeUsingCPU();
  86. /* 硬件解码线程 */
  87. void threadDecodeUsingGPU();
  88. /* 退出线程 */
  89. void exitThread();
  90. /* 暂停解码 */
  91. void pauseDecode();
  92. /* 继续解码 */
  93. void continueDecode();
  94. /* 将AVRational转换为double */
  95. qreal rationalToDouble(AVRational* rational);
  96. /* 释放所有资源 */
  97. void freeAll();
  98. private slots:
  99. void do_startDecodeVedio(); /* 开启解码 */
  100. private:
  101. QThread* m_thread = nullptr; /* 解码线程 */
  102. /* 线程状态 */
  103. std::atomic_bool m_threadRuning = false; /* 解码线程是运行标志 */
  104. std::atomic_bool m_initFFmpeg = false; /* ffmpeg初始化标志 */
  105. std::atomic_bool m_pauseDecode = false; /* 暂停解码 */
  106. std::atomic_bool m_decodeStatus = false; /* 解码状态,这里主要是检测是否暂停解码 */
  107. std::atomic_bool m_isSeek = false; /* 是否在跳转中 */
  108. std::atomic_bool m_flushDecoder = false; /* 刷新解码器 */
  109. std::atomic<DecodeState> m_decodeState = DecodeState::NONE;
  110. /* 视频解码相关变量信息 */
  111. QString m_fileName; /* 解码的视频文件名称 */
  112. AVFormatContext *m_pFormatContext = nullptr; /* 格式上下文,贯穿全局 */
  113. AVCodecContext *m_pCodecContext = nullptr; /* 解码器上下文 */
  114. AVPacket* m_packet = nullptr; /* 存储解码前的数据,一个数据包 */
  115. AVFrame* m_pFrameSRC = nullptr; /* 存储解码后的一帧数据原始视频编码 */
  116. AVFrame* m_pFrameHW = nullptr; /* 存储解码后的一帧数据,硬件解码 */
  117. struct SwsContext *m_sws_ctx = nullptr; /* 视频转换上下文 */
  118. uint8_t *m_buffer = nullptr; /* 存储解码后的一帧数据,RGB格式 */
  119. int m_videoStream = -1; /* 记录视频流是第几个流 */
  120. bool m_supportHWDecoder = false; /* 是否使用硬件解码 */
  121. AVBufferRef* m_hw_device_ctx = nullptr; /* 对数据缓冲区的引用 */
  122. QList<int> m_listHWDeviceType; /* 保存当前环境支持的硬件解码器 */
  123. QStringList m_listDecoderName; /* 硬件解码器列表名称 */
  124. /* 视频相关信息 */
  125. QSize m_srcSize; /* 原始视频分辨率大小 */
  126. qint64 m_totalFrame = 0; /* 视频总帧数 */
  127. int m_fps = 0; /* 每秒的帧数 */
  128. qint64 m_duration = 0; /* 视频时长,单位毫秒 */
  129. qint64 m_startPos = 0; /* 开始播放的位置,摄像机视频的位置不是从0开始的,需要在初始化的时候取出这个值 */
  130. std::atomic<qint64> m_pts = 0; /* 当前帧显示时间,也就是当前的进度时间 */
  131. qint64 m_targetPos = -1; /* 跳转的目标播放位置 */
  132. qint64 m_currentFrame = 0; /* 当前已播放的帧数 */
  133. QString m_decoderName; /* 解码器名称 */
  134. RingQueue<QImage*> m_queueImage; /* 环形队列,存储生成的图像 */
  135. };
  136. #endif /* DECODEVEDIO_H */