DeCSSFilter.cpp
上传用户:xjjlds
上传日期:2015-12-05
资源大小:22823k
文件大小:7k
源码类别:

多媒体编程

开发平台:

Visual C++

  1. /* 
  2.  * Copyright (C) 2003-2005 Gabest
  3.  * http://www.gabest.org
  4.  *
  5.  *  This Program is free software; you can redistribute it and/or modify
  6.  *  it under the terms of the GNU General Public License as published by
  7.  *  the Free Software Foundation; either version 2, or (at your option)
  8.  *  any later version.
  9.  *   
  10.  *  This Program is distributed in the hope that it will be useful,
  11.  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
  12.  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13.  *  GNU General Public License for more details.
  14.  *   
  15.  *  You should have received a copy of the GNU General Public License
  16.  *  along with GNU Make; see the file COPYING.  If not, write to
  17.  *  the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. 
  18.  *  http://www.gnu.org/copyleft/gpl.html
  19.  *
  20.  */
  21. #include "stdafx.h"
  22. #include <atlbase.h>
  23. #include "DeCSSFilter.h"
  24. #include "......decssDeCSSInputPin.h"
  25. #include "......DSUtilDSUtil.h"
  26. #ifdef REGISTER_FILTER
  27. const AMOVIESETUP_MEDIATYPE sudPinTypesIn[] =
  28. {
  29. {&MEDIATYPE_DVD_ENCRYPTED_PACK, &MEDIASUBTYPE_NULL},
  30. };
  31. const AMOVIESETUP_MEDIATYPE sudPinTypesOut[] =
  32. {
  33. {&MEDIATYPE_MPEG2_PACK, &MEDIASUBTYPE_NULL},
  34. {&MEDIATYPE_MPEG2_PES, &MEDIASUBTYPE_NULL},
  35. };
  36. const AMOVIESETUP_PIN sudpPins[] =
  37. {
  38.     {L"Input", FALSE, FALSE, FALSE, FALSE, &CLSID_NULL, NULL, countof(sudPinTypesIn), sudPinTypesIn},
  39.     {L"Output", FALSE, TRUE, FALSE, FALSE, &CLSID_NULL, NULL, countof(sudPinTypesOut), sudPinTypesOut}
  40. };
  41. const AMOVIESETUP_FILTER sudFilter[] =
  42. {
  43. {&__uuidof(CDeCSSFilter), L"DeCSSFilter", MERIT_DO_NOT_USE, countof(sudpPins), sudpPins},
  44. };
  45. CFactoryTemplate g_Templates[] =
  46. {
  47.     {sudFilter[0].strName, sudFilter[0].clsID, CreateInstance<CDeCSSFilter>, NULL, &sudFilter[0]},
  48. };
  49. int g_cTemplates = countof(g_Templates);
  50. STDAPI DllRegisterServer()
  51. {
  52. return AMovieDllRegisterServer2(TRUE);
  53. }
  54. STDAPI DllUnregisterServer()
  55. {
  56. return AMovieDllRegisterServer2(FALSE);
  57. }
  58. extern "C" BOOL WINAPI DllEntryPoint(HINSTANCE, ULONG, LPVOID);
  59. BOOL APIENTRY DllMain(HANDLE hModule, DWORD dwReason, LPVOID lpReserved)
  60. {
  61.     return DllEntryPoint((HINSTANCE)hModule, dwReason, 0); // "DllMain" of the dshow baseclasses;
  62. }
  63. #endif
  64. //
  65. // CDeCSSFilter
  66. //
  67. class CKsPSInputPin : public CDeCSSInputPin
  68. {
  69. public:
  70.     CKsPSInputPin(TCHAR* pObjectName, CTransformFilter* pFilter, HRESULT* phr, LPWSTR pName)
  71. : CDeCSSInputPin(pObjectName, pFilter, phr, pName)
  72. {
  73. }
  74. // IKsPropertySet
  75.     STDMETHODIMP Set(REFGUID PropSet, ULONG Id, LPVOID InstanceData, ULONG InstanceLength, LPVOID PropertyData, ULONG DataLength)
  76. {
  77. if(CComQIPtr<IKsPropertySet> pKsPS = ((CDeCSSFilter*)m_pFilter)->m_pOutput->GetConnected())
  78. return pKsPS->Set(PropSet, Id, InstanceData, InstanceLength, PropertyData, DataLength);
  79. return E_NOTIMPL;
  80. }
  81.     STDMETHODIMP Get(REFGUID PropSet, ULONG Id, LPVOID InstanceData, ULONG InstanceLength, LPVOID PropertyData, ULONG DataLength, ULONG* pBytesReturned)
  82. {
  83. if(CComQIPtr<IKsPropertySet> pKsPS = ((CDeCSSFilter*)m_pFilter)->m_pOutput->GetConnected())
  84. return pKsPS->Get(PropSet, Id, InstanceData, InstanceLength, PropertyData, DataLength, pBytesReturned);
  85. return E_NOTIMPL;
  86. }
  87.     STDMETHODIMP QuerySupported(REFGUID PropSet, ULONG Id, ULONG* pTypeSupport)
  88. {
  89. if(CComQIPtr<IKsPropertySet> pKsPS = ((CDeCSSFilter*)m_pFilter)->m_pOutput->GetConnected())
  90. return pKsPS->QuerySupported(PropSet, Id, pTypeSupport);
  91. return E_NOTIMPL;
  92. }
  93. };
  94. CDeCSSFilter::CDeCSSFilter(LPUNKNOWN lpunk, HRESULT* phr) 
  95. : CTransformFilter(NAME("CDeCSSFilter"), lpunk, __uuidof(this))
  96. {
  97. if(phr) *phr = S_OK;
  98. if(!(m_pInput = new CKsPSInputPin(NAME("CKsPSInputPin"), this, phr, L"In"))) *phr = E_OUTOFMEMORY;
  99. if(FAILED(*phr)) return;
  100. if(!(m_pOutput = new CTransformOutputPin(NAME("CTransformOutputPin"), this, phr, L"Out"))) *phr = E_OUTOFMEMORY;
  101. if(FAILED(*phr))  {delete m_pInput, m_pInput = NULL; return;}
  102. }
  103. CDeCSSFilter::~CDeCSSFilter()
  104. {
  105. }
  106. HRESULT CDeCSSFilter::Transform(IMediaSample* pIn, IMediaSample* pOut)
  107. {
  108. AM_MEDIA_TYPE* pmt;
  109. if(SUCCEEDED(pIn->GetMediaType(&pmt)) && pmt)
  110. {
  111. CMediaType mt = *pmt;
  112. m_pInput->SetMediaType(&mt);
  113. mt.majortype = m_pOutput->CurrentMediaType().majortype;
  114. m_pOutput->SetMediaType(&mt);
  115. pOut->SetMediaType(&mt);
  116. DeleteMediaType(pmt);
  117. }
  118. BYTE* pDataIn = NULL;
  119. BYTE* pDataOut = NULL;
  120. pIn->GetPointer(&pDataIn);
  121. pOut->GetPointer(&pDataOut);
  122. long len = pIn->GetActualDataLength();
  123. long size = pOut->GetSize();
  124. if(len == 0 || pDataIn == NULL) // format changes do not carry any data
  125. {
  126. pOut->SetActualDataLength(0);
  127. return S_OK;
  128. }
  129. if(m_pOutput->CurrentMediaType().majortype == MEDIATYPE_MPEG2_PES)
  130. {
  131. if(*(DWORD*)pDataIn == 0xBA010000)
  132. {
  133. len -= 14; pDataIn += 14;
  134. if(int stuffing = (pDataIn[-1]&7)) {len -= stuffing; pDataIn += stuffing;}
  135. }
  136. if(len <= 0) return S_FALSE;
  137. if(*(DWORD*)pDataIn == 0xBB010000)
  138. {
  139. len -= 4; pDataIn += 4;
  140. int hdrlen = ((pDataIn[0]<<8)|pDataIn[1]) + 2;
  141. len -= hdrlen; pDataIn += hdrlen;
  142. }
  143. if(len <= 0) return S_FALSE;
  144. }
  145. if(!pDataIn || !pDataOut || len > size || len < 0) return S_FALSE;
  146. memcpy(pDataOut, pDataIn, min(len, size));
  147. pOut->SetActualDataLength(min(len, size));
  148. return S_OK;
  149. }
  150. HRESULT CDeCSSFilter::CheckInputType(const CMediaType* mtIn)
  151. {
  152. return mtIn->majortype == MEDIATYPE_DVD_ENCRYPTED_PACK
  153. ? S_OK 
  154. : VFW_E_TYPE_NOT_ACCEPTED;
  155. }
  156. HRESULT CDeCSSFilter::CheckTransform(const CMediaType* mtIn, const CMediaType* mtOut)
  157. {
  158. return SUCCEEDED(CheckInputType(mtIn))
  159. && mtOut->majortype == MEDIATYPE_MPEG2_PACK || mtOut->majortype == MEDIATYPE_MPEG2_PES
  160. ? S_OK 
  161. : VFW_E_TYPE_NOT_ACCEPTED;
  162. }
  163. HRESULT CDeCSSFilter::DecideBufferSize(IMemAllocator* pAllocator, ALLOCATOR_PROPERTIES* pProperties)
  164. {
  165. if(m_pInput->IsConnected() == FALSE) return E_UNEXPECTED;
  166. pProperties->cbAlign = 1;
  167. pProperties->cBuffers = 1;
  168. pProperties->cbBuffer = 2048;
  169. pProperties->cbPrefix = 0;
  170. HRESULT hr;
  171. ALLOCATOR_PROPERTIES Actual;
  172.     if(FAILED(hr = pAllocator->SetProperties(pProperties, &Actual))) 
  173. return hr;
  174.     return(pProperties->cBuffers > Actual.cBuffers || pProperties->cbBuffer > Actual.cbBuffer
  175. ? E_FAIL
  176. : NOERROR);
  177. }
  178. HRESULT CDeCSSFilter::GetMediaType(int iPosition, CMediaType* pmt)
  179. {
  180.     if(m_pInput->IsConnected() == FALSE) return E_UNEXPECTED;
  181. if(iPosition < 0) return E_INVALIDARG;
  182.     if(iPosition > 1) return VFW_S_NO_MORE_ITEMS;
  183. CopyMediaType(pmt, &m_pInput->CurrentMediaType());
  184. if(iPosition == 0) pmt->majortype = MEDIATYPE_MPEG2_PACK;
  185. if(iPosition == 1) pmt->majortype = MEDIATYPE_MPEG2_PES;
  186. return S_OK;
  187. }