2  * QuickTime Toolkit decoder filter for video
 
   4  * Copyright 2010 Aric Stewart, CodeWeavers
 
   6  * This library is free software; you can redistribute it and/or
 
   7  * modify it under the terms of the GNU Lesser General Public
 
   8  * License as published by the Free Software Foundation; either
 
   9  * version 2.1 of the License, or (at your option) any later version.
 
  11  * This library is distributed in the hope that it will be useful,
 
  12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
  13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 
  14  * Lesser General Public License for more details.
 
  16  * You should have received a copy of the GNU Lesser General Public
 
  17  * License along with this library; if not, write to the Free Software
 
  18  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
 
  23 #define ULONG CoreFoundation_ULONG
 
  24 #define HRESULT CoreFoundation_HRESULT
 
  26 #define LoadResource __carbon_LoadResource
 
  27 #define CompareString __carbon_CompareString
 
  28 #define GetCurrentThread __carbon_GetCurrentThread
 
  29 #define GetCurrentProcess __carbon_GetCurrentProcess
 
  30 #define AnimatePalette __carbon_AnimatePalette
 
  31 #define EqualRgn __carbon_EqualRgn
 
  32 #define FillRgn __carbon_FillRgn
 
  33 #define FrameRgn __carbon_FrameRgn
 
  34 #define GetPixel __carbon_GetPixel
 
  35 #define InvertRgn __carbon_InvertRgn
 
  36 #define LineTo __carbon_LineTo
 
  37 #define OffsetRgn __carbon_OffsetRgn
 
  38 #define PaintRgn __carbon_PaintRgn
 
  39 #define Polygon __carbon_Polygon
 
  40 #define ResizePalette __carbon_ResizePalette
 
  41 #define SetRectRgn __carbon_SetRectRgn
 
  43 #define CheckMenuItem __carbon_CheckMenuItem
 
  44 #define DeleteMenu __carbon_DeleteMenu
 
  45 #define DrawMenuBar __carbon_DrawMenuBar
 
  46 #define EnableMenuItem __carbon_EnableMenuItem
 
  47 #define EqualRect __carbon_EqualRect
 
  48 #define FillRect __carbon_FillRect
 
  49 #define FrameRect __carbon_FrameRect
 
  50 #define GetCursor __carbon_GetCursor
 
  51 #define GetMenu __carbon_GetMenu
 
  52 #define InvertRect __carbon_InvertRect
 
  53 #define IsWindowVisible __carbon_IsWindowVisible
 
  54 #define MoveWindow __carbon_MoveWindow
 
  55 #define OffsetRect __carbon_OffsetRect
 
  56 #define PtInRect __carbon_PtInRect
 
  57 #define SetCursor __carbon_SetCursor
 
  58 #define SetRect __carbon_SetRect
 
  59 #define ShowCursor __carbon_ShowCursor
 
  60 #define ShowWindow __carbon_ShowWindow
 
  61 #define UnionRect __carbon_UnionRect
 
  63 #include <QuickTime/ImageCompression.h>
 
  64 #include <CoreVideo/CVPixelBuffer.h>
 
  68 #undef GetCurrentThread
 
  71 #undef GetCurrentProcess
 
  94 #undef IsWindowVisible
 
 107 #undef STDMETHODCALLTYPE
 
 122 #include "dvdmedia.h"
 
 126 #include "wine/unicode.h"
 
 127 #include "wine/debug.h"
 
 128 #include "wine/strmbase.h"
 
 130 #include "qtprivate.h"
 
 132 extern CLSID CLSID_QTVDecoder;
 
 134 WINE_DEFAULT_DEBUG_CHANNEL(qtdecoder);
 
 136 typedef struct QTVDecoderImpl
 
 139     IUnknown *seekthru_unk;
 
 141     ImageDescriptionHandle hImageDescription;
 
 142     CFMutableDictionaryRef outputBufferAttributes;
 
 143     ICMDecompressionSessionRef decompressionSession;
 
 150 static const IBaseFilterVtbl QTVDecoder_Vtbl;
 
 152 static void trackingCallback(
 
 153                     void *decompressionTrackingRefCon,
 
 155                     ICMDecompressionTrackingFlags decompressionTrackingFlags,
 
 156                     CVPixelBufferRef pixelBuffer,
 
 157                     TimeValue64 displayTime,
 
 158                     TimeValue64 displayDuration,
 
 159                     ICMValidTimeFlags validTimeFlags,
 
 161                     void *sourceFrameRefCon )
 
 163     QTVDecoderImpl *This = (QTVDecoderImpl*)decompressionTrackingRefCon;
 
 164     IMediaSample *pSample = (IMediaSample*)sourceFrameRefCon;
 
 166     IMediaSample* pOutSample = NULL;
 
 172         ERR("Error from Codec, no frame decompressed\n");
 
 178         ERR("No pixel buffer, no frame decompressed\n");
 
 182     EnterCriticalSection(&This->tf.csReceive);
 
 183     hr = BaseOutputPinImpl_GetDeliveryBuffer((BaseOutputPin*)This->tf.ppPins[1], &pOutSample, NULL, NULL, 0);
 
 185         ERR("Unable to get delivery buffer (%x)\n", hr);
 
 189     hr = IMediaSample_SetActualDataLength(pOutSample, 0);
 
 192     hr = IMediaSample_GetPointer(pOutSample, &pbDstStream);
 
 194         ERR("Unable to get pointer to buffer (%x)\n", hr);
 
 198     cbDstStream = IMediaSample_GetSize(pOutSample);
 
 199     if (cbDstStream < This->outputSize) {
 
 200         ERR("Sample size is too small %d < %d\n", cbDstStream, This->outputSize);
 
 205     hr = AccessPixelBufferPixels(pixelBuffer, pbDstStream);
 
 209     IMediaSample_SetActualDataLength(pOutSample, This->outputSize);
 
 211     IMediaSample_SetPreroll(pOutSample, (IMediaSample_IsPreroll(pSample) == S_OK));
 
 212     IMediaSample_SetDiscontinuity(pOutSample, (IMediaSample_IsDiscontinuity(pSample) == S_OK));
 
 213     IMediaSample_SetSyncPoint(pOutSample, (IMediaSample_IsSyncPoint(pSample) == S_OK));
 
 216         IMediaSample_SetTime(pOutSample, NULL, NULL);
 
 219         LONGLONG tStart, tStop;
 
 221         if (validTimeFlags & kICMValidTime_DisplayTimeStampIsValid)
 
 222             tStart = displayTime;
 
 225         if (validTimeFlags & kICMValidTime_DisplayDurationIsValid)
 
 226             tStop = tStart + displayDuration;
 
 230         IMediaSample_SetTime(pOutSample, &tStart, &tStop);
 
 233     LeaveCriticalSection(&This->tf.csReceive);
 
 234     hr = BaseOutputPinImpl_Deliver((BaseOutputPin*)This->tf.ppPins[1], pOutSample);
 
 235     EnterCriticalSection(&This->tf.csReceive);
 
 236     if (hr != S_OK && hr != VFW_E_NOT_CONNECTED)
 
 237         ERR("Error sending sample (%x)\n", hr);
 
 240     LeaveCriticalSection(&This->tf.csReceive);
 
 242         IMediaSample_Release(pOutSample);
 
 247 static HRESULT WINAPI QTVDecoder_StartStreaming(TransformFilter* pTransformFilter)
 
 249     QTVDecoderImpl* This = (QTVDecoderImpl*)pTransformFilter;
 
 251     ICMDecompressionSessionOptionsRef sessionOptions = NULL;
 
 252     ICMDecompressionTrackingCallbackRecord trackingCallbackRecord;
 
 254     TRACE("(%p)->()\n", This);
 
 256     trackingCallbackRecord.decompressionTrackingCallback = trackingCallback;
 
 257     trackingCallbackRecord.decompressionTrackingRefCon = (void*)This;
 
 259     err = ICMDecompressionSessionCreate(NULL, This->hImageDescription, sessionOptions, This->outputBufferAttributes, &trackingCallbackRecord, &This->decompressionSession);
 
 263         ERR("Error with ICMDecompressionSessionCreate %i\n",err);
 
 270 static HRESULT WINAPI QTVDecoder_Receive(TransformFilter *tf, IMediaSample *pSample)
 
 272     QTVDecoderImpl* This = (QTVDecoderImpl *)tf;
 
 277     ICMFrameTimeRecord frameTime = {{0}};
 
 279     TimeScale timeScale = 1;
 
 280     OSStatus err = noErr;
 
 281     LONGLONG tStart, tStop;
 
 283     hr = IMediaSample_GetPointer(pSample, &pbSrcStream);
 
 286         ERR("Cannot get pointer to sample data (%x)\n", hr);
 
 290     cbSrcStream = IMediaSample_GetActualDataLength(pSample);
 
 292     if (IMediaSample_GetTime(pSample, &tStart, &tStop) != S_OK)
 
 297     frameTime.recordSize = sizeof(ICMFrameTimeRecord);
 
 298     *(TimeValue64 *)&frameTime.value = tStart;
 
 300     frameTime.rate = fixed1;
 
 301     frameTime.duration = tStop - tStart;
 
 302     frameTime.frameNumber = 0;
 
 303     frameTime.flags = icmFrameTimeIsNonScheduledDisplayTime;
 
 305     err = ICMDecompressionSessionDecodeFrame(This->decompressionSession,
 
 306             (UInt8 *)pbSrcStream, cbSrcStream, NULL, &frameTime, pSample);
 
 310         ERR("Error with ICMDecompressionSessionDecodeFrame\n");
 
 315     ICMDecompressionSessionSetNonScheduledDisplayTime(This->decompressionSession, time, timeScale, 0);
 
 316     ICMDecompressionSessionFlush(This->decompressionSession);
 
 323 static HRESULT WINAPI QTVDecoder_StopStreaming(TransformFilter* pTransformFilter)
 
 325     QTVDecoderImpl* This = (QTVDecoderImpl*)pTransformFilter;
 
 327     TRACE("(%p)->()\n", This);
 
 329     if (This->decompressionSession)
 
 330         ICMDecompressionSessionRelease(This->decompressionSession);
 
 331     This->decompressionSession = NULL;
 
 336 static HRESULT WINAPI QTVDecoder_SetMediaType(TransformFilter *tf, PIN_DIRECTION dir, const AM_MEDIA_TYPE * pmt)
 
 338     QTVDecoderImpl* This = (QTVDecoderImpl*)tf;
 
 339     HRESULT hr = VFW_E_TYPE_NOT_ACCEPTED;
 
 341     AM_MEDIA_TYPE *outpmt = &This->tf.pmt;
 
 342     CFNumberRef n = NULL;
 
 344     TRACE("(%p)->(%p)\n", This, pmt);
 
 346     if (dir != PINDIR_INPUT)
 
 349     FreeMediaType(outpmt);
 
 350     CopyMediaType(outpmt, pmt);
 
 352     if (This->hImageDescription)
 
 353         DisposeHandle((Handle)This->hImageDescription);
 
 355     This->hImageDescription = (ImageDescriptionHandle)
 
 356         NewHandleClear(sizeof(ImageDescription));
 
 358     if (This->hImageDescription != NULL)
 
 360         (**This->hImageDescription).idSize = sizeof(ImageDescription);
 
 361         (**This->hImageDescription).spatialQuality = codecNormalQuality;
 
 362         (**This->hImageDescription).frameCount = 1;
 
 363         (**This->hImageDescription).clutID = -1;
 
 367         ERR("Failed to create ImageDescription\n");
 
 371     /* Check root (GUID w/o FOURCC) */
 
 372     if ((IsEqualIID(&pmt->majortype, &MEDIATYPE_Video)) &&
 
 373         (!memcmp(((const char *)&pmt->subtype)+4, ((const char *)&MEDIATYPE_Video)+4, sizeof(GUID)-4)))
 
 375         VIDEOINFOHEADER *format1 = (VIDEOINFOHEADER *)outpmt->pbFormat;
 
 376         VIDEOINFOHEADER2 *format2 = (VIDEOINFOHEADER2 *)outpmt->pbFormat;
 
 377         BITMAPINFOHEADER *bmi;
 
 379         DecompressorComponent dc;
 
 381         DWORD outputWidth, outputHeight, outputDepth;
 
 383         if (IsEqualIID(&pmt->formattype, &FORMAT_VideoInfo))
 
 384             bmi = &format1->bmiHeader;
 
 385         else if (IsEqualIID(&pmt->formattype, &FORMAT_VideoInfo2))
 
 386             bmi = &format2->bmiHeader;
 
 390         TRACE("Fourcc: %s\n", debugstr_an((const char *)&pmt->subtype.Data1, 4));
 
 391         fourCC = ((const char *)&pmt->subtype.Data1)[3] |
 
 392                  (((const char *)&pmt->subtype.Data1)[2]<<8) |
 
 393                  (((const char *)&pmt->subtype.Data1)[1]<<16) |
 
 394                  (((const char *)&pmt->subtype.Data1)[0]<<24);
 
 396         err = FindCodec(fourCC,NULL,NULL,&dc);
 
 397         if (err != noErr || dc == 0x0)
 
 399             TRACE("Codec not found\n");
 
 403         outputWidth = bmi->biWidth;
 
 404         outputHeight = bmi->biHeight;
 
 406         (**This->hImageDescription).cType = fourCC;
 
 407         (**This->hImageDescription).width = outputWidth;
 
 408         (**This->hImageDescription).height = outputHeight;
 
 409         (**This->hImageDescription).depth = bmi->biBitCount;
 
 410         (**This->hImageDescription).hRes = 72<<16;
 
 411         (**This->hImageDescription).vRes = 72<<16;
 
 413         if (This->outputBufferAttributes)
 
 414             CFRelease(This->outputBufferAttributes);
 
 416         This->outputBufferAttributes = CFDictionaryCreateMutable(NULL, 0, &kCFTypeDictionaryKeyCallBacks, &kCFTypeDictionaryValueCallBacks);
 
 417         if (!This->outputBufferAttributes)
 
 419             ERR("Failed to create outputBufferAttributes\n");
 
 423         n = CFNumberCreate(NULL, kCFNumberIntType, &outputWidth);
 
 424         CFDictionaryAddValue(This->outputBufferAttributes, kCVPixelBufferWidthKey, n);
 
 427         n = CFNumberCreate(NULL, kCFNumberIntType, &outputHeight);
 
 428         CFDictionaryAddValue(This->outputBufferAttributes, kCVPixelBufferHeightKey, n);
 
 431         /* yes this looks wrong.  but 32ARGB is 24 RGB with an alpha channel */
 
 432         format = k32ARGBPixelFormat;
 
 433         n = CFNumberCreate(NULL, kCFNumberIntType, &format);
 
 434         CFDictionaryAddValue(This->outputBufferAttributes, kCVPixelBufferPixelFormatTypeKey, n);
 
 437         CFDictionaryAddValue(This->outputBufferAttributes, kCVPixelBufferCGBitmapContextCompatibilityKey, kCFBooleanTrue);
 
 438         CFDictionaryAddValue(This->outputBufferAttributes, kCVPixelBufferCGImageCompatibilityKey, kCFBooleanTrue);
 
 441         This->outputSize = outputWidth * outputHeight * outputDepth;
 
 442         bmi->biCompression =  BI_RGB;
 
 443         bmi->biBitCount =  24;
 
 444         outpmt->subtype = MEDIASUBTYPE_RGB24;
 
 450     if (This->hImageDescription)
 
 452         DisposeHandle((Handle)This->hImageDescription);
 
 453         This->hImageDescription =  NULL;
 
 455     if (This->outputBufferAttributes)
 
 457         CFRelease(This->outputBufferAttributes);
 
 458         This->outputBufferAttributes = NULL;
 
 461     TRACE("Connection refused\n");
 
 465 static HRESULT WINAPI QTVDecoder_BreakConnect(TransformFilter *tf, PIN_DIRECTION dir)
 
 467     QTVDecoderImpl *This = (QTVDecoderImpl *)tf;
 
 469     TRACE("(%p)->()\n", This);
 
 471     if (This->hImageDescription)
 
 472         DisposeHandle((Handle)This->hImageDescription);
 
 473     if (This->outputBufferAttributes)
 
 474         CFRelease(This->outputBufferAttributes);
 
 476     This->hImageDescription = NULL;
 
 477     This->outputBufferAttributes = NULL;
 
 482 static HRESULT WINAPI QTVDecoder_DecideBufferSize(TransformFilter *tf, IMemAllocator *pAlloc, ALLOCATOR_PROPERTIES *ppropInputRequest)
 
 484     QTVDecoderImpl *This = (QTVDecoderImpl*)tf;
 
 485     ALLOCATOR_PROPERTIES actual;
 
 489     if (!ppropInputRequest->cbAlign)
 
 490         ppropInputRequest->cbAlign = 1;
 
 492     if (ppropInputRequest->cbBuffer < This->outputSize)
 
 493             ppropInputRequest->cbBuffer = This->outputSize + ppropInputRequest->cbAlign;
 
 495     if (!ppropInputRequest->cBuffers)
 
 496         ppropInputRequest->cBuffers = 1;
 
 498     return IMemAllocator_SetProperties(pAlloc, ppropInputRequest, &actual);
 
 501 static const TransformFilterFuncTable QTVDecoder_FuncsTable = {
 
 502     QTVDecoder_DecideBufferSize,
 
 503     QTVDecoder_StartStreaming,
 
 505     QTVDecoder_StopStreaming,
 
 507     QTVDecoder_SetMediaType,
 
 509     QTVDecoder_BreakConnect,
 
 516 IUnknown * CALLBACK QTVDecoder_create(IUnknown * pUnkOuter, HRESULT* phr)
 
 519     QTVDecoderImpl * This;
 
 521     TRACE("(%p, %p)\n", pUnkOuter, phr);
 
 527         *phr = CLASS_E_NOAGGREGATION;
 
 531     hr = TransformFilter_Construct(&QTVDecoder_Vtbl, sizeof(QTVDecoderImpl), &CLSID_QTVDecoder, &QTVDecoder_FuncsTable, (IBaseFilter**)&This);
 
 540         ISeekingPassThru *passthru;
 
 541         hr = CoCreateInstance(&CLSID_SeekingPassThru, (IUnknown*)This, CLSCTX_INPROC_SERVER, &IID_IUnknown, (void**)&This->seekthru_unk);
 
 542         IUnknown_QueryInterface(This->seekthru_unk, &IID_ISeekingPassThru, (void**)&passthru);
 
 543         ISeekingPassThru_Init(passthru, FALSE, (IPin*)This->tf.ppPins[0]);
 
 544         ISeekingPassThru_Release(passthru);
 
 548     return (IUnknown*)This;
 
 551 HRESULT WINAPI QTVDecoder_QueryInterface(IBaseFilter * iface, REFIID riid, LPVOID * ppv)
 
 554     QTVDecoderImpl *This = (QTVDecoderImpl *)iface;
 
 555     TRACE("(%p/%p)->(%s, %p)\n", This, iface, debugstr_guid(riid), ppv);
 
 557     if (IsEqualIID(riid, &IID_IMediaSeeking))
 
 558         return IUnknown_QueryInterface(This->seekthru_unk, riid, ppv);
 
 560     hr = TransformFilterImpl_QueryInterface(iface, riid, ppv);
 
 565 static const IBaseFilterVtbl QTVDecoder_Vtbl =
 
 567     QTVDecoder_QueryInterface,
 
 568     BaseFilterImpl_AddRef,
 
 569     TransformFilterImpl_Release,
 
 570     BaseFilterImpl_GetClassID,
 
 571     TransformFilterImpl_Stop,
 
 572     TransformFilterImpl_Pause,
 
 573     TransformFilterImpl_Run,
 
 574     BaseFilterImpl_GetState,
 
 575     BaseFilterImpl_SetSyncSource,
 
 576     BaseFilterImpl_GetSyncSource,
 
 577     BaseFilterImpl_EnumPins,
 
 578     TransformFilterImpl_FindPin,
 
 579     BaseFilterImpl_QueryFilterInfo,
 
 580     BaseFilterImpl_JoinFilterGraph,
 
 581     BaseFilterImpl_QueryVendorInfo