| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | //  SPDX-License-Identifier: MIT
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //  EmulationStation Desktop Edition
 | 
					
						
							|  |  |  | //  VideoFFmpegComponent.h
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //  Video player based on FFmpeg.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef ES_CORE_COMPONENTS_VIDEO_FFMPEG_COMPONENT_H
 | 
					
						
							|  |  |  | #define ES_CORE_COMPONENTS_VIDEO_FFMPEG_COMPONENT_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:55:00 +00:00
										 |  |  | // Audio buffer in seconds.
 | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  | #define AUDIO_BUFFER 0.1l
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "VideoComponent.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-07 18:31:46 +00:00
										 |  |  | extern "C" { | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | #include <libavcodec/avcodec.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  | #include <libavfilter/avfilter.h>
 | 
					
						
							|  |  |  | #include <libavfilter/buffersink.h>
 | 
					
						
							|  |  |  | #include <libavfilter/buffersrc.h>
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | #include <libavformat/avformat.h>
 | 
					
						
							|  |  |  | #include <libavutil/imgutils.h>
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-16 16:49:05 +00:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | #include <queue>
 | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class VideoFFmpegComponent : public VideoComponent | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2022-01-19 17:01:54 +00:00
										 |  |  |     VideoFFmpegComponent(); | 
					
						
							| 
									
										
										
										
											2022-02-19 16:04:23 +00:00
										 |  |  |     virtual ~VideoFFmpegComponent() { stopVideoPlayer(); } | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Resize the video to fit this size. If one axis is zero, scale that axis to maintain
 | 
					
						
							|  |  |  |     // aspect ratio. If both are non-zero, potentially break the aspect ratio. If both are
 | 
					
						
							|  |  |  |     // zero, no resizing. This can be set before or after a video is loaded.
 | 
					
						
							|  |  |  |     // setMaxSize() and setResize() are mutually exclusive.
 | 
					
						
							|  |  |  |     void setResize(float width, float height) override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Resize the video to be as large as possible but fit within a box of this size.
 | 
					
						
							|  |  |  |     // This can be set before or after a video is loaded.
 | 
					
						
							|  |  |  |     // Never breaks the aspect ratio. setMaxSize() and setResize() are mutually exclusive.
 | 
					
						
							|  |  |  |     void setMaxSize(float width, float height) override; | 
					
						
							| 
									
										
										
										
											2022-02-19 16:04:23 +00:00
										 |  |  |     // Basic video controls.
 | 
					
						
							|  |  |  |     void stopVideoPlayer() override; | 
					
						
							|  |  |  |     void pauseVideoPlayer() override; | 
					
						
							|  |  |  |     // Handle looping of the video. Must be called periodically.
 | 
					
						
							|  |  |  |     void handleLooping() override; | 
					
						
							|  |  |  |     // Used to immediately mute audio even if there are samples to play in the buffer.
 | 
					
						
							|  |  |  |     void muteVideoPlayer() override; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2022-02-19 16:04:23 +00:00
										 |  |  |     void startVideoStream() override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     // Calculates the correct mSize from our resizing information (set by setResize/setMaxSize).
 | 
					
						
							|  |  |  |     // Used internally whenever the resizing parameters or texture change.
 | 
					
						
							|  |  |  |     void resize(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-15 17:30:31 +00:00
										 |  |  |     void render(const glm::mat4& parentTrans) override; | 
					
						
							| 
									
										
										
										
											2022-01-18 16:40:47 +00:00
										 |  |  |     void updatePlayer() override; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  |     // This will run the frame processing in a separate thread.
 | 
					
						
							|  |  |  |     void frameProcessing(); | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     // Setup libavfilter.
 | 
					
						
							|  |  |  |     bool setupVideoFilters(); | 
					
						
							|  |  |  |     bool setupAudioFilters(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Read frames from the video file and add them to the filter source.
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     void readFrames(); | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     // Get the frames that have been processed by the filters.
 | 
					
						
							|  |  |  |     void getProcessedFrames(); | 
					
						
							|  |  |  |     // Output frames to AudioManager and to the video surface (via the main thread).
 | 
					
						
							| 
									
										
										
										
											2021-05-10 16:08:45 +00:00
										 |  |  |     void outputFrames(); | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     // Calculate the black rectangle that is shown behind videos with non-standard aspect ratios.
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     void calculateBlackRectangle(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-14 17:13:25 +00:00
										 |  |  |     // Detect and initialize the hardware decoder.
 | 
					
						
							|  |  |  |     static void detectHWDecoder(); | 
					
						
							|  |  |  |     bool decoderInitHW(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     static enum AVHWDeviceType sDeviceType; | 
					
						
							|  |  |  |     static enum AVPixelFormat sPixelFormat; | 
					
						
							|  |  |  |     static std::vector<std::string> sSWDecodedVideos; | 
					
						
							|  |  |  |     static std::vector<std::string> sHWDecodedVideos; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     std::shared_ptr<TextureResource> mTexture; | 
					
						
							|  |  |  |     std::vector<float> mVideoRectangleCoords; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:49:24 +00:00
										 |  |  |     std::unique_ptr<std::thread> mFrameProcessingThread; | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  |     std::mutex mPictureMutex; | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     std::mutex mAudioMutex; | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     AVFormatContext* mFormatContext; | 
					
						
							|  |  |  |     AVStream* mVideoStream; | 
					
						
							|  |  |  |     AVStream* mAudioStream; | 
					
						
							| 
									
										
										
										
											2021-07-07 18:31:46 +00:00
										 |  |  |     AVCodec* mVideoCodec; | 
					
						
							|  |  |  |     AVCodec* mAudioCodec; | 
					
						
							| 
									
										
										
										
											2021-07-14 17:13:25 +00:00
										 |  |  |     AVCodec* mHardwareCodec; | 
					
						
							|  |  |  |     AVBufferRef* mHwContext; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     AVCodecContext* mVideoCodecContext; | 
					
						
							|  |  |  |     AVCodecContext* mAudioCodecContext; | 
					
						
							|  |  |  |     int mVideoStreamIndex; | 
					
						
							|  |  |  |     int mAudioStreamIndex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AVPacket* mPacket; | 
					
						
							|  |  |  |     AVFrame* mVideoFrame; | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     AVFrame* mVideoFrameResampled; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     AVFrame* mAudioFrame; | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     AVFrame* mAudioFrameResampled; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     struct VideoFrame { | 
					
						
							|  |  |  |         std::vector<uint8_t> frameRGBA; | 
					
						
							|  |  |  |         int width; | 
					
						
							|  |  |  |         int height; | 
					
						
							|  |  |  |         double pts; | 
					
						
							| 
									
										
										
										
											2021-05-19 17:47:40 +00:00
										 |  |  |         double frameDuration; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     struct AudioFrame { | 
					
						
							|  |  |  |         std::vector<uint8_t> resampledData; | 
					
						
							|  |  |  |         double pts; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  |     struct OutputPicture { | 
					
						
							|  |  |  |         std::vector<uint8_t> pictureRGBA; | 
					
						
							| 
									
										
										
										
											2021-05-19 17:47:40 +00:00
										 |  |  |         bool hasBeenRendered; | 
					
						
							|  |  |  |         int width; | 
					
						
							|  |  |  |         int height; | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     std::queue<VideoFrame> mVideoFrameQueue; | 
					
						
							|  |  |  |     std::queue<AudioFrame> mAudioFrameQueue; | 
					
						
							| 
									
										
										
										
											2021-05-11 15:35:55 +00:00
										 |  |  |     OutputPicture mOutputPicture; | 
					
						
							| 
									
										
										
										
											2021-05-29 08:58:51 +00:00
										 |  |  |     std::vector<uint8_t> mOutputAudio; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AVFilterContext* mVBufferSrcContext; | 
					
						
							|  |  |  |     AVFilterContext* mVBufferSinkContext; | 
					
						
							|  |  |  |     AVFilterGraph* mVFilterGraph; | 
					
						
							|  |  |  |     AVFilterInOut* mVFilterInputs; | 
					
						
							|  |  |  |     AVFilterInOut* mVFilterOutputs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AVFilterContext* mABufferSrcContext; | 
					
						
							|  |  |  |     AVFilterContext* mABufferSinkContext; | 
					
						
							|  |  |  |     AVFilterGraph* mAFilterGraph; | 
					
						
							|  |  |  |     AVFilterInOut* mAFilterInputs; | 
					
						
							|  |  |  |     AVFilterInOut* mAFilterOutputs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int mVideoTargetQueueSize; | 
					
						
							|  |  |  |     int mAudioTargetQueueSize; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  |     double mVideoTimeBase; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Used for audio and video synchronization.
 | 
					
						
							|  |  |  |     std::chrono::high_resolution_clock::time_point mTimeReference; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 17:47:33 +00:00
										 |  |  |     int mAudioFrameCount; | 
					
						
							|  |  |  |     int mVideoFrameCount; | 
					
						
							| 
									
										
										
										
											2021-11-07 17:14:38 +00:00
										 |  |  |     int mVideoFrameReadCount; | 
					
						
							|  |  |  |     int mVideoFrameDroppedCount; | 
					
						
							| 
									
										
										
										
											2021-07-09 17:47:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-16 16:49:05 +00:00
										 |  |  |     std::atomic<double> mAccumulatedTime; | 
					
						
							|  |  |  |     std::atomic<bool> mStartTimeAccumulation; | 
					
						
							|  |  |  |     std::atomic<bool> mDecodedFrame; | 
					
						
							|  |  |  |     std::atomic<bool> mEndOfVideo; | 
					
						
							| 
									
										
										
										
											2021-07-14 17:13:25 +00:00
										 |  |  |     bool mSWDecoder; | 
					
						
							| 
									
										
										
										
											2021-05-09 20:52:26 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // ES_CORE_COMPONENTS_VIDEO_FFMPEG_COMPONENT_H
 |