Blame libfreerdp/codec/h264_openh264.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * H.264 Bitmap Compression
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Mike McDonald <Mike.McDonald@software.dell.com>
Packit 1fb8d4
 * Copyright 2015 Vic Lee <llyzs.vic@gmail.com>
Packit 1fb8d4
 * Copyright 2014 Armin Novak <armin.novak@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit Service 5a9772
#ifdef HAVE_CONFIG_H
Packit Service 5a9772
#include "config.h"
Packit Service 5a9772
#endif
Packit Service 5a9772
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
#include <freerdp/codec/h264.h>
Packit Service 5a9772
#include <winpr/library.h>
Packit 1fb8d4
Packit 1fb8d4
#include "wels/codec_def.h"
Packit 1fb8d4
#include "wels/codec_api.h"
Packit 1fb8d4
#include "wels/codec_ver.h"
Packit 1fb8d4
Packit Service 5a9772
typedef void (*pWelsGetCodecVersionEx)(OpenH264Version* pVersion);
Packit Service 5a9772
Packit Service 5a9772
typedef long (*pWelsCreateDecoder)(ISVCDecoder** ppDecoder);
Packit Service 5a9772
typedef void (*pWelsDestroyDecoder)(ISVCDecoder* pDecoder);
Packit Service 5a9772
Packit Service 5a9772
typedef int (*pWelsCreateSVCEncoder)(ISVCEncoder** ppEncoder);
Packit Service 5a9772
typedef void (*pWelsDestroySVCEncoder)(ISVCEncoder* pEncoder);
Packit 1fb8d4
Packit 1fb8d4
struct _H264_CONTEXT_OPENH264
Packit 1fb8d4
{
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
	HMODULE lib;
Packit Service 5a9772
	OpenH264Version version;
Packit Service 5a9772
#endif
Packit Service 5a9772
	pWelsGetCodecVersionEx WelsGetCodecVersionEx;
Packit Service 5a9772
	pWelsCreateDecoder WelsCreateDecoder;
Packit Service 5a9772
	pWelsDestroyDecoder WelsDestroyDecoder;
Packit Service 5a9772
	pWelsCreateSVCEncoder WelsCreateSVCEncoder;
Packit Service 5a9772
	pWelsDestroySVCEncoder WelsDestroySVCEncoder;
Packit 1fb8d4
	ISVCDecoder* pDecoder;
Packit 1fb8d4
	ISVCEncoder* pEncoder;
Packit 1fb8d4
	SEncParamExt EncParamExt;
Packit 1fb8d4
};
Packit 1fb8d4
typedef struct _H264_CONTEXT_OPENH264 H264_CONTEXT_OPENH264;
Packit 1fb8d4
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
static const char* openh264_library_names[] = {
Packit Service 5a9772
#if defined(_WIN32)
Packit Service 5a9772
	"openh264.dll"
Packit Service 5a9772
#elif defined(__APPLE__)
Packit Service 5a9772
	"libopenh264.dylib"
Packit Service 5a9772
#else
Packit Service 5a9772
	"libopenh264.so"
Packit Service 5a9772
#endif
Packit Service 5a9772
};
Packit Service 5a9772
#endif
Packit Service 5a9772
Packit Service 5a9772
static void openh264_trace_callback(H264_CONTEXT* h264, int level, const char* message)
Packit 1fb8d4
{
Packit 1fb8d4
	if (h264)
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_TRACE, "%d - %s", level, message);
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static int openh264_decompress(H264_CONTEXT* h264, const BYTE* pSrcData, UINT32 SrcSize)
Packit 1fb8d4
{
Packit 1fb8d4
	DECODING_STATE state;
Packit 1fb8d4
	SBufferInfo sBufferInfo;
Packit 1fb8d4
	SSysMEMBuffer* pSystemBuffer;
Packit Service 5a9772
	H264_CONTEXT_OPENH264* sys = (H264_CONTEXT_OPENH264*)h264->pSystemData;
Packit 1fb8d4
	UINT32* iStride = h264->iStride;
Packit 1fb8d4
	BYTE** pYUVData = h264->pYUVData;
Packit 1fb8d4
Packit 1fb8d4
	if (!sys->pDecoder)
Packit 1fb8d4
		return -2001;
Packit 1fb8d4
Packit 1fb8d4
	/*
Packit 1fb8d4
	 * Decompress the image.  The RDP host only seems to send I420 format.
Packit 1fb8d4
	 */
Packit 1fb8d4
	pYUVData[0] = NULL;
Packit 1fb8d4
	pYUVData[1] = NULL;
Packit 1fb8d4
	pYUVData[2] = NULL;
Packit 1fb8d4
	ZeroMemory(&sBufferInfo, sizeof(sBufferInfo));
Packit Service 5a9772
	state =
Packit Service 5a9772
	    (*sys->pDecoder)->DecodeFrame2(sys->pDecoder, pSrcData, SrcSize, pYUVData, &sBufferInfo);
Packit 1fb8d4
Packit 1fb8d4
	if (sBufferInfo.iBufferStatus != 1)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (state == dsNoParamSets)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* this happens on the first frame due to missing parameter sets */
Packit Service 5a9772
			state = (*sys->pDecoder)->DecodeFrame2(sys->pDecoder, NULL, 0, pYUVData, &sBufferInfo);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (state == dsErrorFree)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* call DecodeFrame2 again to decode without delay */
Packit Service 5a9772
			state = (*sys->pDecoder)->DecodeFrame2(sys->pDecoder, NULL, 0, pYUVData, &sBufferInfo);
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_WARN, "DecodeFrame2 state: 0x%04X iBufferStatus: %d", state,
Packit 1fb8d4
			           sBufferInfo.iBufferStatus);
