1、选择所要创建的filter的用途,据此来选择基类。基类可以从CTransformFilter、CTransInPlaceFilter、CVideoTransformFilter和CBaseFilter中来选取。 (1) CTransInPlaceFilter提供了本地处理Sample的机制(Sample可以认为是存储一个视频帧的结构),当一个trans-in-place filter收到一个sample时,你可以通过重载它的Transform()函数来修改其中的数据,trans-in-place filter会在Transform()函数执行完后直接把这个sample传递给下一个filter。 (2) CTransformFilter完成的功能与CTransInPlaceFilter一样,它们的区别就是CTransformFilter总是把上游filter传递过来的sample复制一份,并把复制后的sample传递给下一个filter。当然,你可以通过重载Transform()函数来控制这个过程,包括修改其中的数据(这也是自己写filter的原因)。 (3)CVideoTransformFilter与CTransformFilter一样,只是多加了质量控制功能。 (4)以上三个filter都继承于CBaseFilter,所以如果想对filter进行更多的控制,就要直接从CBaseFilter来继承,但是所要做的工作也最多。在这个例子中,我选择CTransformFilter,因为CTransInPlaceFilter太简单了,dx9sdk中的例子NullNull就是一个完整的CTransInPlaceFilter的框架,并且只是一个框架,什么工作也没有做,如果要用的话直接修改就可以用了。 2、在vc中选择win32 dll,创建一个dll,名字随便取,这里我取SplitFilter,选择空dll。 3、然后,创建一个类CSplitFilter,继承自CTransformFilter,当然要选择public方式。 4、为filter生成一个CLSID,可以使用Guidgen,它的用法是在命令行中打Guidgen,然后回车,Guidgen就执行了,单击New GUID就会生成一个新的GUID;单击Copy就可以把新生成的GUID复制到剪贴板上。然后在SplitFilter.h文件上粘贴进来,最后是这个样子: //////////////////////////////////////////////////////////////////////// GUID////////////////////////////////////////////////////////////////////////{3DCD790F-B7A0-429a-B9E1-3CE3255D8D1C} DEFINE_GUID(<<name>>, 0x3dcd790f, 0xb7a0, 0x429a, 0xb9, 0xe1, 0x3c, 0xe3, 0x25, 0x5d, 0x8d, 0x1c); 然后把其中的<<name>>换成自己设置的名字,如下: // {3DCD790F-B7A0-429a-B9E1-3CE3255D8D1C} DEFINE_GUID(CLSID_SplitFilter, 0x3dcd790f, 0xb7a0, 0x429a, 0xb9, 0xe1, 0x3c, 0xe3, 0x25, 0x5d, 0x8d, 0x1c); 然后在SplitFilter.cpp中加入#include <initguid.h>再来修改构造函数,形式如下: CSplitFilter::CSplitFilter() : CTransformFilter(NAME("SplitFilter"), 0, CLSID_SplitFilter){} 5、处理媒体类型 首先要明白一点,两个filter连接,也就是两个filter的输出pin和输入pin在进行连接,这个工作是由输出pin发起的,由输入pin来检查媒体类型是否匹配,并决定是否接受这个连接。在CTransformFilter中协商媒体类型的工作是由CTransformFilter来完成的,这个工作本来是应该由pin来完成的,但是CTransformFilter中的pin只是简单的调用CTransformFilter中相应的函数而已,所以我们所有的工作只是重载CTransformFilter中的三个虚函数而已: (1)实现CheckInputType(const CMediaType *mtIn)(不要问我如何添加这个函数,如果这个都不会的话趁早别看dshow了,赶紧去看vc的书去)。这个函数是由输入pin来调用,当上游输出pin要来进行连接的时候,我们的filter的输入pin就会调用这个函数来检查是否支持上游输出pin的媒体类型。其中的CMediaType类是对AM_MEDIA_TYPE结构的封装,AM_MEDIA_TYPE包含了有关的媒体类型,具体可查阅dxsdk文档。因为我这里只是写一个框架来为大家演示,功能只是传递sample,并不对数据有任何的修改,所以任何的格式都可以,所以不管什么格式我们都应该返回ok,实际函数如下: HRESULT CSplitFilter::CheckInputType(const CMediaType *mtIn){ // Everything is good. return S_OK;} (2)实现GetMediaType(int iPosition, CMediaType *pMediaType)前面我们已经讲了输入pin接受连接的时候用的函数,那么这个函数呢就是输入pin进行连接的时候用的,输出pin进行连接的时候首先要有一个支持的媒体类型列表,这个函数就是来生成这个列表的。实际上我们也可以直接返回一个ok了事,但是这样做有点太不负责任,我们虽然什么工作都不做,但是还是要把例行的检查做完了,这样老板(filter graph)看到了也会很满意,你们是不是也很赞同?首先,我们要确定输入pin是否已经连接了,如果输入pin没有连接,那我们和下面的filter连接了也没有意义,因为上游没有数据传过来,我们拿什么给后面的filter?画饼是不能充饥的:) ASSERT(m_pInput->IsConnected()); 然后,看看pin的位置是否正确if (iPosition < 0) { return E_INVALIDARG;} 这个函数最后的结果是这样的: HRESULT CSplitFilter::GetMediaType(int iPosition, CMediaType *pMediaType){ // Is the input pin connected if(m_pInput->IsConnected() == FALSE) { return E_UNEXPECTED; } // This should never happen if(iPosition < 0) { return E_INVALIDARG; } // Do we have more items to offer if(iPosition > 0) { return VFW_S_NO_MORE_ITEMS; } CheckPointer(pMediaType,E_POINTER); *pMediaType = m_pInput->CurrentMediaType(); return NOERROR;} (3)实现CheckTransform(const CMediaType *mtIn, const CMediaType *mtOut)虽然输入pin和输出pin的媒体类型都已经协商过了,但是我们的filter是否能够完成两中媒体类型的转换还是个未知数,那么就要调用CheckTransform来判断我们的filter是否能够支持这两个媒体类型的转换。我在这里的选择是直接返回ok。呵呵:) 6、设置分配器虽然我们已经完成了媒体类型的匹配,但filter的连接还没有完成,我们的pin还必须为连接选择allocator,设置allocator的属性,比如缓冲区的大小和数量等。输入pin我们可以不用管它,因为默认的情况下它只需要同意输出pin提供的allocator就可以了,但是输出pin的allocator就要我们自己来搞定了。如果后面的filter提供了一个allocator,输出pin可以使用这个allocator,否则就要建立一个新的allocator。这就要用到CTransformFilter的DecideBufferSize(IMemAllocator *pAlloc, ALLOCATOR_PROPERTIES *pProp),pAlloc是一个allocator的指针,pProp是一个ALLOCATOR_PROPERTIES结构体,代表后面的filter所需要的allocator的属性。我们可以在这个函数中根据我们自己的filter的需要和后面的filter的需要来设置allocator的各项属性,设置allocator的属性可以用IMemAllocator::SetProperties函数。这里面最重要的就是设置buffer的大小,一般是从输入pin的媒体类型的大小来决定,如果从这里没有得到固定的大小的话,我们就要猜buffer的大小了。最后的函数是这样的: HRESULT CSplitFilter::DecideBufferSize(IMemAllocator *pAlloc, ALLOCATOR_PROPERTIES *pProperties){ CheckPointer(pAlloc,E_POINTER); CheckPointer(pProperties,E_POINTER); // Is the input pin connected if(m_pInput->IsConnected() == FALSE) { return E_UNEXPECTED; } HRESULT hr = NOERROR; pProperties->cBuffers = 1; pProperties->cbBuffer = m_pInput->CurrentMediaType().GetSampleSize(); ASSERT(pProperties->cbBuffer); // If we don't have fixed sized samples we must guess some size if(!m_pInput->CurrentMediaType().bFixedSizeSamples) { if(pProperties->cbBuffer < 100000) { // nothing more than a guess!! pProperties->cbBuffer = 100000; } } // Ask the allocator to reserve us some sample memory, NOTE the function // can succeed (that is return NOERROR) but still not have allocated the // memory that we requested, so we must check we got whatever we wanted ALLOCATOR_PROPERTIES Actual; hr = pAlloc->SetProperties(pProperties,&Actual); if(FAILED(hr)) { return hr; } ASSERT(Actual.cBuffers == 1); if(pProperties->cBuffers > Actual.cBuffers || pProperties->cbBuffer > Actual.cbBuffer) { return E_FAIL; } return NOERROR;} 7、处理数据终于可以开始处理数据了,我们重载Transform(IMediaSample *pSource, IMediaSample *pDest)来实现对数据的处理,从字面上都可以看出这两个sample的意义,因为这里我们什么工作都不做,所以我们只需要把源sample复制到目的sample就可以了。这个函数如下: HRESULT CSplitFilter::Transform(IMediaSample *pIn, IMediaSample *pOut){ HRESULT hr = Copy(pIn, pOut); return hr;} 其中用到了功能函数copy,其代码如下: HRESULT CSplitFilter::Copy(IMediaSample *pSource, IMediaSample *pDest) const{ CheckPointer(pSource,E_POINTER); CheckPointer(pDest,E_POINTER); // Copy the sample data BYTE *pSourceBuffer, *pDestBuffer; long lSourceSize = pSource->GetActualDataLength(); #ifdef DEBUG long lDestSize = pDest->GetSize(); ASSERT(lDestSize >= lSourceSize); #endif pSource->GetPointer(&pSourceBuffer); pDest->GetPointer(&pDestBuffer); CopyMemory((PVOID) pDestBuffer,(PVOID) pSourceBuffer,lSourceSize); // Copy the sample times REFERENCE_TIME TimeStart, TimeEnd; if(NOERROR == pSource->GetTime(&TimeStart, &TimeEnd)) { pDest->SetTime(&TimeStart, &TimeEnd); } LONGLONG MediaStart, MediaEnd; if(pSource->GetMediaTime(&MediaStart,&MediaEnd) == NOERROR) { pDest->SetMediaTime(&MediaStart,&MediaEnd); } // Copy the Sync point property HRESULT hr = pSource->IsSyncPoint(); if(hr == S_OK) { pDest->SetSyncPoint(TRUE); } else if(hr == S_FALSE) { pDest->SetSyncPoint(FALSE); } else { // an unexpected error has occured... return E_UNEXPECTED; } // Copy the media type AM_MEDIA_TYPE *pMediaType; pSource->GetMediaType(&pMediaType); pDest->SetMediaType(pMediaType); DeleteMediaType(pMediaType); // Copy the preroll property hr = pSource->IsPreroll(); if(hr == S_OK) { pDest->SetPreroll(TRUE); } else if(hr == S_FALSE) { pDest->SetPreroll(FALSE); } else { // an unexpected error has occured... return E_UNEXPECTED; } // Copy the discontinuity property hr = pSource->IsDiscontinuity(); if(hr == S_OK) { pDest->SetDiscontinuity(TRUE); } else if(hr == S_FALSE) { pDest->SetDiscontinuity(FALSE); } else { // an unexpected error has occured... return E_UNEXPECTED; } // Copy the actual data length long lDataLength = pSource->GetActualDataLength(); pDest->SetActualDataLength(lDataLength); return NOERROR;} 8、增加对com的支持因为filter是一个com组件,所以要符合com规范。又因为CTransformFilter是从CUnknown继承而来的,所以不用再自己实现AddRef 和 Release,如果我们要是有自己定义的接口的话,就要自己实现QueryInterface,幸好我们这个filter不实现自己定义的接口,就不用那么麻烦了,呵呵。好了,闲话少续,开始了:首先,创建一个静态的方法来返回一个我们的filter的实例,这个方法的名字可以任意指定,但是一般都叫做CreateInstance,那我们也就取这个名字吧,这个函数的参数如下(内含一般代码): CUnknown * WINAPI CSplitFilter::CreateInstance(LPUNKNOWN pUnk, HRESULT *pHr){ CSplitFilter *pFilter = new CSplitFilter(); if (pFilter== NULL) { *pHr = E_OUTOFMEMORY; } return pFilter; } 然后,声明一个全局的CFactoryTemplate类数组,取名g_Templates,每一个CFactoryTemplate实例包含一个filter或者我们定义的接口的注册信息,因为我们这里就只有一个filter,也没有自定义接口,所以数组中就只有一项内容。CFactoryTemplate g_Templates[] = { { L"SplitFilter", &CLSID_SplitFilter, CSplitFilter::CreateInstance, NULL, NULL }};int g_cTemplates = sizeof(g_Templates) / sizeof(g_Templates[0]); 最后,实现dll的注册函数 STDAPI DllRegisterServer(){ return AMovieDllRegisterServer2( TRUE ); } STDAPI DllUnregisterServer(){ return AMovieDllRegisterServer2( FALSE ); } 再加上#include <Streams.h>引入静态库Strmbasd.lib Msvcrtd.lib Winmm.lib最后再建立其导出文件SplitFilter.def,内容如下: LIBRARY SplitFilter.ax EXPORTS DllMain PRIVATE DllGetClassObject PRIVATE DllCanUnloadNow PRIVATE DllRegisterServer PRIVATE DllUnregisterServer PRIVATE 这样我们就可以建立一个最基本的filter的框架了。 当然,如果你要播放高清视频,如果在分配器上分配的内存不够的话你可以加大内存,也就是刚开始猜内存需要多大的时候尽量大一点,我反正在测试2732×768的时候报错,但是我把分配的内存加了个0就ok了。 |