20#include <winpr/winpr.h>
21#include <winpr/library.h>
23#include <freerdp/log.h>
24#include <freerdp/codec/h264.h>
31#include <wmcodecdsp.h>
32#include <mftransform.h>
36#define TAG FREERDP_TAG("codec")
38static const GUID sCLSID_CMSH264DecoderMFT = {
39 0x62CE7E72, 0x4C71, 0x4d20, { 0xB1, 0x5D, 0x45, 0x28, 0x31, 0xA8, 0x7D, 0x9D }
41static const GUID sIID_IMFTransform = {
42 0xbf94c121, 0x5b05, 0x4e6f, { 0x80, 0x00, 0xba, 0x59, 0x89, 0x61, 0x41, 0x4d }
44static const GUID sMF_MT_MAJOR_TYPE = {
45 0x48eba18e, 0xf8c9, 0x4687, { 0xbf, 0x11, 0x0a, 0x74, 0xc9, 0xf9, 0x6a, 0x8f }
47static const GUID sMF_MT_FRAME_SIZE = {
48 0x1652c33d, 0xd6b2, 0x4012, { 0xb8, 0x34, 0x72, 0x03, 0x08, 0x49, 0xa3, 0x7d }
50static const GUID sMF_MT_DEFAULT_STRIDE = {
51 0x644b4e48, 0x1e02, 0x4516, { 0xb0, 0xeb, 0xc0, 0x1c, 0xa9, 0xd4, 0x9a, 0xc6 }
53static const GUID sMF_MT_SUBTYPE = {
54 0xf7e34c9a, 0x42e8, 0x4714, { 0xb7, 0x4b, 0xcb, 0x29, 0xd7, 0x2c, 0x35, 0xe5 }
56static const GUID sMFMediaType_Video = {
57 0x73646976, 0x0000, 0x0010, { 0x80, 0x00, 0x00, 0xAA, 0x00, 0x38, 0x9B, 0x71 }
59static const GUID sMFVideoFormat_H264 = {
60 0x34363248, 0x0000, 0x0010, { 0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71 }
62static const GUID sMFVideoFormat_IYUV = {
63 0x56555949, 0x0000, 0x0010, { 0x80, 0x00, 0x00, 0xaa, 0x00, 0x38, 0x9b, 0x71 }
65static const GUID sIID_ICodecAPI = {
66 0x901db4c7, 0x31ce, 0x41a2, { 0x85, 0xdc, 0x8f, 0xa0, 0xbf, 0x41, 0xb8, 0xda }
68static const GUID sCODECAPI_AVLowLatencyMode = {
69 0x9c27891a, 0xed7a, 0x40e1, { 0x88, 0xe8, 0xb2, 0x27, 0x27, 0xa0, 0x24, 0xee }
72typedef HRESULT(__stdcall* pfnMFStartup)(ULONG Version, DWORD dwFlags);
73typedef HRESULT(__stdcall* pfnMFShutdown)(void);
74typedef HRESULT(__stdcall* pfnMFCreateSample)(IMFSample** ppIMFSample);
75typedef HRESULT(__stdcall* pfnMFCreateMemoryBuffer)(DWORD cbMaxLength, IMFMediaBuffer** ppBuffer);
76typedef HRESULT(__stdcall* pfnMFCreateMediaType)(IMFMediaType** ppMFType);
81 IMFTransform* transform;
82 IMFMediaType* inputType;
83 IMFMediaType* outputType;
87 IMFSample* outputSample;
88 IMFMediaBuffer* outputBuffer;
90 pfnMFStartup MFStartup;
91 pfnMFShutdown MFShutdown;
92 pfnMFCreateSample MFCreateSample;
93 pfnMFCreateMemoryBuffer MFCreateMemoryBuffer;
94 pfnMFCreateMediaType MFCreateMediaType;
97static HRESULT mf_find_output_type(H264_CONTEXT_MF* sys,
const GUID* guid,
98 IMFMediaType** ppMediaType)
103 IMFMediaType* pMediaType =
nullptr;
107 hr = sys->transform->lpVtbl->GetOutputAvailableType(sys->transform, 0, idx, &pMediaType);
112 pMediaType->lpVtbl->GetGUID(pMediaType, &sMF_MT_SUBTYPE, &mediaGuid);
114 if (IsEqualGUID(&mediaGuid, guid))
116 *ppMediaType = pMediaType;
120 pMediaType->lpVtbl->Release(pMediaType);
127static HRESULT mf_create_output_sample(H264_CONTEXT* h264, H264_CONTEXT_MF* sys)
130 MFT_OUTPUT_STREAM_INFO streamInfo;
132 if (sys->outputSample)
134 sys->outputSample->lpVtbl->Release(sys->outputSample);
135 sys->outputSample =
nullptr;
138 hr = sys->MFCreateSample(&sys->outputSample);
142 WLog_Print(h264->log, WLOG_ERROR,
"MFCreateSample failure: 0x%08" PRIX32
"", hr);
146 hr = sys->transform->lpVtbl->GetOutputStreamInfo(sys->transform, 0, &streamInfo);
150 WLog_Print(h264->log, WLOG_ERROR,
"GetOutputStreamInfo failure: 0x%08" PRIX32
"", hr);
154 hr = sys->MFCreateMemoryBuffer(streamInfo.cbSize, &sys->outputBuffer);
158 WLog_Print(h264->log, WLOG_ERROR,
"MFCreateMemoryBuffer failure: 0x%08" PRIX32
"", hr);
162 sys->outputSample->lpVtbl->AddBuffer(sys->outputSample, sys->outputBuffer);
166 WLog_Print(h264->log, WLOG_ERROR,
"AddBuffer failure: 0x%08" PRIX32
"", hr);
170 sys->outputBuffer->lpVtbl->Release(sys->outputBuffer);
175static int mf_decompress(H264_CONTEXT* WINPR_RESTRICT h264,
const BYTE* WINPR_RESTRICT pSrcData,
179 BYTE* pbBuffer =
nullptr;
180 DWORD cbMaxLength = 0;
181 DWORD cbCurrentLength = 0;
182 DWORD outputStatus = 0;
183 IMFSample* inputSample =
nullptr;
184 IMFMediaBuffer* inputBuffer =
nullptr;
185 IMFMediaBuffer* outputBuffer =
nullptr;
186 MFT_OUTPUT_DATA_BUFFER outputDataBuffer;
187 H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*)h264->pSystemData;
188 UINT32* iStride = h264->iStride;
189 BYTE** pYUVData = h264->pYUVData;
190 hr = sys->MFCreateMemoryBuffer(SrcSize, &inputBuffer);
194 WLog_Print(h264->log, WLOG_ERROR,
"MFCreateMemoryBuffer failure: 0x%08" PRIX32
"", hr);
198 hr = inputBuffer->lpVtbl->Lock(inputBuffer, &pbBuffer, &cbMaxLength, &cbCurrentLength);
202 WLog_Print(h264->log, WLOG_ERROR,
"Lock failure: 0x%08" PRIX32
"", hr);
206 CopyMemory(pbBuffer, pSrcData, SrcSize);
207 hr = inputBuffer->lpVtbl->SetCurrentLength(inputBuffer, SrcSize);
211 WLog_Print(h264->log, WLOG_ERROR,
"SetCurrentLength failure: 0x%08" PRIX32
"", hr);
215 hr = inputBuffer->lpVtbl->Unlock(inputBuffer);
219 WLog_Print(h264->log, WLOG_ERROR,
"Unlock failure: 0x%08" PRIX32
"", hr);
223 hr = sys->MFCreateSample(&inputSample);
227 WLog_Print(h264->log, WLOG_ERROR,
"MFCreateSample failure: 0x%08" PRIX32
"", hr);
231 inputSample->lpVtbl->AddBuffer(inputSample, inputBuffer);
235 WLog_Print(h264->log, WLOG_ERROR,
"AddBuffer failure: 0x%08" PRIX32
"", hr);
239 inputBuffer->lpVtbl->Release(inputBuffer);
240 hr = sys->transform->lpVtbl->ProcessInput(sys->transform, 0, inputSample, 0);
244 WLog_Print(h264->log, WLOG_ERROR,
"ProcessInput failure: 0x%08" PRIX32
"", hr);
248 hr = mf_create_output_sample(h264, sys);
252 WLog_Print(h264->log, WLOG_ERROR,
"mf_create_output_sample failure: 0x%08" PRIX32
"", hr);
256 outputDataBuffer.dwStreamID = 0;
257 outputDataBuffer.dwStatus = 0;
258 outputDataBuffer.pEvents =
nullptr;
259 outputDataBuffer.pSample = sys->outputSample;
260 hr = sys->transform->lpVtbl->ProcessOutput(sys->transform, 0, 1, &outputDataBuffer,
263 if (hr == MF_E_TRANSFORM_STREAM_CHANGE)
266 UINT64 frameSize = 0;
270 sys->outputType->lpVtbl->Release(sys->outputType);
271 sys->outputType =
nullptr;
274 hr = mf_find_output_type(sys, &sMFVideoFormat_IYUV, &sys->outputType);
278 WLog_Print(h264->log, WLOG_ERROR,
"mf_find_output_type failure: 0x%08" PRIX32
"", hr);
282 hr = sys->transform->lpVtbl->SetOutputType(sys->transform, 0, sys->outputType, 0);
286 WLog_Print(h264->log, WLOG_ERROR,
"SetOutputType failure: 0x%08" PRIX32
"", hr);
290 hr = mf_create_output_sample(h264, sys);
294 WLog_Print(h264->log, WLOG_ERROR,
"mf_create_output_sample failure: 0x%08" PRIX32
"",
299 hr = sys->outputType->lpVtbl->GetUINT64(sys->outputType, &sMF_MT_FRAME_SIZE, &frameSize);
303 WLog_Print(h264->log, WLOG_ERROR,
304 "GetUINT64(MF_MT_FRAME_SIZE) failure: 0x%08" PRIX32
"", hr);
308 sys->frameWidth = (UINT32)(frameSize >> 32);
309 sys->frameHeight = (UINT32)frameSize;
310 hr = sys->outputType->lpVtbl->GetUINT32(sys->outputType, &sMF_MT_DEFAULT_STRIDE, &stride);
314 WLog_Print(h264->log, WLOG_ERROR,
315 "GetUINT32(MF_MT_DEFAULT_STRIDE) failure: 0x%08" PRIX32
"", hr);
319 if (!avc420_ensure_buffer(h264, stride, sys->frameWidth, sys->frameHeight))
322 else if (hr == MF_E_TRANSFORM_NEED_MORE_INPUT)
327 WLog_Print(h264->log, WLOG_ERROR,
"ProcessOutput failure: 0x%08" PRIX32
"", hr);
333 BYTE* buffer =
nullptr;
334 DWORD bufferCount = 0;
335 DWORD cbMaxLength = 0;
336 DWORD cbCurrentLength = 0;
337 hr = sys->outputSample->lpVtbl->GetBufferCount(sys->outputSample, &bufferCount);
341 WLog_Print(h264->log, WLOG_ERROR,
"GetBufferCount failure: 0x%08" PRIX32
"", hr);
345 hr = sys->outputSample->lpVtbl->GetBufferByIndex(sys->outputSample, 0, &outputBuffer);
349 WLog_Print(h264->log, WLOG_ERROR,
"GetBufferByIndex failure: 0x%08" PRIX32
"", hr);
353 hr = outputBuffer->lpVtbl->Lock(outputBuffer, &buffer, &cbMaxLength, &cbCurrentLength);
357 WLog_Print(h264->log, WLOG_ERROR,
"Lock failure: 0x%08" PRIX32
"", hr);
361 CopyMemory(pYUVData[0], &buffer[offset], iStride[0] * sys->frameHeight);
362 offset += iStride[0] * sys->frameHeight;
363 CopyMemory(pYUVData[1], &buffer[offset], iStride[1] * (sys->frameHeight / 2));
364 offset += iStride[1] * (sys->frameHeight / 2);
365 CopyMemory(pYUVData[2], &buffer[offset], iStride[2] * (sys->frameHeight / 2));
366 offset += iStride[2] * (sys->frameHeight / 2);
367 hr = outputBuffer->lpVtbl->Unlock(outputBuffer);
371 WLog_Print(h264->log, WLOG_ERROR,
"Unlock failure: 0x%08" PRIX32
"", hr);
375 outputBuffer->lpVtbl->Release(outputBuffer);
378 inputSample->lpVtbl->Release(inputSample);
381 (void)fprintf(stderr,
"mf_decompress error\n");
385static int mf_compress(H264_CONTEXT* WINPR_RESTRICT h264,
const BYTE** WINPR_RESTRICT ppSrcYuv,
386 const UINT32* WINPR_RESTRICT pStride, BYTE** WINPR_RESTRICT ppDstData,
387 UINT32* WINPR_RESTRICT pDstSize)
389 H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*)h264->pSystemData;
393static BOOL mf_plat_loaded(H264_CONTEXT_MF* sys)
395 return sys->MFStartup && sys->MFShutdown && sys->MFCreateSample && sys->MFCreateMemoryBuffer &&
396 sys->MFCreateMediaType;
399static void mf_uninit(H264_CONTEXT* h264)
401 H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*)h264->pSystemData;
407 sys->transform->lpVtbl->Release(sys->transform);
408 sys->transform =
nullptr;
413 sys->codecApi->lpVtbl->Release(sys->codecApi);
414 sys->codecApi =
nullptr;
419 sys->inputType->lpVtbl->Release(sys->inputType);
420 sys->inputType =
nullptr;
425 sys->outputType->lpVtbl->Release(sys->outputType);
426 sys->outputType =
nullptr;
429 if (sys->outputSample)
431 sys->outputSample->lpVtbl->Release(sys->outputSample);
432 sys->outputSample =
nullptr;
437 if (mf_plat_loaded(sys))
440 FreeLibrary(sys->mfplat);
441 sys->mfplat =
nullptr;
443 if (mf_plat_loaded(sys))
447 for (
size_t x = 0; x <
sizeof(h264->pYUVData) /
sizeof(h264->pYUVData[0]); x++)
448 winpr_aligned_free(h264->pYUVData[x]);
450 memset(h264->pYUVData, 0,
sizeof(h264->pYUVData));
451 memset(h264->iStride, 0,
sizeof(h264->iStride));
454 h264->pSystemData =
nullptr;
458static BOOL mf_init(H264_CONTEXT* h264)
461 H264_CONTEXT_MF* sys = (H264_CONTEXT_MF*)calloc(1,
sizeof(H264_CONTEXT_MF));
466 h264->pSystemData = (
void*)sys;
469 sys->mfplat = LoadLibraryA(
"mfplat.dll");
474 sys->MFStartup = GetProcAddressAs(sys->mfplat,
"MFStartup", pfnMFStartup);
475 sys->MFShutdown = GetProcAddressAs(sys->mfplat,
"MFShutdown", pfnMFShutdown);
476 sys->MFCreateSample = GetProcAddressAs(sys->mfplat,
"MFCreateSample", pfnMFCreateSample);
477 sys->MFCreateMemoryBuffer =
478 GetProcAddressAs(sys->mfplat,
"MFCreateMemoryBuffer", pfnMFCreateMemoryBuffer);
479 sys->MFCreateMediaType =
480 GetProcAddressAs(sys->mfplat,
"MFCreateMediaType", pfnMFCreateMediaType);
482 if (!mf_plat_loaded(sys))
485 CoInitializeEx(
nullptr, COINIT_APARTMENTTHREADED);
487 if (h264->Compressor)
492 VARIANT var = WINPR_C_ARRAY_INIT;
493 hr = sys->MFStartup(MF_VERSION, 0);
497 WLog_Print(h264->log, WLOG_ERROR,
"MFStartup failure: 0x%08" PRIX32
"", hr);
501 hr = CoCreateInstance(&sCLSID_CMSH264DecoderMFT,
nullptr, CLSCTX_INPROC_SERVER,
502 &sIID_IMFTransform, (
void**)&sys->transform);
506 WLog_Print(h264->log, WLOG_ERROR,
507 "CoCreateInstance(CLSID_CMSH264DecoderMFT) failure: 0x%08" PRIX32
"", hr);
511 hr = sys->transform->lpVtbl->QueryInterface(sys->transform, &sIID_ICodecAPI,
512 (
void**)&sys->codecApi);
516 WLog_Print(h264->log, WLOG_ERROR,
517 "QueryInterface(IID_ICodecAPI) failure: 0x%08" PRIX32
"", hr);
521 var.n1.n2.vt = VT_UI4;
522 var.n1.n2.n3.ulVal = 1;
523 hr = sys->codecApi->lpVtbl->SetValue(sys->codecApi, &sCODECAPI_AVLowLatencyMode, &var);
527 WLog_Print(h264->log, WLOG_ERROR,
528 "SetValue(CODECAPI_AVLowLatencyMode) failure: 0x%08" PRIX32
"", hr);
532 hr = sys->MFCreateMediaType(&sys->inputType);
536 WLog_Print(h264->log, WLOG_ERROR,
"MFCreateMediaType failure: 0x%08" PRIX32
"", hr);
540 hr = sys->inputType->lpVtbl->SetGUID(sys->inputType, &sMF_MT_MAJOR_TYPE,
541 &sMFMediaType_Video);
545 WLog_Print(h264->log, WLOG_ERROR,
"SetGUID(MF_MT_MAJOR_TYPE) failure: 0x%08" PRIX32
"",
550 hr = sys->inputType->lpVtbl->SetGUID(sys->inputType, &sMF_MT_SUBTYPE, &sMFVideoFormat_H264);
554 WLog_Print(h264->log, WLOG_ERROR,
"SetGUID(MF_MT_SUBTYPE) failure: 0x%08" PRIX32
"",
559 hr = sys->transform->lpVtbl->SetInputType(sys->transform, 0, sys->inputType, 0);
563 WLog_Print(h264->log, WLOG_ERROR,
"SetInputType failure: 0x%08" PRIX32
"", hr);
567 hr = mf_find_output_type(sys, &sMFVideoFormat_IYUV, &sys->outputType);
571 WLog_Print(h264->log, WLOG_ERROR,
"mf_find_output_type failure: 0x%08" PRIX32
"", hr);
575 hr = sys->transform->lpVtbl->SetOutputType(sys->transform, 0, sys->outputType, 0);
579 WLog_Print(h264->log, WLOG_ERROR,
"SetOutputType failure: 0x%08" PRIX32
"", hr);
583 hr = mf_create_output_sample(h264, sys);
587 WLog_Print(h264->log, WLOG_ERROR,
"mf_create_output_sample failure: 0x%08" PRIX32
"",
595 WLog_Print(h264->log, WLOG_ERROR,
"mf_init failure");
600const H264_CONTEXT_SUBSYSTEM g_Subsystem_MF = {
"MediaFoundation", mf_init, mf_uninit,
601 mf_decompress, mf_compress };