Blame libfreerdp/codec/h264_mf.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
 *
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 1fb8d4
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
#include <freerdp/codec/h264.h>
Packit 1fb8d4
Packit 1fb8d4
#include <ks.h>
Packit 1fb8d4
#include <codecapi.h>
Packit 1fb8d4
Packit 1fb8d4
#include <mfapi.h>
Packit 1fb8d4
#include <mferror.h>
Packit 1fb8d4
#include <wmcodecdsp.h>
Packit 1fb8d4
#include <mftransform.h>
Packit 1fb8d4
Packit 1fb8d4
#include "h264.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("codec")
Packit 1fb8d4
Packit 1fb8d4
static const GUID sCLSID_CMSH264DecoderMFT = { 0x62CE7E72, 0x4C71, 0x4d20, { 0xB1, 0x5D,
Packit 1fb8d4
            0x45, 0x28, 0x31, 0xA8, 0x7D, 0x9D } };
Packit 1fb8d4
static const GUID sCLSID_VideoProcessorMFT = { 0x88753b26, 0x5b24, 0x49bd, { 0xb2, 0xe7,
Packit 1fb8d4
            0x0c, 0x44, 0x5c, 0x78, 0xc9, 0x82 } };
Packit 1fb8d4
static const GUID sIID_IMFTransform = { 0xbf94c121, 0x5b05, 0x4e6f, { 0x80, 0x00, 0xba,
Packit 1fb8d4
            0x59, 0x89, 0x61, 0x41, 0x4d } };
Packit 1fb8d4
static const GUID sMF_MT_MAJOR_TYPE = { 0x48eba18e, 0xf8c9, 0x4687, { 0xbf, 0x11, 0x0a,
Packit 1fb8d4
            0x74, 0xc9, 0xf9, 0x6a, 0x8f } };
Packit 1fb8d4
static const GUID sMF_MT_FRAME_SIZE = { 0x1652c33d, 0xd6b2, 0x4012, { 0xb8, 0x34, 0x72,
Packit 1fb8d4
            0x03, 0x08, 0x49, 0xa3, 0x7d } };
Packit 1fb8d4
static const GUID sMF_MT_DEFAULT_STRIDE = { 0x644b4e48, 0x1e02, 0x4516, { 0xb0, 0xeb, 0xc0,
Packit 1fb8d4
            0x1c, 0xa9, 0xd4, 0x9a, 0xc6 } };
Packit 1fb8d4
static const GUID sMF_MT_SUBTYPE = { 0xf7e34c9a, 0x42e8, 0x4714, { 0xb7, 0x4b, 0xcb, 0x29,
Packit 1fb8d4
            0xd7, 0x2c, 0x35, 0xe5 } };
Packit 1fb8d4
static const GUID sMF_XVP_DISABLE_FRC = { 0x2c0afa19, 0x7a97, 0x4d5a, { 0x9e, 0xe8, 0x16,
Packit 1fb8d4
            0xd4, 0xfc, 0x51, 0x8d, 0x8c } };
Packit 1fb8d4
static const GUID sMFMediaType_Video = { 0x73646976, 0x0000, 0x0010, { 0x80, 0x00, 0x00,
Packit 1fb8d4
            0xAA, 0x00, 0x38, 0x9B, 0x71 } };
Packit 1fb8d4
static const GUID sMFVideoFormat_RGB32 = { 22, 0x0000, 0x0010, { 0x80, 0x00, 0x00, 0xAA,
Packit 1fb8d4
            0x00, 0x38, 0x9B, 0x71 } };
Packit 1fb8d4
static const GUID sMFVideoFormat_ARGB32 = { 21, 0x0000, 0x0010, { 0x80, 0x00, 0x00, 0xAA,
Packit 1fb8d4
            0x00, 0x38, 0x9B, 0x71 } };
Packit 1fb8d4
static const GUID sMFVideoFormat_H264 = { 0x34363248, 0x0000, 0x0010, { 0x80, 0x00, 0x00,
Packit 1fb8d4
            0xaa, 0x00, 0x38, 0x9b, 0x71 } };
Packit 1fb8d4
static const GUID sMFVideoFormat_IYUV = { 0x56555949, 0x0000, 0x0010, { 0x80, 0x00, 0x00,
Packit 1fb8d4
            0xaa, 0x00, 0x38, 0x9b, 0x71 } };
