DirectShow Filter 基礎(chǔ)與簡(jiǎn)單的示例程序
DirectShow 是一個(gè) Windows 平臺(tái)上的流媒體框架,提供了高質(zhì)量的多媒體流采集和回放功能。
Filter 實(shí)質(zhì)是一個(gè) COM 組件,所以學(xué)習(xí)開(kāi)發(fā) Filter 前你應(yīng)該對(duì) COM 相關(guān)知識(shí)有點(diǎn)了解。COM 組件的實(shí)質(zhì)是實(shí)現(xiàn)了純虛指針接口的 C++ 對(duì)象。
應(yīng)用程序開(kāi)發(fā)者只需要基本的 COM 組件知識(shí):實(shí)例化COM組件、調(diào)用接口、管理接口的引用計(jì)數(shù)。Filter 開(kāi)發(fā)者則需要更多。
選擇一個(gè)基類(lèi),聲明自己的類(lèi)。
應(yīng)該清楚這個(gè) Filter 在整個(gè) Filter Graph 的位置,這個(gè) Filter 的輸入是什么數(shù)據(jù),輸出是什么數(shù)據(jù),有幾個(gè)輸入 Pin、幾個(gè)輸出 Pin 等,可以畫(huà)出這個(gè) Filter 的草圖。
Win7: DirectShow SDK 做為 Windows SDK(GRMSDK_EN_DVD.iso) 的一部分,不再有單獨(dú)的 DirectX SDK 包。
在 DirectShow 中,應(yīng)用程序要實(shí)現(xiàn)功能就必須將這些 Filter 鏈接在一起,因而一個(gè) Filter 的輸出就變成了另一個(gè) Filter 的輸入。這一系列串在一起的 Filter 稱(chēng)為 Filter Graph。
使用 VS2008 建立 Filter 開(kāi)發(fā)工程的過(guò)程如下:
(1) 新建?
Visual C++/Win32 項(xiàng)目,工程名如:FilterSample。點(diǎn)擊“確定”進(jìn)入下一步;
(2) 應(yīng)用程序類(lèi)型選擇:DLL,一般不選擇 MFC 和 ATL;
工程的建立完成,編譯后可以先將輸入文件改名。方法如下:
屬性頁(yè)/鏈接器/常規(guī)/輸出文件,修改為:$(OutDir)/FilterSample.ax
DirectShow 必須用到以下頭文件的庫(kù)文件:
#include?"streams.h"
#include?"initguid.h"
strmbasd.lib
winmm.lib
uuid.lib
Quartz.lib???輸出?AMGetErrorText?函數(shù),如果不調(diào)用此函數(shù),此庫(kù)不是必需的。
FilterSample.def?文件的內(nèi)容:
LIBRARY?FilterSample.ax?
EXPORTS
;?需要定義的導(dǎo)出函數(shù)
DllMain?PRIVATE
DllRegisterServer?PRIVATE
DllUnregisterServer?PRIVATE
;?在基類(lèi)中已經(jīng)定義的導(dǎo)出函數(shù)
DllGetClassObject?PRIVATE
DllCanUnloadNow?PRIVATE
dllmain.cpp?的代碼如下:
//?dllmain.cpp?:?定義?DLL?應(yīng)用程序的入口點(diǎn)。
#include?"stdafx.h"
#include?"streams.h"
//?BOOL?APIENTRY?DllMain(?HMODULE?hModule,
//????????????????????????DWORD??ul_reason_for_call,
//????????????????????????LPVOID?lpReserved
//???????????)
//?{
//??switch?(ul_reason_for_call)
//??{
//??case?DLL_PROCESS_ATTACH:
//??case?DLL_THREAD_ATTACH:
//??case?DLL_THREAD_DETACH:
//??case?DLL_PROCESS_DETACH:
//????break;
//??}
//??return?TRUE;
//?}
STDAPI?DllRegisterServer()
{
??return?AMovieDllRegisterServer2(TRUE);
}
STDAPI?DllUnregisterServer()
{
??return?AMovieDllRegisterServer2(FALSE);
}
extern?"C"?BOOL?WINAPI?DllEntryPoint(HINSTANCE,?ULONG,?LPVOID);
BOOL?APIENTRY?DllMain(HANDLE?hModule,?DWORD??dwReason,???LPVOID?lpReserved)
{
??return?DllEntryPoint((HINSTANCE)(hModule),?dwReason,?lpReserved);
}
編譯時(shí),可能會(huì)遇到如下問(wèn)題:
1>e:worksourcecodetestcodefiltersampledllmain.cpp(23) : error C3861: “AMovieDllRegisterServer2”: 找不到標(biāo)識(shí)符
1>e:worksourcecodetestcodefiltersampledllmain.cpp(27) : error C3861: “AMovieDllRegisterServer2”: 找不到標(biāo)識(shí)符
編譯錯(cuò)誤, 一般是頭文件沒(méi)有包含。所以,包含: #include "streams.h"
1>dllmain.obj : error LNK2019: 無(wú)法解析的外部符號(hào) _AMovieDllRegisterServer2@4,該符號(hào)在函數(shù) _DllRegisterServer@0 中被引用
1>dllmain.obj : error LNK2019: 無(wú)法解析的外部符號(hào) _DllEntryPoint@12,該符號(hào)在函數(shù) _DllMain@12 中被引用
1>E:WorkSourceCodeTestCodeFilterSampleDebugFilterSample.dll : fatal error LNK1120: 2 個(gè)無(wú)法解析的外部命令
鏈接錯(cuò)誤, 一般是庫(kù)文件沒(méi)有包含。所以,庫(kù)輸入中包含: strmbasd.lib
1>正在鏈接...
1>strmbasd.lib(wxdebug.obj) : error LNK2019: 無(wú)法解析的外部符號(hào) __imp__timeGetTime@0,該符號(hào)在函數(shù) "void __stdcall DbgInitialise(struct HINSTANCE__ *)" (?DbgInitialise@@YGXPAUHINSTANCE__@@@Z) 中被引用
1>strmbasd.lib(wxutil.obj) : error LNK2001: 無(wú)法解析的外部符號(hào) __imp__timeGetTime@0
1>strmbasd.lib(wxutil.obj) : error LNK2019: 無(wú)法解析的外部符號(hào) __imp__timeSetEvent@20,該符號(hào)在函數(shù) "unsigned int __cdecl CompatibleTimeSetEvent(unsigned int,unsigned int,void (__stdcall*)(unsigned int,unsigned
int,unsigned long,unsigned long,unsigned long),unsigned long,unsigned int)" (?CompatibleTimeSetEvent@@YAIIIP6GXIIKKK@ZKI@Z) 中被引用
1>E:WorkSourceCodeTestCodeFilterSampleDebugFilterSample.dll : fatal error LNK1120: 4 個(gè)無(wú)法解析的外部命令
包含 ?winmm.lib, 可以解決 timeGetTime 鏈接錯(cuò)誤的問(wèn)題
增加頭文件 FilterSample.h,其內(nèi)容如下:
其中 GUID 的生成,需要用到下面的工具: Microsoft SDKsWindowsv7.1Binguidgen.exe
#ifndef?_FILTER_SAMPLE_H_
#define?_FILTER_SAMPLE_H_
//?{33B57142-BD07-4a77-AE91-A8F6C24A8F40}
DEFINE_GUID(CLSID_FilterSample,?
????0x33b57142,?0xbd07,?0x4a77,?0xae,?0x91,?0xa8,?0xf6,?0xc2,?0x4a,?0x8f,?0x40);
class?CFilterSample:?public?CCritSec,?public?CBaseFilter
{
public:
??CFilterSample(TCHAR?*pName,LPUNKNOWN?pUnk,HRESULT?*hr);
??virtual?~CFilterSample();?
??static?CUnknown?*?WINAPI?CreateInstance(LPUNKNOWN?pUnk,?HRESULT?*phr);?
??CBasePin?*GetPin(int?n);
??int?GetPinCount();??
};
#endif
FilterSample.cpp 的內(nèi)容如下:
//?FilterSample.cpp?:?定義?DLL?應(yīng)用程序的導(dǎo)出函數(shù)。
//
#include?"stdafx.h"
#include?"streams.h"
#include?"initguid.h"
#include?"FilterSample.h"
//?Using?this?pointer?in?constructor
#pragma?warning(disable:4355?4127)
//////////////////////////////////////////////////////////////////////////
//?AMOVIESETUP_FILTER?描述一個(gè)?Filter
//?AMOVIESETUP_PIN?描述?pin
//?AMOVIESETUP_MEDIATYPE?描述數(shù)據(jù)類(lèi)型
const?AMOVIESETUP_MEDIATYPE?sudPinTypes?=
{
??&MEDIATYPE_NULL,?????????//?Major?CLSID
??&MEDIASUBTYPE_NULL???????//?Minor?type
};
const?AMOVIESETUP_PIN?psudPins[]?=
{
??{?
????L"Input",?????????????//?Pin's?string?name
????FALSE,????????????????//?Is?it?rendered
????FALSE,????????????????//?Is?it?an?output
????FALSE,????????????????//?Allowed?none
????FALSE,????????????????//?Allowed?many
????&CLSID_NULL,??????????//?Connects?to?filter
????L"Output",????????????//?Connects?to?pin
????1,????????????????????//?Number?of?types
????&sudPinTypes??????????//?Pin?information
??},
??{
????L"Output",????????????//?Pin's?string?name
????FALSE,????????????????//?Is?it?rendered
????TRUE,?????????????????//?Is?it?an?output
????FALSE,????????????????//?Allowed?none
????FALSE,????????????????//?Allowed?many
????&CLSID_NULL,??????????//?Connects?to?filter
????L"Input",?????????????//?Connects?to?pin
????1,????????????????????//?Number?of?types
????&sudPinTypes??????????//?Pin?information
??}
};
const?AMOVIESETUP_FILTER?sudInfTee?=
{
??&CLSID_FilterSample,???????????//?CLSID?of?filter
??L"Filter?Sample?Test?Lib",?????//?Filter's?name
??MERIT_DO_NOT_USE,??????????????//?Filter?merit
??2,?????????????????????????????//?Number?of?pins
??psudPins???????????????????????//?Pin?information
};
//////////////////////////////////////////////////////////////////////////
CFactoryTemplate?g_Templates[1]?=?
{
??{
????L"Filter?Sample",????????????????//?Name
????&CLSID_FilterSample,?????????????//?CLSID
????CFilterSample::CreateInstance,???//?Method?to?create?an?instance?of?MyComponent
????NULL,????????????????????????????//?Initialization?function
????&sudInfTee???????????????????????//?Set-up?information?(for?filters)
??}
};
int?g_cTemplates?=?sizeof(g_Templates)?/?sizeof(g_Templates[0]);
CFilterSample::CFilterSample(TCHAR?*pName,LPUNKNOWN?pUnk,HRESULT?*hr)
??:CBaseFilter(NAME("Filter?Sample"),?pUnk,?this,?CLSID_FilterSample)
{
}
CFilterSample::~CFilterSample()
{
}?
CBasePin?*?CFilterSample::GetPin(int?n)
{
??return?NULL;
}
int?CFilterSample::GetPinCount()
{
??return?0;
}
CUnknown?*?WINAPI?CFilterSample::CreateInstance(LPUNKNOWN?pUnk,?HRESULT?*pHr)
{
??CFilterSample?*pFilter?=?new?CFilterSample(NAME("Filter?Sample"),?pUnk,?pHr);
??if?(pFilter==?NULL)
??{
????*pHr?=?E_OUTOFMEMORY;
??}
??return?pFilter;
}
編譯時(shí),可能遇到如下問(wèn)題:
1>FilterSample.obj : error LNK2019: 無(wú)法解析的外部符號(hào) "public: __thiscall CBaseFilter::CBaseFilter(wchar_t const *,struct IUnknown *,class CCritSec *,struct _GUID const &)" (??0CBaseFilter@@QAE@PB_WPAUIUnknown@@PAVCCritSec@@ABU_GUID@@@Z),該符號(hào)在函數(shù)
"public: __thiscall CFilterSample::CFilterSample(wchar_t *,struct IUnknown *,long *)" (??0CFilterSample@@QAE@PA_WPAUIUnknown@@PAJ@Z) 中被引用
1>E:WorkSourceCodeTestCodeFilterSampleDebugFilterSample.dll : fatal error LNK1120: 1 個(gè)無(wú)法解析的外部命令
解決方法: 屬性頁(yè) 常規(guī)->字符集 修改為“使用多字節(jié)字符集”
編譯通過(guò)后,如何驗(yàn)證此 Filter 呢?
regsvr32 FilterSample.ax
regsvr32 提示錯(cuò)誤:
模塊“FilterSample.ax”已加載,但找不到入口點(diǎn) DllRegisterServer。
請(qǐng)確保“FilterSample.ax”為有效的 DLL 或 OCX 文件,然后重試。
出現(xiàn)此錯(cuò)誤的原因,是 FilterSample.def 未鏈接到工程中。在工程屬性頁(yè),“鏈接器”/“輸入”/“模塊定義文件” 中輸入: FilterSample.def 后,重新編譯。
然后在命令行執(zhí)行: regsvr32 FilterSample.ax,則提示成功。
如何確認(rèn)成功了呢?可通過(guò) Microsoft SDKsWindowsv7.1Bingraphedt.exe 來(lái)查看。方法如下:
點(diǎn)擊 graphedt.exe 的 Graph 菜單,選擇 Insert Filters...。在彈出的對(duì)話(huà)框: Which filters do you want to insert? 中,選擇 Directshow Filters/Filter Sample Test Lib。
這個(gè)名字看起來(lái)是不是很熟悉?看看上面的代碼中是不是有這樣的字符串,呵呵...。如果還不確認(rèn),請(qǐng)看 Filter Sample Test Lib 的 GUID 和 文件名分別是:
@device:sw:{083863F1-70DE-11D0-BD40-00A0C911CE86}{33B57142-BD07-4A77-AE91-A8F6C24A8F40} 和 E:FiltersFilterSample.ax(具體的目錄與執(zhí)行 regsvr32 時(shí) FilterSample.ax 所在的路徑相關(guān))。





