21#include <freerdp/config.h>
23#include <winpr/wlog.h>
24#include <freerdp/log.h>
25#include <freerdp/codec/h264.h>
26#include <libavcodec/avcodec.h>
27#include <libavutil/opt.h>
31#ifdef WITH_VIDEOTOOLBOX
32#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(55, 9, 0)
33#include <libavutil/hwcontext.h>
35#pragma warning You have asked for VideoToolbox decoding, \
36 but your version of libavutil is too old !Disabling.
37#undef WITH_VIDEOTOOLBOX
42#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(55, 9, 0)
43#include <libavutil/hwcontext.h>
45#pragma warning You have asked for VA - API decoding, \
46 but your version of libavutil is too old !Disabling.
52#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(54, 59, 100)
53#define AV_CODEC_ID_H264 CODEC_ID_H264
56#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(56, 34, 2)
57#define AV_CODEC_FLAG_LOOP_FILTER CODEC_FLAG_LOOP_FILTER
58#define AV_CODEC_CAP_TRUNCATED CODEC_CAP_TRUNCATED
59#define AV_CODEC_FLAG_TRUNCATED CODEC_FLAG_TRUNCATED
62#if LIBAVUTIL_VERSION_MAJOR < 52
63#define AV_PIX_FMT_YUV420P PIX_FMT_YUV420P
68#if !defined(av_err2str)
69static inline char* error_string(
char* errbuf,
size_t errbuf_size,
int errnum)
71 av_strerror(errnum, errbuf, errbuf_size);
75#define av_err2str(errnum) error_string((char[64])WINPR_C_ARRAY_INIT, 64, errnum)
78#if defined(WITH_VAAPI) || defined(WITH_VAAPI_H264_ENCODING)
79static const char* get_vaapi_device(
void)
81 static char device[MAX_PATH] = WINPR_C_ARRAY_INIT;
82 static bool initialized =
false;
87 const char* env = getenv(
"FREERDP_VAAPI_DEVICE");
89 (void)_snprintf(device,
sizeof(device),
"%s", env);
91 (
void)_snprintf(device,
sizeof(device),
"/dev/dri/renderD128");
99 const AVCodec* codecDecoder;
100 AVCodecContext* codecDecoderContext;
101 const AVCodec* codecEncoder;
102 AVCodecContext* codecEncoderContext;
103 AVCodecParserContext* codecParser;
105#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 133, 100)
106 AVPacket bufferpacket;
109#if defined(WITH_VAAPI) || defined(WITH_VAAPI_H264_ENCODING) || defined(WITH_VIDEOTOOLBOX)
111 AVFrame* hwVideoFrame;
112 enum AVPixelFormat hw_pix_fmt;
113#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(57, 80, 100) || defined(WITH_VIDEOTOOLBOX)
114 AVBufferRef* hw_frames_ctx;
118} H264_CONTEXT_LIBAVCODEC;
120static void libavcodec_destroy_encoder_context(H264_CONTEXT* WINPR_RESTRICT h264)
122 H264_CONTEXT_LIBAVCODEC* sys =
nullptr;
124 if (!h264 || !h264->subsystem)
127 sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
129 if (sys->codecEncoderContext)
131#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(55, 69, 100)
132 avcodec_free_context(&sys->codecEncoderContext);
134 avcodec_close(sys->codecEncoderContext);
135 av_free(sys->codecEncoderContext);
139 sys->codecEncoderContext =
nullptr;
142#ifdef WITH_VAAPI_H264_ENCODING
143static int set_hw_frames_ctx(H264_CONTEXT* WINPR_RESTRICT h264)
145 H264_CONTEXT_LIBAVCODEC* sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
146 AVBufferRef* hw_frames_ref =
nullptr;
147 AVHWFramesContext* frames_ctx =
nullptr;
150 if (!(hw_frames_ref = av_hwframe_ctx_alloc(sys->hwctx)))
152 WLog_Print(h264->log, WLOG_ERROR,
"Failed to create VAAPI frame context");
155 frames_ctx = (AVHWFramesContext*)(hw_frames_ref->data);
156 frames_ctx->format = AV_PIX_FMT_VAAPI;
157 frames_ctx->sw_format = AV_PIX_FMT_NV12;
158 frames_ctx->width = sys->codecEncoderContext->width;
159 frames_ctx->height = sys->codecEncoderContext->height;
160 frames_ctx->initial_pool_size = 20;
161 if ((err = av_hwframe_ctx_init(hw_frames_ref)) < 0)
163 WLog_Print(h264->log, WLOG_ERROR,
164 "Failed to initialize VAAPI frame context."
167 av_buffer_unref(&hw_frames_ref);
170 sys->codecEncoderContext->hw_frames_ctx = av_buffer_ref(hw_frames_ref);
171 if (!sys->codecEncoderContext->hw_frames_ctx)
172 err = AVERROR(ENOMEM);
174 av_buffer_unref(&hw_frames_ref);
179static BOOL libavcodec_create_encoder_context(H264_CONTEXT* WINPR_RESTRICT h264)
181 BOOL recreate = FALSE;
182 H264_CONTEXT_LIBAVCODEC* sys =
nullptr;
184 if (!h264 || !h264->subsystem)
187 if ((h264->width > INT_MAX) || (h264->height > INT_MAX))
190 sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
191 if (!sys || !sys->codecEncoder)
194 recreate = !sys->codecEncoderContext;
196 if (sys->codecEncoderContext)
198 if ((sys->codecEncoderContext->width != (
int)h264->width) ||
199 (sys->codecEncoderContext->height != (
int)h264->height))
206 libavcodec_destroy_encoder_context(h264);
208 sys->codecEncoderContext = avcodec_alloc_context3(sys->codecEncoder);
210 if (!sys->codecEncoderContext)
213 switch (h264->RateControlMode)
215 case H264_RATECONTROL_VBR:
216 sys->codecEncoderContext->bit_rate = h264->BitRate;
219 case H264_RATECONTROL_CQP:
220 if (av_opt_set_int(sys->codecEncoderContext,
"qp", h264->QP, AV_OPT_SEARCH_CHILDREN) <
223 WLog_Print(h264->log, WLOG_ERROR,
"av_opt_set_int failed");
231 sys->codecEncoderContext->width = (int)MIN(INT32_MAX, h264->width);
232 sys->codecEncoderContext->height = (int)MIN(INT32_MAX, h264->height);
233 sys->codecEncoderContext->delay = 0;
234#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(56, 13, 100)
235 sys->codecEncoderContext->framerate =
236 (AVRational){ WINPR_ASSERTING_INT_CAST(
int, h264->FrameRate), 1 };
238 sys->codecEncoderContext->time_base =
239 (AVRational){ 1, WINPR_ASSERTING_INT_CAST(
int, h264->FrameRate) };
240 av_opt_set(sys->codecEncoderContext,
"tune",
"zerolatency", AV_OPT_SEARCH_CHILDREN);
242 sys->codecEncoderContext->flags |= AV_CODEC_FLAG_LOOP_FILTER;
244#ifdef WITH_VAAPI_H264_ENCODING
247 av_opt_set(sys->codecEncoderContext,
"preset",
"veryslow", AV_OPT_SEARCH_CHILDREN);
249 sys->codecEncoderContext->pix_fmt = AV_PIX_FMT_VAAPI;
251 if (set_hw_frames_ctx(h264) < 0)
257 av_opt_set(sys->codecEncoderContext,
"preset",
"medium", AV_OPT_SEARCH_CHILDREN);
258 sys->codecEncoderContext->pix_fmt = AV_PIX_FMT_YUV420P;
261 if (avcodec_open2(sys->codecEncoderContext, sys->codecEncoder,
nullptr) < 0)
266 libavcodec_destroy_encoder_context(h264);
270static int libavcodec_decompress(H264_CONTEXT* WINPR_RESTRICT h264,
271 const BYTE* WINPR_RESTRICT pSrcData, UINT32 SrcSize)
281 AVPacket* packet =
nullptr;
284 WINPR_ASSERT(pSrcData || (SrcSize == 0));
286 H264_CONTEXT_LIBAVCODEC* sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
287 BYTE** pYUVData = h264->pYUVData;
288 UINT32* iStride = h264->iStride;
292#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 133, 100)
293 packet = &sys->bufferpacket;
294 WINPR_ASSERT(packet);
295 av_init_packet(packet);
297 packet = av_packet_alloc();
301 WLog_Print(h264->log, WLOG_ERROR,
"Failed to allocate AVPacket");
306 packet->data = cnv.pv;
307 packet->size = (int)MIN(SrcSize, INT32_MAX);
309 WINPR_ASSERT(sys->codecDecoderContext);
311#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 48, 101)
312 status = avcodec_send_packet(sys->codecDecoderContext, packet);
316 WLog_Print(h264->log, WLOG_ERROR,
"Failed to decode video frame (status=%d)", status);
320 sys->videoFrame->format = AV_PIX_FMT_YUV420P;
322#if defined(WITH_VAAPI) || defined(WITH_VIDEOTOOLBOX)
323 status = avcodec_receive_frame(sys->codecDecoderContext,
324 sys->hwctx ? sys->hwVideoFrame : sys->videoFrame);
326 status = avcodec_receive_frame(sys->codecDecoderContext, sys->videoFrame);
328 if (status == AVERROR(EAGAIN))
334 gotFrame = (status == 0);
336#if defined(WITH_VAAPI) || defined(WITH_VIDEOTOOLBOX)
338 avcodec_decode_video2(sys->codecDecoderContext,
339 sys->hwctx ? sys->hwVideoFrame : sys->videoFrame, &gotFrame, packet);
341 status = avcodec_decode_video2(sys->codecDecoderContext, sys->videoFrame, &gotFrame, packet);
346 WLog_Print(h264->log, WLOG_ERROR,
"Failed to decode video frame (status=%d)", status);
350#if defined(WITH_VAAPI) || defined(WITH_VIDEOTOOLBOX)
354 if (sys->hwVideoFrame->format == sys->hw_pix_fmt)
356 sys->videoFrame->width = sys->hwVideoFrame->width;
357 sys->videoFrame->height = sys->hwVideoFrame->height;
358 status = av_hwframe_transfer_data(sys->videoFrame, sys->hwVideoFrame, 0);
362 status = av_frame_copy(sys->videoFrame, sys->hwVideoFrame);
366 gotFrame = (status == 0);
370 WLog_Print(h264->log, WLOG_ERROR,
"Failed to transfer video frame (status=%d) (%s)", status,
379 WINPR_ASSERT(sys->videoFrame);
381 pYUVData[0] = sys->videoFrame->data[0];
382 pYUVData[1] = sys->videoFrame->data[1];
383 pYUVData[2] = sys->videoFrame->data[2];
384 iStride[0] = (UINT32)MAX(0, sys->videoFrame->linesize[0]);
385 iStride[1] = (UINT32)MAX(0, sys->videoFrame->linesize[1]);
386 iStride[2] = (UINT32)MAX(0, sys->videoFrame->linesize[2]);
393#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 133, 100)
394 av_packet_unref(packet);
396 av_packet_free(&packet);
402static int libavcodec_compress(H264_CONTEXT* WINPR_RESTRICT h264,
403 const BYTE** WINPR_RESTRICT pSrcYuv,
404 const UINT32* WINPR_RESTRICT pStride,
405 BYTE** WINPR_RESTRICT ppDstData, UINT32* WINPR_RESTRICT pDstSize)
418 H264_CONTEXT_LIBAVCODEC* sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
421 if (!libavcodec_create_encoder_context(h264))
424#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 133, 100)
425 sys->packet = &sys->bufferpacket;
426 av_packet_unref(sys->packet);
427 av_init_packet(sys->packet);
429 av_packet_free(&sys->packet);
430 sys->packet = av_packet_alloc();
434 WLog_Print(h264->log, WLOG_ERROR,
"Failed to allocate AVPacket");
438 WINPR_ASSERT(sys->packet);
439 sys->packet->data =
nullptr;
440 sys->packet->size = 0;
442 WINPR_ASSERT(sys->videoFrame);
443 WINPR_ASSERT(sys->codecEncoderContext);
444 sys->videoFrame->format = AV_PIX_FMT_YUV420P;
445 sys->videoFrame->width = sys->codecEncoderContext->width;
446 sys->videoFrame->height = sys->codecEncoderContext->height;
447#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(52, 48, 100)
448 sys->videoFrame->colorspace = AVCOL_SPC_BT709;
450#if LIBAVUTIL_VERSION_INT >= AV_VERSION_INT(52, 92, 100)
451 sys->videoFrame->chroma_location = AVCHROMA_LOC_LEFT;
453 cnv.cpv = pSrcYuv[0];
454 sys->videoFrame->data[0] = cnv.pv;
456 cnv.cpv = pSrcYuv[1];
457 sys->videoFrame->data[1] = cnv.pv;
459 cnv.cpv = pSrcYuv[2];
460 sys->videoFrame->data[2] = cnv.pv;
462 sys->videoFrame->linesize[0] = (int)pStride[0];
463 sys->videoFrame->linesize[1] = (int)pStride[1];
464 sys->videoFrame->linesize[2] = (int)pStride[2];
465 sys->videoFrame->pts++;
467#ifdef WITH_VAAPI_H264_ENCODING
470 av_frame_unref(sys->hwVideoFrame);
471 if ((status = av_hwframe_get_buffer(sys->codecEncoderContext->hw_frames_ctx,
472 sys->hwVideoFrame, 0)) < 0 ||
473 !sys->hwVideoFrame->hw_frames_ctx)
475 WLog_Print(h264->log, WLOG_ERROR,
"av_hwframe_get_buffer failed (%s [%d])",
476 av_err2str(status), status);
479 sys->videoFrame->format = AV_PIX_FMT_NV12;
480 if ((status = av_hwframe_transfer_data(sys->hwVideoFrame, sys->videoFrame, 0)) < 0)
482 WLog_Print(h264->log, WLOG_ERROR,
"av_hwframe_transfer_data failed (%s [%d])",
483 av_err2str(status), status);
490#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 48, 101)
491#ifdef WITH_VAAPI_H264_ENCODING
492 status = avcodec_send_frame(sys->codecEncoderContext,
493 sys->hwctx ? sys->hwVideoFrame : sys->videoFrame);
495 status = avcodec_send_frame(sys->codecEncoderContext, sys->videoFrame);
500 WLog_Print(h264->log, WLOG_ERROR,
"Failed to encode video frame (%s [%d])",
501 av_err2str(status), status);
505 status = avcodec_receive_packet(sys->codecEncoderContext, sys->packet);
509 WLog_Print(h264->log, WLOG_ERROR,
"Failed to encode video frame (%s [%d])",
510 av_err2str(status), status);
514 gotFrame = (status == 0);
515#elif LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(54, 59, 100)
519 status = avcodec_encode_video2(sys->codecEncoderContext, sys->packet, sys->videoFrame,
521 }
while ((status >= 0) && (gotFrame == 0));
525 avpicture_get_size(sys->codecDecoderContext->pix_fmt, sys->codecDecoderContext->width,
526 sys->codecDecoderContext->height);
527 sys->packet->data = av_malloc(sys->packet->size);
529 if (!sys->packet->data)
533 status = avcodec_encode_video(sys->codecDecoderContext, sys->packet->data,
534 sys->packet->size, sys->videoFrame);
541 WLog_Print(h264->log, WLOG_ERROR,
"Failed to encode video frame (%s [%d])",
542 av_err2str(status), status);
546 WINPR_ASSERT(sys->packet);
547 *ppDstData = sys->packet->data;
548 *pDstSize = (UINT32)MAX(0, sys->packet->size);
552 WLog_Print(h264->log, WLOG_ERROR,
"Did not get frame! (%s [%d])", av_err2str(status),
562static void libavcodec_uninit(H264_CONTEXT* h264)
566 H264_CONTEXT_LIBAVCODEC* sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
573#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 133, 100)
574 av_packet_unref(sys->packet);
576 av_packet_free(&sys->packet);
582#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(55, 18, 102)
583 av_frame_free(&sys->videoFrame);
585 av_free(sys->videoFrame);
589#if defined(WITH_VAAPI) || defined(WITH_VAAPI_H264_ENCODING) || defined(WITH_VIDEOTOOLBOX)
590 if (sys->hwVideoFrame)
592#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(55, 18, 102)
593 av_frame_free(&sys->hwVideoFrame);
595 av_free(sys->hwVideoFrame);
600 av_buffer_unref(&sys->hwctx);
602#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(57, 80, 100) || defined(WITH_VIDEOTOOLBOX)
604 if (sys->hw_frames_ctx)
605 av_buffer_unref(&sys->hw_frames_ctx);
611 if (sys->codecParser)
612 av_parser_close(sys->codecParser);
614 if (sys->codecDecoderContext)
616#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(55, 69, 100)
617 avcodec_free_context(&sys->codecDecoderContext);
619 avcodec_close(sys->codecDecoderContext);
620 av_free(sys->codecDecoderContext);
624 libavcodec_destroy_encoder_context(h264);
626 h264->pSystemData =
nullptr;
629#if defined(WITH_VAAPI) || defined(WITH_VIDEOTOOLBOX)
630static enum AVPixelFormat libavcodec_get_format(
struct AVCodecContext* ctx,
631 const enum AVPixelFormat* fmts)
635 H264_CONTEXT* h264 = (H264_CONTEXT*)ctx->opaque;
638 H264_CONTEXT_LIBAVCODEC* sys = (H264_CONTEXT_LIBAVCODEC*)h264->pSystemData;
641 for (
const enum AVPixelFormat* p = fmts; *p != AV_PIX_FMT_NONE; p++)
643 if (*p == sys->hw_pix_fmt)
645#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(57, 80, 100) || defined(WITH_VIDEOTOOLBOX)
646 if (sys->hw_frames_ctx)
647 av_buffer_unref(&sys->hw_frames_ctx);
649 sys->hw_frames_ctx = av_hwframe_ctx_alloc(sys->hwctx);
651 if (!sys->hw_frames_ctx)
653 return AV_PIX_FMT_NONE;
656 sys->codecDecoderContext->pix_fmt = *p;
657 AVHWFramesContext* frames = (AVHWFramesContext*)sys->hw_frames_ctx->data;
659 frames->height = sys->codecDecoderContext->coded_height;
660 frames->width = sys->codecDecoderContext->coded_width;
661#ifdef WITH_VIDEOTOOLBOX
662 frames->sw_format = AV_PIX_FMT_YUV420P;
665 (sys->codecDecoderContext->sw_pix_fmt == AV_PIX_FMT_YUV420P10 ? AV_PIX_FMT_P010
668 frames->initial_pool_size = 20;
670 if (sys->codecDecoderContext->active_thread_type & FF_THREAD_FRAME)
671 frames->initial_pool_size += sys->codecDecoderContext->thread_count;
673 int err = av_hwframe_ctx_init(sys->hw_frames_ctx);
677 WLog_Print(h264->log, WLOG_ERROR,
"Could not init hwframes context: %s",
679 return AV_PIX_FMT_NONE;
682 sys->codecDecoderContext->hw_frames_ctx = av_buffer_ref(sys->hw_frames_ctx);
688 return AV_PIX_FMT_NONE;
692static BOOL libavcodec_init(H264_CONTEXT* h264)
695 H264_CONTEXT_LIBAVCODEC* sys =
696 (H264_CONTEXT_LIBAVCODEC*)calloc(1,
sizeof(H264_CONTEXT_LIBAVCODEC));
703 h264->pSystemData = (
void*)sys;
705#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 10, 100)
706 avcodec_register_all();
709 if (!h264->Compressor)
711 sys->codecDecoder = avcodec_find_decoder(AV_CODEC_ID_H264);
713 if (!sys->codecDecoder)
715 WLog_Print(h264->log, WLOG_ERROR,
"Failed to find libav H.264 codec");
719 sys->codecDecoderContext = avcodec_alloc_context3(sys->codecDecoder);
721 if (!sys->codecDecoderContext)
723 WLog_Print(h264->log, WLOG_ERROR,
"Failed to allocate libav codec context");
727#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(59, 18, 100)
728 if (sys->codecDecoder->capabilities & AV_CODEC_CAP_TRUNCATED)
730 sys->codecDecoderContext->flags |= AV_CODEC_FLAG_TRUNCATED;
738 int ret = av_hwdevice_ctx_create(&sys->hwctx, AV_HWDEVICE_TYPE_VAAPI,
739 get_vaapi_device(),
nullptr, 0);
743 WLog_Print(h264->log, WLOG_ERROR,
744 "Could not initialize hardware decoder, falling back to software: %s",
746 sys->hwctx =
nullptr;
747 goto fail_hwdevice_create;
750 WLog_Print(h264->log, WLOG_INFO,
"Using VAAPI for accelerated H264 decoding");
752 sys->codecDecoderContext->get_format = libavcodec_get_format;
753 sys->hw_pix_fmt = AV_PIX_FMT_VAAPI;
754#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 80, 100)
755 sys->codecDecoderContext->hw_device_ctx = av_buffer_ref(sys->hwctx);
757 sys->codecDecoderContext->opaque = (
void*)h264;
758 fail_hwdevice_create:
761#ifdef WITH_VIDEOTOOLBOX
765 int ret = av_hwdevice_ctx_create(&sys->hwctx, AV_HWDEVICE_TYPE_VIDEOTOOLBOX,
nullptr,
771 h264->log, WLOG_ERROR,
772 "Could not initialize VideoToolbox decoder, falling back to software: %s",
774 sys->hwctx =
nullptr;
778 WLog_Print(h264->log, WLOG_INFO,
"Using VideoToolbox for accelerated H264 decoding");
780 sys->codecDecoderContext->get_format = libavcodec_get_format;
781 sys->hw_pix_fmt = AV_PIX_FMT_VIDEOTOOLBOX;
782 sys->codecDecoderContext->hw_device_ctx = av_buffer_ref(sys->hwctx);
783 sys->codecDecoderContext->opaque = (
void*)h264;
787 if (avcodec_open2(sys->codecDecoderContext, sys->codecDecoder,
nullptr) < 0)
789 WLog_Print(h264->log, WLOG_ERROR,
"Failed to open libav codec");
793 sys->codecParser = av_parser_init(AV_CODEC_ID_H264);
795 if (!sys->codecParser)
797 WLog_Print(h264->log, WLOG_ERROR,
"Failed to initialize libav parser");
803#ifdef WITH_VAAPI_H264_ENCODING
806 sys->codecEncoder = avcodec_find_encoder_by_name(
"h264_vaapi");
807 if (!sys->codecEncoder)
809 WLog_Print(h264->log, WLOG_ERROR,
"H264 VAAPI encoder not found");
811 else if (av_hwdevice_ctx_create(&sys->hwctx, AV_HWDEVICE_TYPE_VAAPI, get_vaapi_device(),
814 WLog_Print(h264->log, WLOG_ERROR,
"av_hwdevice_ctx_create failed");
815 sys->codecEncoder =
nullptr;
816 sys->hwctx =
nullptr;
820 WLog_Print(h264->log, WLOG_INFO,
"Using VAAPI for accelerated H264 encoding");
824 if (!sys->codecEncoder)
826 sys->codecEncoder = avcodec_find_encoder(AV_CODEC_ID_H264);
827 h264->hwAccel = FALSE;
830 if (!sys->codecEncoder)
832 WLog_Print(h264->log, WLOG_ERROR,
"Failed to initialize H264 encoder");
837#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(55, 18, 102)
838 sys->videoFrame = av_frame_alloc();
839#if defined(WITH_VAAPI) || defined(WITH_VAAPI_H264_ENCODING) || defined(WITH_VIDEOTOOLBOX)
840 sys->hwVideoFrame = av_frame_alloc();
843 sys->videoFrame = avcodec_alloc_frame();
846 if (!sys->videoFrame)
848 WLog_Print(h264->log, WLOG_ERROR,
"Failed to allocate libav frame");
852#if defined(WITH_VAAPI) || defined(WITH_VAAPI_H264_ENCODING) || defined(WITH_VIDEOTOOLBOX)
853 if (!sys->hwVideoFrame)
855 WLog_Print(h264->log, WLOG_ERROR,
"Failed to allocate libav hw frame");
860 sys->videoFrame->pts = 0;
863 libavcodec_uninit(h264);
867const H264_CONTEXT_SUBSYSTEM g_Subsystem_libavcodec = {
"libavcodec", libavcodec_init,
868 libavcodec_uninit, libavcodec_decompress,
869 libavcodec_compress };