Packit 1fb8d4
static const GUID sIID_ICodecAPI = { 0x901db4c7, 0x31ce, 0x41a2, { 0x85, 0xdc, 0x8f, 0xa0,
Packit 1fb8d4
            0xbf, 0x41, 0xb8, 0xda } };
Packit 1fb8d4
static const GUID sCODECAPI_AVLowLatencyMode = { 0x9c27891a, 0xed7a, 0x40e1, { 0x88, 0xe8,
Packit 1fb8d4
            0xb2, 0x27, 0x27, 0xa0, 0x24, 0xee } };
Packit 1fb8d4
static const GUID sCODECAPI_AVDecVideoMaxCodedWidth = { 0x5ae557b8, 0x77af, 0x41f5, { 0x9f,
Packit 1fb8d4
            0xa6, 0x4d, 0xb2, 0xfe, 0x1d, 0x4b, 0xca } };
Packit 1fb8d4
static const GUID sCODECAPI_AVDecVideoMaxCodedHeight = { 0x7262a16a, 0xd2dc, 0x4e75, { 0x9b,
Packit 1fb8d4
            0xa8, 0x65, 0xc0, 0xc6, 0xd3, 0x2b, 0x13 } };
Packit 1fb8d4
Packit 1fb8d4
#ifndef __IMFDXGIDeviceManager_FWD_DEFINED__
Packit 1fb8d4
#define __IMFDXGIDeviceManager_FWD_DEFINED__
Packit 1fb8d4
typedef interface IMFDXGIDeviceManager IMFDXGIDeviceManager;
Packit 1fb8d4
#endif /* __IMFDXGIDeviceManager_FWD_DEFINED__ */
Packit 1fb8d4
Packit 1fb8d4
#ifndef __IMFDXGIDeviceManager_INTERFACE_DEFINED__
Packit 1fb8d4
#define __IMFDXGIDeviceManager_INTERFACE_DEFINED__
Packit 1fb8d4
Packit 1fb8d4
typedef struct IMFDXGIDeviceManagerVtbl
Packit 1fb8d4
{
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* QueryInterface)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	        REFIID riid, void** ppvObject);
Packit 1fb8d4
	ULONG(STDMETHODCALLTYPE* AddRef)(IMFDXGIDeviceManager* This);
Packit 1fb8d4
	ULONG(STDMETHODCALLTYPE* Release)(IMFDXGIDeviceManager* This);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* CloseDeviceHandle)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	        HANDLE hDevice);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* GetVideoService)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	        HANDLE hDevice, REFIID riid, void** ppService);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* LockDevice)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	                                       HANDLE hDevice, REFIID riid, void** ppUnkDevice, BOOL fBlock);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* OpenDeviceHandle)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	        HANDLE* phDevice);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* ResetDevice)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	                                        IUnknown* pUnkDevice, UINT resetToken);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* TestDevice)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	                                       HANDLE hDevice);
Packit 1fb8d4
	HRESULT(STDMETHODCALLTYPE* UnlockDevice)(IMFDXGIDeviceManager* This,
Packit 1fb8d4
	        HANDLE hDevice, BOOL fSaveState);