Packit 1fb8d4
			return -2002;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pSystemBuffer = &sBufferInfo.UsrData.sSystemBuffer;
Packit 1fb8d4
	iStride[0] = pSystemBuffer->iStride[0];
Packit 1fb8d4
	iStride[1] = pSystemBuffer->iStride[1];
Packit 1fb8d4
	iStride[2] = pSystemBuffer->iStride[1];
Packit 1fb8d4
Packit 1fb8d4
	if (sBufferInfo.iBufferStatus != 1)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_Print(h264->log, WLOG_WARN, "DecodeFrame2 iBufferStatus: %d",
Packit Service 5a9772
		           sBufferInfo.iBufferStatus);
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (state != dsErrorFree)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_WARN, "DecodeFrame2 state: 0x%02X", state);
Packit 1fb8d4
		return -2003;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
#if 0
Packit 1fb8d4
	WLog_Print(h264->log, WLOG_INFO,
Packit 1fb8d4
	           "h264_decompress: state=%u, pYUVData=[%p,%p,%p], bufferStatus=%d, width=%d, height=%d, format=%d, stride=[%d,%d]",
Packit 1fb8d4
	           state, (void*) pYUVData[0], (void*) pYUVData[1], (void*) pYUVData[2], sBufferInfo.iBufferStatus,
Packit 1fb8d4
	           pSystemBuffer->iWidth, pSystemBuffer->iHeight, pSystemBuffer->iFormat,
Packit 1fb8d4
	           pSystemBuffer->iStride[0], pSystemBuffer->iStride[1]);
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
	if (pSystemBuffer->iFormat != videoFormatI420)
Packit 1fb8d4
		return -2004;
Packit 1fb8d4
Packit 1fb8d4
	if (!pYUVData[0] || !pYUVData[1] || !pYUVData[2])
Packit 1fb8d4
		return -2005;
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int openh264_compress(H264_CONTEXT* h264, const BYTE** pYUVData, const UINT32* iStride,
Packit 1fb8d4
                             BYTE** ppDstData, UINT32* pDstSize)
