2016-08-27 15:51:23 +00:00
|
|
|
#ifndef ENGINE_CLIENT_VIDEO_H
|
|
|
|
#define ENGINE_CLIENT_VIDEO_H
|
|
|
|
|
2021-04-30 22:42:37 +00:00
|
|
|
#include <base/system.h>
|
|
|
|
|
2021-05-01 21:33:42 +00:00
|
|
|
#include "graphics_defines.h"
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
extern "C" {
|
|
|
|
#include <libavcodec/avcodec.h>
|
|
|
|
#include <libavformat/avformat.h>
|
|
|
|
#include <libavutil/imgutils.h>
|
|
|
|
#include <libavutil/opt.h>
|
|
|
|
#include <libswresample/swresample.h>
|
|
|
|
#include <libswscale/swscale.h>
|
2016-08-27 15:51:23 +00:00
|
|
|
};
|
|
|
|
|
2019-10-26 11:54:25 +00:00
|
|
|
#include <engine/shared/demo.h>
|
2020-09-26 19:41:58 +00:00
|
|
|
#include <engine/shared/video.h>
|
2022-03-20 17:03:25 +00:00
|
|
|
|
|
|
|
#include <atomic>
|
2022-03-02 08:32:51 +00:00
|
|
|
#include <condition_variable>
|
|
|
|
#include <mutex>
|
|
|
|
#include <thread>
|
2022-03-20 17:03:25 +00:00
|
|
|
#include <vector>
|
2019-11-02 08:09:00 +00:00
|
|
|
#define ALEN 2048
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2020-12-02 18:11:19 +00:00
|
|
|
extern LOCK g_WriteLock;
|
|
|
|
|
2016-08-27 19:10:27 +00:00
|
|
|
// a wrapper around a single output AVStream
|
2020-09-26 19:41:58 +00:00
|
|
|
typedef struct OutputStream
|
|
|
|
{
|
2022-03-02 08:32:51 +00:00
|
|
|
AVStream *pSt = nullptr;
|
|
|
|
AVCodecContext *pEnc = nullptr;
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2016-08-30 23:39:59 +00:00
|
|
|
/* pts of the next frame that will be generated */
|
2022-03-02 08:32:51 +00:00
|
|
|
int64_t NextPts = 0;
|
|
|
|
int64_t m_SamplesCount = 0;
|
|
|
|
int64_t m_SamplesFrameCount = 0;
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
std::vector<AVFrame *> m_vpFrames;
|
|
|
|
std::vector<AVFrame *> m_vpTmpFrames;
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
std::vector<struct SwsContext *> m_vpSwsCtxs;
|
|
|
|
std::vector<struct SwrContext *> m_vpSwrCtxs;
|
2016-08-27 19:10:27 +00:00
|
|
|
} OutputStream;
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2016-08-30 23:39:59 +00:00
|
|
|
class CVideo : public IVideo
|
2016-08-27 15:51:23 +00:00
|
|
|
{
|
|
|
|
public:
|
2022-03-02 08:32:51 +00:00
|
|
|
CVideo(class CGraphics_Threaded *pGraphics, class ISound *pSound, class IStorage *pStorage, class IConsole *pConsole, int width, int height, const char *name);
|
2016-08-27 15:51:23 +00:00
|
|
|
~CVideo();
|
|
|
|
|
2020-06-22 21:59:37 +00:00
|
|
|
virtual void Start();
|
|
|
|
virtual void Stop();
|
|
|
|
virtual void Pause(bool Pause);
|
|
|
|
virtual bool IsRecording() { return m_Recording; }
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2020-06-22 21:59:37 +00:00
|
|
|
virtual void NextVideoFrame();
|
|
|
|
virtual void NextVideoFrameThread();
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
virtual void NextAudioFrame(ISoundMixFunc Mix);
|
|
|
|
virtual void NextAudioFrameTimeline(ISoundMixFunc Mix);
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
static IVideo *Current() { return IVideo::ms_pCurrentVideo; }
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2019-11-12 13:41:30 +00:00
|
|
|
static void Init() { av_log_set_level(AV_LOG_DEBUG); }
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2016-08-27 15:51:23 +00:00
|
|
|
private:
|
2022-03-02 08:32:51 +00:00
|
|
|
void RunVideoThread(size_t ParentThreadIndex, size_t ThreadIndex);
|
|
|
|
void FillVideoFrame(size_t ThreadIndex);
|
|
|
|
void ReadRGBFromGL(size_t ThreadIndex);
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
void RunAudioThread(size_t ParentThreadIndex, size_t ThreadIndex);
|
|
|
|
void FillAudioFrame(size_t ThreadIndex);
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2020-09-13 20:49:50 +00:00
|
|
|
bool OpenVideo();
|
|
|
|
bool OpenAudio();
|
2020-06-22 21:59:37 +00:00
|
|
|
AVFrame *AllocPicture(enum AVPixelFormat PixFmt, int Width, int Height);
|
2021-06-23 05:05:49 +00:00
|
|
|
AVFrame *AllocAudioFrame(enum AVSampleFormat SampleFmt, uint64_t ChannelLayout, int SampleRate, int NbSamples);
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
void WriteFrame(OutputStream *pStream, size_t ThreadIndex) REQUIRES(g_WriteLock);
|
2020-09-26 19:41:58 +00:00
|
|
|
void FinishFrames(OutputStream *pStream);
|
2020-06-22 21:59:37 +00:00
|
|
|
void CloseStream(OutputStream *pStream);
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2022-01-17 23:30:34 +00:00
|
|
|
bool AddStream(OutputStream *pStream, AVFormatContext *pOC, const AVCodec **ppCodec, enum AVCodecID CodecId);
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
class CGraphics_Threaded *m_pGraphics;
|
|
|
|
class IStorage *m_pStorage;
|
2022-03-02 08:32:51 +00:00
|
|
|
class ISound *m_pSound;
|
2016-08-27 15:51:23 +00:00
|
|
|
|
|
|
|
int m_Width;
|
|
|
|
int m_Height;
|
2019-09-27 03:06:02 +00:00
|
|
|
char m_Name[256];
|
2019-10-31 14:01:12 +00:00
|
|
|
//FILE *m_dbgfile;
|
2022-03-02 08:32:51 +00:00
|
|
|
uint64_t m_VSeq = 0;
|
|
|
|
uint64_t m_ASeq = 0;
|
|
|
|
uint64_t m_Vframe;
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2016-08-31 16:07:27 +00:00
|
|
|
int m_FPS;
|
|
|
|
|
2016-08-30 23:39:59 +00:00
|
|
|
bool m_Started;
|
2016-08-27 15:51:23 +00:00
|
|
|
bool m_Recording;
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
size_t m_VideoThreads = 2;
|
|
|
|
size_t m_CurVideoThreadIndex = 0;
|
|
|
|
size_t m_AudioThreads = 2;
|
|
|
|
size_t m_CurAudioThreadIndex = 0;
|
|
|
|
|
|
|
|
struct SVideoRecorderThread
|
|
|
|
{
|
|
|
|
std::thread m_Thread;
|
|
|
|
std::mutex m_Mutex;
|
|
|
|
std::condition_variable m_Cond;
|
|
|
|
|
|
|
|
bool m_Started = false;
|
|
|
|
bool m_Finished = false;
|
|
|
|
bool m_HasVideoFrame = false;
|
|
|
|
|
|
|
|
std::mutex m_VideoFillMutex;
|
|
|
|
std::condition_variable m_VideoFillCond;
|
|
|
|
uint64_t m_VideoFrameToFill = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<SVideoRecorderThread>> m_vVideoThreads;
|
|
|
|
|
|
|
|
struct SAudioRecorderThread
|
|
|
|
{
|
|
|
|
std::thread m_Thread;
|
|
|
|
std::mutex m_Mutex;
|
|
|
|
std::condition_variable m_Cond;
|
|
|
|
|
|
|
|
bool m_Started = false;
|
|
|
|
bool m_Finished = false;
|
|
|
|
bool m_HasAudioFrame = false;
|
|
|
|
|
|
|
|
std::mutex m_AudioFillMutex;
|
|
|
|
std::condition_variable m_AudioFillCond;
|
|
|
|
uint64_t m_AudioFrameToFill = 0;
|
|
|
|
int64_t m_SampleCountStart = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<std::unique_ptr<SAudioRecorderThread>> m_vAudioThreads;
|
|
|
|
|
|
|
|
std::atomic<int32_t> m_ProcessingVideoFrame;
|
|
|
|
std::atomic<int32_t> m_ProcessingAudioFrame;
|
2016-08-30 23:39:59 +00:00
|
|
|
|
2022-03-20 17:03:25 +00:00
|
|
|
std::atomic<bool> m_NextFrame;
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2016-08-27 19:10:27 +00:00
|
|
|
bool m_HasAudio;
|
|
|
|
|
2022-03-02 08:32:51 +00:00
|
|
|
struct SVideoSoundBuffer
|
|
|
|
{
|
|
|
|
int16_t m_aBuffer[ALEN * 2];
|
|
|
|
};
|
|
|
|
std::vector<SVideoSoundBuffer> m_vBuffer;
|
|
|
|
std::vector<std::vector<uint8_t>> m_vPixelHelper;
|
2016-08-27 15:51:23 +00:00
|
|
|
|
2016-08-27 19:10:27 +00:00
|
|
|
OutputStream m_VideoStream;
|
|
|
|
OutputStream m_AudioStream;
|
|
|
|
|
2022-01-17 23:30:34 +00:00
|
|
|
const AVCodec *m_VideoCodec;
|
|
|
|
const AVCodec *m_AudioCodec;
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
AVDictionary *m_pOptDict;
|
2016-08-27 19:10:27 +00:00
|
|
|
|
2020-09-26 19:41:58 +00:00
|
|
|
AVFormatContext *m_pFormatContext;
|
2022-01-17 23:30:34 +00:00
|
|
|
const AVOutputFormat *m_pFormat;
|
2016-08-27 15:51:23 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|