Packit 1fb8d4
}
Packit 1fb8d4
IMFDXGIDeviceManagerVtbl;
Packit 1fb8d4
Packit 1fb8d4
interface IMFDXGIDeviceManager
Packit 1fb8d4
{
Packit 1fb8d4
	CONST_VTBL struct IMFDXGIDeviceManagerVtbl* lpVtbl;
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
#endif /* __IMFDXGIDeviceManager_INTERFACE_DEFINED__ */
Packit 1fb8d4
Packit 1fb8d4
typedef HRESULT(__stdcall* pfnMFStartup)(ULONG Version, DWORD dwFlags);
Packit 1fb8d4
typedef HRESULT(__stdcall* pfnMFShutdown)(void);
Packit 1fb8d4
typedef HRESULT(__stdcall* pfnMFCreateSample)(IMFSample** ppIMFSample);
Packit 1fb8d4
typedef HRESULT(__stdcall* pfnMFCreateMemoryBuffer)(DWORD cbMaxLength,
Packit 1fb8d4
        IMFMediaBuffer** ppBuffer);
Packit 1fb8d4
typedef HRESULT(__stdcall* pfnMFCreateMediaType)(IMFMediaType** ppMFType);
Packit 1fb8d4
Packit 1fb8d4
struct _H264_CONTEXT_MF
Packit 1fb8d4
{
Packit 1fb8d4
	ICodecAPI* codecApi;
Packit 1fb8d4
	IMFTransform* transform;
Packit 1fb8d4
	IMFMediaType* inputType;
Packit 1fb8d4
	IMFMediaType* outputType;
Packit 1fb8d4
	IMFSample* sample;
Packit 1fb8d4
	UINT32 frameWidth;
Packit 1fb8d4
	UINT32 frameHeight;
Packit 1fb8d4
	IMFSample* outputSample;
Packit 1fb8d4
	IMFMediaBuffer* outputBuffer;
Packit 1fb8d4
	HMODULE mfplat;
Packit 1fb8d4
	pfnMFStartup MFStartup;
Packit 1fb8d4
	pfnMFShutdown MFShutdown;
Packit 1fb8d4
	pfnMFCreateSample MFCreateSample;
Packit 1fb8d4
	pfnMFCreateMemoryBuffer MFCreateMemoryBuffer;
Packit 1fb8d4
	pfnMFCreateMediaType MFCreateMediaType;
Packit 1fb8d4
};
Packit 1fb8d4
typedef struct _H264_CONTEXT_MF H264_CONTEXT_MF;
Packit 1fb8d4
Packit 1fb8d4
static HRESULT mf_find_output_type(H264_CONTEXT_MF* sys, const GUID* guid,
Packit 1fb8d4
                                   IMFMediaType** ppMediaType)
Packit 1fb8d4
{
Packit 1fb8d4
	DWORD idx = 0;
Packit 1fb8d4
	GUID mediaGuid;
Packit 1fb8d4
	HRESULT hr = S_OK;
Packit 1fb8d4
	IMFMediaType* pMediaType = NULL;
Packit 1fb8d4
Packit 1fb8d4
	while (1)
Packit 1fb8d4
	{
Packit 1fb8d4
		hr = sys->transform->lpVtbl->GetOutputAvailableType(sys->transform, 0, idx,
Packit 1fb8d4
		        &pMediaType);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		pMediaType->lpVtbl->GetGUID(pMediaType, &sMF_MT_SUBTYPE, &mediaGuid);
Packit 1fb8d4
Packit 1fb8d4
		if (IsEqualGUID(&mediaGuid, guid))
Packit 1fb8d4
		{
Packit 1fb8d4
			*ppMediaType = pMediaType;
Packit 1fb8d4
			return S_OK;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pMediaType->lpVtbl->Release(pMediaType);
Packit 1fb8d4
		idx++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return hr;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static HRESULT mf_create_output_sample(H264_CONTEXT* h264, H264_CONTEXT_MF* sys)
Packit 1fb8d4
{
Packit 1fb8d4
	HRESULT hr = S_OK;
Packit 1fb8d4
	MFT_OUTPUT_STREAM_INFO streamInfo;
Packit 1fb8d4
Packit 1fb8d4
	if (sys->outputSample)
Packit 1fb8d4
	{
Packit 1fb8d4
		sys->outputSample->lpVtbl->Release(sys->outputSample);
Packit 1fb8d4
		sys->outputSample = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = sys->MFCreateSample(&sys->outputSample);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "MFCreateSample failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = sys->transform->lpVtbl->GetOutputStreamInfo(sys->transform, 0,
Packit 1fb8d4
	        &streamInfo);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "GetOutputStreamInfo failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = sys->MFCreateMemoryBuffer(streamInfo.cbSize, &sys->outputBuffer);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "MFCreateMemoryBuffer failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	sys->outputSample->lpVtbl->AddBuffer(sys->outputSample, sys->outputBuffer);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "AddBuffer failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	sys->outputBuffer->lpVtbl->Release(sys->outputBuffer);
Packit 1fb8d4
error:
Packit 1fb8d4
	return hr;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int mf_decompress(H264_CONTEXT* h264, const BYTE* pSrcData, UINT32 SrcSize)
Packit 1fb8d4
{
Packit 1fb8d4
	HRESULT hr;
Packit 1fb8d4
	BYTE* pbBuffer = NULL;
Packit 1fb8d4
	DWORD cbMaxLength = 0;
Packit 1fb8d4
	DWORD cbCurrentLength = 0;
Packit 1fb8d4
	DWORD outputStatus = 0;
Packit 1fb8d4
	IMFSample* inputSample = NULL;
Packit 1fb8d4
	IMFMediaBuffer* inputBuffer = NULL;
Packit 1fb8d4
	IMFMediaBuffer* outputBuffer = NULL;
Packit 1fb8d4
	MFT_OUTPUT_DATA_BUFFER outputDataBuffer;
Packit 1fb8d4
	H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*) h264->pSystemData;
Packit 1fb8d4
	INT32* iStride = h264->iStride;
Packit 1fb8d4
	BYTE** pYUVData = h264->pYUVData;
Packit 1fb8d4
	hr = sys->MFCreateMemoryBuffer(SrcSize, &inputBuffer);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "MFCreateMemoryBuffer failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = inputBuffer->lpVtbl->Lock(inputBuffer, &pbBuffer, &cbMaxLength,
Packit 1fb8d4
	                               &cbCurrentLength);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "Lock failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	CopyMemory(pbBuffer, pSrcData, SrcSize);
Packit 1fb8d4
	hr = inputBuffer->lpVtbl->SetCurrentLength(inputBuffer, SrcSize);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "SetCurrentLength failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = inputBuffer->lpVtbl->Unlock(inputBuffer);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "Unlock failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = sys->MFCreateSample(&inputSample);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "MFCreateSample failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	inputSample->lpVtbl->AddBuffer(inputSample, inputBuffer);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "AddBuffer failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	inputBuffer->lpVtbl->Release(inputBuffer);
Packit 1fb8d4
	hr = sys->transform->lpVtbl->ProcessInput(sys->transform, 0, inputSample, 0);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "ProcessInput failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	hr = mf_create_output_sample(h264, sys);
Packit 1fb8d4
Packit 1fb8d4
	if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "mf_create_output_sample failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	outputDataBuffer.dwStreamID = 0;
Packit 1fb8d4
	outputDataBuffer.dwStatus = 0;
Packit 1fb8d4
	outputDataBuffer.pEvents = NULL;
Packit 1fb8d4
	outputDataBuffer.pSample = sys->outputSample;
Packit 1fb8d4
	hr = sys->transform->lpVtbl->ProcessOutput(sys->transform, 0, 1,
Packit 1fb8d4
	        &outputDataBuffer, &outputStatus);
Packit 1fb8d4
Packit 1fb8d4
	if (hr == MF_E_TRANSFORM_STREAM_CHANGE)
Packit 1fb8d4
	{
Packit 1fb8d4
		UINT32 stride = 0;
Packit 1fb8d4
		UINT64 frameSize = 0;
Packit 1fb8d4
Packit 1fb8d4
		if (sys->outputType)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->outputType->lpVtbl->Release(sys->outputType);
Packit 1fb8d4
			sys->outputType = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = mf_find_output_type(sys, &sMFVideoFormat_IYUV, &sys->outputType);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "mf_find_output_type failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->transform->lpVtbl->SetOutputType(sys->transform, 0, sys->outputType,
Packit 1fb8d4
		        0);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetOutputType failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = mf_create_output_sample(h264, sys);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "mf_create_output_sample failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->outputType->lpVtbl->GetUINT64(sys->outputType, &sMF_MT_FRAME_SIZE,
Packit 1fb8d4
		                                        &frameSize);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "GetUINT64(MF_MT_FRAME_SIZE) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		sys->frameWidth = (UINT32)(frameSize >> 32);
Packit 1fb8d4
		sys->frameHeight = (UINT32) frameSize;
Packit 1fb8d4
		hr = sys->outputType->lpVtbl->GetUINT32(sys->outputType, &sMF_MT_DEFAULT_STRIDE,
Packit 1fb8d4
		                                        &stride);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "GetUINT32(MF_MT_DEFAULT_STRIDE) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (!avc420_ensure_buffer(h264, stride, sys->frameWidth, sys->frameHeight))
Packit 1fb8d4
			goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (hr == MF_E_TRANSFORM_NEED_MORE_INPUT)
Packit 1fb8d4
	{
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (FAILED(hr))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(h264->log, WLOG_ERROR, "ProcessOutput failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
		goto error;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		int offset = 0;
Packit 1fb8d4
		BYTE* buffer = NULL;
Packit 1fb8d4
		DWORD bufferCount = 0;
Packit 1fb8d4
		DWORD cbMaxLength = 0;
Packit 1fb8d4
		DWORD cbCurrentLength = 0;
Packit 1fb8d4
		hr = sys->outputSample->lpVtbl->GetBufferCount(sys->outputSample, &bufferCount);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "GetBufferCount failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->outputSample->lpVtbl->GetBufferByIndex(sys->outputSample, 0,
Packit 1fb8d4
		        &outputBuffer);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "GetBufferByIndex failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = outputBuffer->lpVtbl->Lock(outputBuffer, &buffer, &cbMaxLength,
Packit 1fb8d4
		                                &cbCurrentLength);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "Lock failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		CopyMemory(pYUVData[0], &buffer[offset], iStride[0] * sys->frameHeight);
Packit 1fb8d4
		offset += iStride[0] * sys->frameHeight;
Packit 1fb8d4
		CopyMemory(pYUVData[1], &buffer[offset], iStride[1] * (sys->frameHeight / 2));
Packit 1fb8d4
		offset += iStride[1] * (sys->frameHeight / 2);
Packit 1fb8d4
		CopyMemory(pYUVData[2], &buffer[offset], iStride[2] * (sys->frameHeight / 2));
Packit 1fb8d4
		offset += iStride[2] * (sys->frameHeight / 2);
Packit 1fb8d4
		hr = outputBuffer->lpVtbl->Unlock(outputBuffer);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "Unlock failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		outputBuffer->lpVtbl->Release(outputBuffer);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	inputSample->lpVtbl->Release(inputSample);
Packit 1fb8d4
	return 1;
Packit 1fb8d4
error:
Packit 1fb8d4
	fprintf(stderr, "mf_decompress error\n");
Packit 1fb8d4
	return -1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int mf_compress(H264_CONTEXT* h264, const BYTE** ppSrcYuv, const UINT32* pStride,
Packit 1fb8d4
                       BYTE** ppDstData, UINT32* pDstSize)
Packit 1fb8d4
{
Packit 1fb8d4
	H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*) h264->pSystemData;
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL mf_plat_loaded(H264_CONTEXT_MF* sys)
Packit 1fb8d4
{
Packit 1fb8d4
	return sys->MFStartup && sys->MFShutdown && sys->MFCreateSample
Packit 1fb8d4
		&& sys->MFCreateMemoryBuffer && sys->MFCreateMediaType;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void mf_uninit(H264_CONTEXT* h264)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 x;
Packit 1fb8d4
	H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*) h264->pSystemData;
Packit 1fb8d4
Packit 1fb8d4
	if (sys)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (sys->transform)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->transform->lpVtbl->Release(sys->transform);
Packit 1fb8d4
			sys->transform = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->codecApi)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->codecApi->lpVtbl->Release(sys->codecApi);
Packit 1fb8d4
			sys->codecApi = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->inputType)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->inputType->lpVtbl->Release(sys->inputType);
Packit 1fb8d4
			sys->inputType = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->outputType)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->outputType->lpVtbl->Release(sys->outputType);