Packit 1fb8d4
{
Packit 1fb8d4
	int i, j;
Packit 1fb8d4
	int status;
Packit 1fb8d4
	SFrameBSInfo info;
Packit 1fb8d4
	SSourcePicture pic;
Packit 1fb8d4
	SBitrateInfo bitrate;
Packit 1fb8d4
	H264_CONTEXT_OPENH264* sys;
Packit Service 5a9772
	sys = &((H264_CONTEXT_OPENH264*)h264->pSystemData)[0];
Packit 1fb8d4
Packit 1fb8d4
	if (!sys->pEncoder)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	if (!pYUVData[0] || !pYUVData[1] || !pYUVData[2])
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit Service 5a9772
	if ((h264->width > INT_MAX) || (h264->height > INT_MAX))
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if ((h264->FrameRate > INT_MAX) || (h264->NumberOfThreads > INT_MAX) ||
Packit Service 5a9772
	    (h264->BitRate > INT_MAX) || (h264->QP > INT_MAX))
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	if ((sys->EncParamExt.iPicWidth != (int)h264->width) ||
Packit Service 5a9772
	    (sys->EncParamExt.iPicHeight != (int)h264->height))
Packit 1fb8d4
	{
Packit 1fb8d4
		status = (*sys->pEncoder)->GetDefaultParams(sys->pEncoder, &sys->EncParamExt);
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
			           "Failed to get OpenH264 default parameters (status=%d)", status);
Packit 1fb8d4
			return status;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		sys->EncParamExt.iUsageType = SCREEN_CONTENT_REAL_TIME;
Packit Service 5a9772
		sys->EncParamExt.iPicWidth = (int)h264->width;
Packit Service 5a9772
		sys->EncParamExt.iPicHeight = (int)h264->height;
Packit Service 5a9772
		sys->EncParamExt.fMaxFrameRate = (int)h264->FrameRate;
Packit 1fb8d4
		sys->EncParamExt.iMaxBitrate = UNSPECIFIED_BIT_RATE;
Packit 1fb8d4
		sys->EncParamExt.bEnableDenoise = 0;
Packit 1fb8d4
		sys->EncParamExt.bEnableLongTermReference = 0;
Packit 1fb8d4
		sys->EncParamExt.bEnableFrameSkip = 0;
Packit 1fb8d4
		sys->EncParamExt.iSpatialLayerNum = 1;
Packit Service 5a9772
		sys->EncParamExt.iMultipleThreadIdc = (int)h264->NumberOfThreads;
Packit 1fb8d4
		sys->EncParamExt.sSpatialLayers[0].fFrameRate = h264->FrameRate;
Packit 1fb8d4
		sys->EncParamExt.sSpatialLayers[0].iVideoWidth = sys->EncParamExt.iPicWidth;
Packit 1fb8d4
		sys->EncParamExt.sSpatialLayers[0].iVideoHeight = sys->EncParamExt.iPicHeight;
Packit Service 5a9772
		sys->EncParamExt.sSpatialLayers[0].iMaxSpatialBitrate = sys->EncParamExt.iMaxBitrate;
Packit 1fb8d4
Packit 1fb8d4
		switch (h264->RateControlMode)
Packit 1fb8d4
		{
Packit 1fb8d4
			case H264_RATECONTROL_VBR:
Packit 1fb8d4
				sys->EncParamExt.iRCMode = RC_BITRATE_MODE;
Packit Service 5a9772
				sys->EncParamExt.iTargetBitrate = (int)h264->BitRate;
Packit 1fb8d4
				sys->EncParamExt.sSpatialLayers[0].iSpatialBitrate =
Packit 1fb8d4
				    sys->EncParamExt.iTargetBitrate;
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case H264_RATECONTROL_CQP:
Packit 1fb8d4
				sys->EncParamExt.iRCMode = RC_OFF_MODE;
Packit Service 5a9772
				sys->EncParamExt.sSpatialLayers[0].iDLayerQp = (int)h264->QP;
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->EncParamExt.iMultipleThreadIdc > 1)
Packit 1fb8d4
		{
Packit 1fb8d4
#if (OPENH264_MAJOR == 1) && (OPENH264_MINOR <= 5)
Packit 1fb8d4
			sys->EncParamExt.sSpatialLayers[0].sSliceCfg.uiSliceMode = SM_AUTO_SLICE;
Packit 1fb8d4
#else
Packit 1fb8d4
			sys->EncParamExt.sSpatialLayers[0].sSliceArgument.uiSliceMode = SM_FIXEDSLCNUM_SLICE;
Packit 1fb8d4
#endif
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		status = (*sys->pEncoder)->InitializeExt(sys->pEncoder, &sys->EncParamExt);
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_Print(h264->log, WLOG_ERROR, "Failed to initialize OpenH264 encoder (status=%d)",
Packit Service 5a9772
			           status);
Packit 1fb8d4
			return status;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit Service 5a9772
		status =
Packit Service 5a9772
		    (*sys->pEncoder)
Packit Service 5a9772
		        ->GetOption(sys->pEncoder, ENCODER_OPTION_SVC_ENCODE_PARAM_EXT, &sys->EncParamExt);
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
		{
Packit Service 5a9772
			WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
			           "Failed to get initial OpenH264 encoder parameters (status=%d)", status);
Packit 1fb8d4
			return status;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		switch (h264->RateControlMode)
Packit 1fb8d4
		{
Packit 1fb8d4
			case H264_RATECONTROL_VBR:
Packit Service 5a9772
				if (sys->EncParamExt.iTargetBitrate != (int)h264->BitRate)
Packit 1fb8d4
				{
Packit Service 5a9772
					sys->EncParamExt.iTargetBitrate = (int)h264->BitRate;
Packit 1fb8d4
					bitrate.iLayer = SPATIAL_LAYER_ALL;
Packit Service 5a9772
					bitrate.iBitrate = (int)h264->BitRate;
Packit Service 5a9772
					status = (*sys->pEncoder)
Packit Service 5a9772
					             ->SetOption(sys->pEncoder, ENCODER_OPTION_BITRATE, &bitrate);
Packit 1fb8d4
Packit 1fb8d4
					if (status < 0)
Packit 1fb8d4
					{
Packit Service 5a9772
						WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
						           "Failed to set encoder bitrate (status=%d)", status);
Packit 1fb8d4
						return status;
Packit 1fb8d4
					}
Packit 1fb8d4
				}
Packit 1fb8d4
Packit Service 5a9772
				if (sys->EncParamExt.fMaxFrameRate != (int)h264->FrameRate)
Packit 1fb8d4
				{
Packit Service 5a9772
					sys->EncParamExt.fMaxFrameRate = (int)h264->FrameRate;
Packit Service 5a9772
					status = (*sys->pEncoder)
Packit Service 5a9772
					             ->SetOption(sys->pEncoder, ENCODER_OPTION_FRAME_RATE,
Packit Service 5a9772
					                         &sys->EncParamExt.fMaxFrameRate);
Packit 1fb8d4
Packit 1fb8d4
					if (status < 0)
Packit 1fb8d4
					{
Packit Service 5a9772
						WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
						           "Failed to set encoder framerate (status=%d)", status);
Packit 1fb8d4
						return status;
Packit 1fb8d4
					}
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case H264_RATECONTROL_CQP:
Packit Service 5a9772
				if (sys->EncParamExt.sSpatialLayers[0].iDLayerQp != (int)h264->QP)
Packit 1fb8d4
				{
Packit Service 5a9772
					sys->EncParamExt.sSpatialLayers[0].iDLayerQp = (int)h264->QP;
Packit Service 5a9772
					status = (*sys->pEncoder)
Packit Service 5a9772
					             ->SetOption(sys->pEncoder, ENCODER_OPTION_SVC_ENCODE_PARAM_EXT,
Packit Service 5a9772
					                         &sys->EncParamExt);
Packit 1fb8d4
Packit 1fb8d4
					if (status < 0)
Packit 1fb8d4
					{
Packit Service 5a9772
						WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
						           "Failed to set encoder parameters (status=%d)", status);
Packit 1fb8d4
						return status;
Packit 1fb8d4
					}
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	memset(&info, 0, sizeof(SFrameBSInfo));
Packit 1fb8d4
	memset(&pic, 0, sizeof(SSourcePicture));
Packit 1fb8d4
	pic.iPicWidth = (int)h264->width;
Packit 1fb8d4
	pic.iPicHeight = (int)h264->height;
Packit 1fb8d4
	pic.iColorFormat = videoFormatI420;
Packit 1fb8d4
	pic.iStride[0] = (int)iStride[0];
Packit 1fb8d4
	pic.iStride[1] = (int)iStride[1];
Packit 1fb8d4
	pic.iStride[2] = (int)iStride[2];
Packit 1fb8d4
	pic.pData[0] = (unsigned char*)pYUVData[0];
Packit 1fb8d4
	pic.pData[1] = (unsigned char*)pYUVData[1];
Packit 1fb8d4
	pic.pData[2] = (unsigned char*)pYUVData[2];
Packit 1fb8d4
	status = (*sys->pEncoder)->EncodeFrame(sys->pEncoder, &pic, &info;;
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "Failed to encode frame (status=%d)", status);
Packit 1fb8d4
		return status;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	*ppDstData = info.sLayerInfo[0].pBsBuf;
Packit 1fb8d4
	*pDstSize = 0;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < info.iLayerNum; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		for (j = 0; j < info.sLayerInfo[i].iNalCount; j++)
Packit 1fb8d4
		{
Packit 1fb8d4
			*pDstSize += info.sLayerInfo[i].pNalLengthInByte[j];
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void openh264_uninit(H264_CONTEXT* h264)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit Service 5a9772
	H264_CONTEXT_OPENH264* sysContexts = (H264_CONTEXT_OPENH264*)h264->pSystemData;
Packit 1fb8d4
Packit 1fb8d4
	if (sysContexts)
Packit 1fb8d4
	{
Packit 1fb8d4
		for (x = 0; x < h264->numSystemData; x++)
Packit 1fb8d4
		{
Packit 1fb8d4
			H264_CONTEXT_OPENH264* sys = &sysContexts[x];
Packit 1fb8d4
Packit 1fb8d4
			if (sys->pDecoder)
Packit 1fb8d4
			{
Packit 1fb8d4
				(*sys->pDecoder)->Uninitialize(sys->pDecoder);
Packit Service 5a9772
				sysContexts->WelsDestroyDecoder(sys->pDecoder);
Packit 1fb8d4
				sys->pDecoder = NULL;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (sys->pEncoder)
Packit 1fb8d4
			{
Packit 1fb8d4
				(*sys->pEncoder)->Uninitialize(sys->pEncoder);
Packit Service 5a9772
				sysContexts->WelsDestroySVCEncoder(sys->pEncoder);
Packit 1fb8d4
				sys->pEncoder = NULL;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
		if (sysContexts->lib)
Packit Service 5a9772
			FreeLibrary(sysContexts->lib);
Packit Service 5a9772
#endif
Packit 1fb8d4
		free(h264->pSystemData);
Packit 1fb8d4
		h264->pSystemData = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
static BOOL openh264_load_functionpointers(H264_CONTEXT* h264, const char* name)
Packit Service 5a9772
{
Packit Service 5a9772
	H264_CONTEXT_OPENH264* sysContexts;
Packit Service 5a9772
Packit Service 5a9772
	if (!h264)
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
Packit Service 5a9772
	sysContexts = h264->pSystemData;
Packit Service 5a9772
Packit Service 5a9772
	if (!sysContexts)
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
Packit Service 5a9772
	sysContexts->lib = LoadLibraryA(name);
Packit Service 5a9772
Packit Service 5a9772
	if (!sysContexts->lib)
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
Packit Service 5a9772
	sysContexts->WelsGetCodecVersionEx =
Packit Service 5a9772
	    (pWelsGetCodecVersionEx)GetProcAddress(sysContexts->lib, "WelsGetCodecVersionEx");
Packit Service 5a9772
	sysContexts->WelsCreateDecoder =
Packit Service 5a9772
	    (pWelsCreateDecoder)GetProcAddress(sysContexts->lib, "WelsCreateDecoder");
Packit Service 5a9772
	sysContexts->WelsDestroyDecoder =
Packit Service 5a9772
	    (pWelsDestroyDecoder)GetProcAddress(sysContexts->lib, "WelsDestroyDecoder");
Packit Service 5a9772
	sysContexts->WelsCreateSVCEncoder =
Packit Service 5a9772
	    (pWelsCreateSVCEncoder)GetProcAddress(sysContexts->lib, "WelsCreateSVCEncoder");
Packit Service 5a9772
	sysContexts->WelsDestroySVCEncoder =
Packit Service 5a9772
	    (pWelsDestroySVCEncoder)GetProcAddress(sysContexts->lib, "WelsDestroySVCEncoder");
Packit Service 5a9772
Packit Service 5a9772
	if (!sysContexts->WelsCreateDecoder || !sysContexts->WelsDestroyDecoder ||
Packit Service 5a9772
	    !sysContexts->WelsCreateSVCEncoder || !sysContexts->WelsDestroySVCEncoder ||
Packit Service 5a9772
	    !sysContexts->WelsGetCodecVersionEx)
Packit Service 5a9772
	{
Packit Service 5a9772
		FreeLibrary(sysContexts->lib);
Packit Service 5a9772
		sysContexts->lib = NULL;
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	sysContexts->WelsGetCodecVersionEx(&sysContexts->version);
Packit Service 5a9772
	WLog_Print(h264->log, WLOG_INFO, "loaded %s %d.%d.%d", name, sysContexts->version.uMajor,
Packit Service 5a9772
	           sysContexts->version.uMinor, sysContexts->version.uRevision);
Packit Service 5a9772
Packit Service 5a9772
	if ((sysContexts->version.uMajor < 1) ||
Packit Service 5a9772
	    ((sysContexts->version.uMajor == 1) && (sysContexts->version.uMinor < 6)))
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_Print(
Packit Service 5a9772
		    h264->log, WLOG_ERROR,
Packit Service 5a9772
		    "OpenH264 %s %d.%d.%d is too old, need at least version 1.6.0 for dynamic loading",
Packit Service 5a9772
		    name, sysContexts->version.uMajor, sysContexts->version.uMinor,
Packit Service 5a9772
		    sysContexts->version.uRevision);
Packit Service 5a9772
		FreeLibrary(sysContexts->lib);
Packit Service 5a9772
		sysContexts->lib = NULL;
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	return TRUE;
Packit Service 5a9772
}
Packit Service 5a9772
#endif
Packit Service 5a9772
Packit 1fb8d4
static BOOL openh264_init(H264_CONTEXT* h264)
Packit 1fb8d4
{
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
	BOOL success = FALSE;
Packit Service 5a9772
	size_t i;
Packit Service 5a9772
#endif
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
	long status;
Packit 1fb8d4
	SDecodingParam sDecParam;
Packit 1fb8d4
	H264_CONTEXT_OPENH264* sysContexts;
Packit 1fb8d4
	static int traceLevel = WELS_LOG_DEBUG;
Packit 1fb8d4
#if (OPENH264_MAJOR == 1) && (OPENH264_MINOR <= 5)
Packit 1fb8d4
	static EVideoFormatType videoFormat = videoFormatI420;
Packit 1fb8d4
#endif
Packit Service 5a9772
	static WelsTraceCallback traceCallback = (WelsTraceCallback)openh264_trace_callback;
Packit 1fb8d4
	h264->numSystemData = 1;
Packit Service 5a9772
	sysContexts =
Packit Service 5a9772
	    (H264_CONTEXT_OPENH264*)calloc(h264->numSystemData, sizeof(H264_CONTEXT_OPENH264));
Packit 1fb8d4
Packit 1fb8d4
	if (!sysContexts)
Packit 1fb8d4
		goto EXCEPTION;
Packit 1fb8d4
Packit Service 5a9772
	h264->pSystemData = (void*)sysContexts;
Packit Service 5a9772
#if defined(WITH_OPENH264_LOADING)
Packit Service 5a9772
Packit Service 5a9772
	for (i = 0; i < ARRAYSIZE(openh264_library_names); i++)
Packit Service 5a9772
	{
Packit Service 5a9772
		const char* current = openh264_library_names[i];
Packit Service 5a9772
		success = openh264_load_functionpointers(h264, current);
Packit Service 5a9772
Packit Service 5a9772
		if (success)
Packit Service 5a9772
			break;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (!success)
Packit Service 5a9772
		goto EXCEPTION;
Packit Service 5a9772
Packit Service 5a9772
#else
Packit Service 5a9772
	sysContexts->WelsGetCodecVersionEx = WelsGetCodecVersionEx;
Packit Service 5a9772
	sysContexts->WelsCreateDecoder = WelsCreateDecoder;
Packit Service 5a9772
	sysContexts->WelsDestroyDecoder = WelsDestroyDecoder;
Packit Service 5a9772
	sysContexts->WelsCreateSVCEncoder = WelsCreateSVCEncoder;
Packit Service 5a9772
	sysContexts->WelsDestroySVCEncoder = WelsDestroySVCEncoder;
Packit Service 5a9772
#endif
Packit 1fb8d4
Packit 1fb8d4
	for (x = 0; x < h264->numSystemData; x++)
Packit 1fb8d4
	{
Packit 1fb8d4
		H264_CONTEXT_OPENH264* sys = &sysContexts[x];
Packit 1fb8d4
Packit 1fb8d4
		if (h264->Compressor)
Packit 1fb8d4
		{
Packit Service 5a9772
			sysContexts->WelsCreateSVCEncoder(&sys->pEncoder);
Packit 1fb8d4
Packit 1fb8d4
			if (!sys->pEncoder)
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_Print(h264->log, WLOG_ERROR, "Failed to create OpenH264 encoder");
Packit 1fb8d4
				goto EXCEPTION;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit Service 5a9772
			sysContexts->WelsCreateDecoder(&sys->pDecoder);
Packit 1fb8d4
Packit 1fb8d4
			if (!sys->pDecoder)
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_Print(h264->log, WLOG_ERROR, "Failed to create OpenH264 decoder");
Packit 1fb8d4
				goto EXCEPTION;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			ZeroMemory(&sDecParam, sizeof(sDecParam));
Packit 1fb8d4
#if (OPENH264_MAJOR == 1) && (OPENH264_MINOR <= 5)
Packit Service 5a9772
			sDecParam.eOutputColorFormat = videoFormatI420;
Packit 1fb8d4
#endif
Packit 1fb8d4
			sDecParam.eEcActiveIdc = ERROR_CON_FRAME_COPY;
Packit 1fb8d4
			sDecParam.sVideoProperty.eVideoBsType = VIDEO_BITSTREAM_AVC;
Packit 1fb8d4
			status = (*sys->pDecoder)->Initialize(sys->pDecoder, &sDecParam);
Packit 1fb8d4
Packit 1fb8d4
			if (status != 0)
Packit 1fb8d4
			{
Packit Service 5a9772
				WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
				           "Failed to initialize OpenH264 decoder (status=%ld)", status);
Packit 1fb8d4
				goto EXCEPTION;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
#if (OPENH264_MAJOR == 1) && (OPENH264_MINOR <= 5)
Packit Service 5a9772
			status =
Packit Service 5a9772
			    (*sys->pDecoder)->SetOption(sys->pDecoder, DECODER_OPTION_DATAFORMAT, &videoFormat);
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
			if (status != 0)
Packit 1fb8d4
			{
Packit 1fb8d4
				WLog_Print(h264->log, WLOG_ERROR,
Packit 1fb8d4
				           "Failed to set data format option on OpenH264 decoder (status=%ld)",
Packit 1fb8d4
				           status);
Packit 1fb8d4
				goto EXCEPTION;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (WLog_GetLogLevel(h264->log) == WLOG_TRACE)
Packit 1fb8d4
			{
Packit Service 5a9772
				status = (*sys->pDecoder)
Packit Service 5a9772
				             ->SetOption(sys->pDecoder, DECODER_OPTION_TRACE_LEVEL, &traceLevel);
Packit 1fb8d4
Packit 1fb8d4
				if (status != 0)
Packit 1fb8d4
				{
Packit 1fb8d4
					WLog_Print(h264->log, WLOG_ERROR,
Packit 1fb8d4
					           "Failed to set trace level option on OpenH264 decoder (status=%ld)",
Packit 1fb8d4
					           status);
Packit 1fb8d4
					goto EXCEPTION;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit Service 5a9772
				status =
Packit Service 5a9772
				    (*sys->pDecoder)
Packit Service 5a9772
				        ->SetOption(sys->pDecoder, DECODER_OPTION_TRACE_CALLBACK_CONTEXT, &h264);
Packit 1fb8d4
Packit 1fb8d4
				if (status != 0)
Packit 1fb8d4
				{
Packit 1fb8d4
					WLog_Print(h264->log, WLOG_ERROR,
Packit Service 5a9772
					           "Failed to set trace callback context option on OpenH264 decoder "
Packit Service 5a9772
					           "(status=%ld)",
Packit 1fb8d4
					           status);
Packit 1fb8d4
					goto EXCEPTION;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit Service 5a9772
				status =
Packit Service 5a9772
				    (*sys->pDecoder)
Packit Service 5a9772
				        ->SetOption(sys->pDecoder, DECODER_OPTION_TRACE_CALLBACK, &traceCallback);
Packit 1fb8d4
Packit 1fb8d4
				if (status != 0)
Packit 1fb8d4
				{
Packit Service 5a9772
					WLog_Print(
Packit Service 5a9772
					    h264->log, WLOG_ERROR,
Packit Service 5a9772
					    "Failed to set trace callback option on OpenH264 decoder (status=%ld)",
Packit Service 5a9772
					    status);
Packit 1fb8d4
					goto EXCEPTION;
Packit 1fb8d4
				}
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
EXCEPTION:
Packit 1fb8d4
	openh264_uninit(h264);
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
H264_CONTEXT_SUBSYSTEM g_Subsystem_OpenH264 = { "OpenH264", openh264_init, openh264_uninit,
Packit Service 5a9772
	                                            openh264_decompress, openh264_compress };