Packit 1fb8d4
			sys->outputType = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->outputSample)
Packit 1fb8d4
		{
Packit 1fb8d4
			sys->outputSample->lpVtbl->Release(sys->outputSample);
Packit 1fb8d4
			sys->outputSample = NULL;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (sys->mfplat)
Packit 1fb8d4
		{
Packit 1fb8d4
			if (mf_plat_loaded(sys))
Packit 1fb8d4
				sys->MFShutdown();
Packit 1fb8d4
Packit 1fb8d4
			FreeLibrary(sys->mfplat);
Packit 1fb8d4
			sys->mfplat = NULL;
Packit 1fb8d4
Packit 1fb8d4
			if (mf_plat_loaded(sys))
Packit 1fb8d4
				CoUninitialize();
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		for (x = 0; x < sizeof(h264->pYUVData) / sizeof(h264->pYUVData[0]); x++)
Packit 1fb8d4
			_aligned_free(h264->pYUVData[x]);
Packit 1fb8d4
Packit 1fb8d4
		memset(h264->pYUVData, 0, sizeof(h264->pYUVData));
Packit 1fb8d4
		memset(h264->iStride, 0, sizeof(h264->iStride));
Packit 1fb8d4
		
Packit 1fb8d4
		free(sys);
Packit 1fb8d4
		h264->pSystemData = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL mf_init(H264_CONTEXT* h264)
Packit 1fb8d4
{
Packit 1fb8d4
	HRESULT hr;
Packit 1fb8d4
	H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*) calloc(1, sizeof(H264_CONTEXT_MF));
Packit 1fb8d4
Packit 1fb8d4
	if (!sys)
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	h264->pSystemData = (void*) sys;
Packit 1fb8d4
	/* http://decklink-sdk-delphi.googlecode.com/svn/trunk/Blackmagic%20DeckLink%20SDK%209.7/Win/Samples/Streaming/StreamingPreview/DecoderMF.cpp */
Packit 1fb8d4
	sys->mfplat = LoadLibraryA("mfplat.dll");
Packit 1fb8d4
Packit 1fb8d4
	if (!sys->mfplat)
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	sys->MFStartup = (pfnMFStartup) GetProcAddress(sys->mfplat, "MFStartup");
Packit 1fb8d4
	sys->MFShutdown = (pfnMFShutdown) GetProcAddress(sys->mfplat, "MFShutdown");
Packit 1fb8d4
	sys->MFCreateSample = (pfnMFCreateSample) GetProcAddress(sys->mfplat,
Packit 1fb8d4
	                      "MFCreateSample");
Packit 1fb8d4
	sys->MFCreateMemoryBuffer = (pfnMFCreateMemoryBuffer) GetProcAddress(
Packit 1fb8d4
	                                sys->mfplat, "MFCreateMemoryBuffer");
Packit 1fb8d4
	sys->MFCreateMediaType = (pfnMFCreateMediaType) GetProcAddress(sys->mfplat,
Packit 1fb8d4
	                         "MFCreateMediaType");
Packit 1fb8d4
Packit 1fb8d4
	if (!mf_plat_loaded(sys))
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	CoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
Packit 1fb8d4
Packit 1fb8d4
	if (h264->Compressor)
Packit 1fb8d4
	{
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		VARIANT var = { 0 };
Packit 1fb8d4
		hr = sys->MFStartup(MF_VERSION, 0);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "MFStartup failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = CoCreateInstance(&sCLSID_CMSH264DecoderMFT, NULL, CLSCTX_INPROC_SERVER,
Packit 1fb8d4
		                      &sIID_IMFTransform, (void**) &sys->transform);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR,
Packit 1fb8d4
			           "CoCreateInstance(CLSID_CMSH264DecoderMFT) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->transform->lpVtbl->QueryInterface(sys->transform, &sIID_ICodecAPI,
Packit 1fb8d4
		        (void**) &sys->codecApi);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "QueryInterface(IID_ICodecAPI) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		var.vt = VT_UI4;
Packit 1fb8d4
		var.ulVal = 1;
Packit 1fb8d4
		hr = sys->codecApi->lpVtbl->SetValue(sys->codecApi, &sCODECAPI_AVLowLatencyMode,
Packit 1fb8d4
		                                     &var);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetValue(CODECAPI_AVLowLatencyMode) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->MFCreateMediaType(&sys->inputType);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "MFCreateMediaType failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->inputType->lpVtbl->SetGUID(sys->inputType, &sMF_MT_MAJOR_TYPE,
Packit 1fb8d4
		                                     &sMFMediaType_Video);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetGUID(MF_MT_MAJOR_TYPE) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->inputType->lpVtbl->SetGUID(sys->inputType, &sMF_MT_SUBTYPE,
Packit 1fb8d4
		                                     &sMFVideoFormat_H264);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetGUID(MF_MT_SUBTYPE) failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->transform->lpVtbl->SetInputType(sys->transform, 0, sys->inputType, 0);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetInputType failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = mf_find_output_type(sys, &sMFVideoFormat_IYUV, &sys->outputType);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "mf_find_output_type failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = sys->transform->lpVtbl->SetOutputType(sys->transform, 0, sys->outputType,
Packit 1fb8d4
		        0);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "SetOutputType failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		hr = mf_create_output_sample(h264, sys);
Packit 1fb8d4
Packit 1fb8d4
		if (FAILED(hr))
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_Print(h264->log, WLOG_ERROR, "mf_create_output_sample failure: 0x%08"PRIX32"", hr);
Packit 1fb8d4
			goto error;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
error:
Packit 1fb8d4
	WLog_Print(h264->log, WLOG_ERROR, "mf_init failure");
Packit 1fb8d4
	mf_uninit(h264);
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
H264_CONTEXT_SUBSYSTEM g_Subsystem_MF =
Packit 1fb8d4
{
Packit 1fb8d4
	"MediaFoundation",
Packit 1fb8d4
	mf_init,
Packit 1fb8d4
	mf_uninit,
Packit 1fb8d4
	mf_decompress,
Packit 1fb8d4
	mf_compress
Packit 1fb8d4
};
Packit 1fb8d4