DirectShow_基础与简单示例说明

来源:互联网 发布:普林科技 大数据 成都 编辑:程序博客网 时间:2024/05/01 14:45


DirectShow 是一个 Windows 平台上的流媒体框架,提供了高质量的多媒体流采集和回放功能。
Filter 实质是一个 COM 组件,所以学习开发 Filter 前你应该对 COM 相关知识有点了解。COM 组件的实质是实现了纯虚指针接口的 C++ 对象。
应用程序开发者只需要基本的 COM 组件知识:实例化COM组件、调用接口、管理接口的引用计数。Filter 开发者则需要更多。
选择一个基类,声明自己的类。
应该清楚这个 Filter 在整个 Filter Graph 的位置,这个 Filter 的输入是什么数据,输出是什么数据,有几个输入 Pin、几个输出 Pin 等,可以画出这个 Filter 的草图。
Win7: DirectShow SDK 做为 Windows SDK(GRMSDK_EN_DVD.iso) 的一部分,不再有单独的 DirectX SDK 包。
在 DirectShow 中,应用程序要实现功能就必须将这些 Filter 链接在一起,因而一个 Filter 的输出就变成了另一个 Filter 的输入。这一系列串在一起的 Filter 称为 Filter Graph。


使用 VS2008 建立 Filter 开发工程的过程如下:
(1) 新建 
Visual C++/Win32 项目,工程名如:FilterSample。点击“确定”进入下一步;
(2) 应用程序类型选择:DLL,一般不选择 MFC 和 ATL;
工程的建立完成,编译后可以先将输入文件改名。方法如下:
属性页/链接器/常规/输出文件,修改为:$(OutDir)/FilterSample.ax


DirectShow 必须用到以下头文件的库文件:
#include "streams.h"
#include "initguid.h"
strmbasd.lib
winmm.lib
uuid.lib
Quartz.lib   输出 AMGetErrorText 函数,如果不调用此函数,此库不是必需的。


FilterSample.def 文件的内容:
LIBRARY FilterSample.ax 
EXPORTS
; 需要定义的导出函数
DllMain PRIVATE
DllRegisterServer PRIVATE
DllUnregisterServer PRIVATE
; 在基类中已经定义的导出函数
DllGetClassObject PRIVATE
DllCanUnloadNow PRIVATE


dllmain.cpp 的代码如下:
// dllmain.cpp : 定义 DLL 应用程序的入口点。
#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);
}


编译时,可能会遇到如下问题:
1>e:\work\sourcecode\testcode\filtersample\dllmain.cpp(23) : error C3861: “AMovieDllRegisterServer2”: 找不到标识符
1>e:\work\sourcecode\testcode\filtersample\dllmain.cpp(27) : error C3861: “AMovieDllRegisterServer2”: 找不到标识符
编译错误, 一般是头文件没有包含。所以,包含: #include "streams.h"


1>dllmain.obj : error LNK2019: 无法解析的外部符号 _AMovieDllRegisterServer2@4,该符号在函数 _DllRegisterServer@0 中被引用
1>dllmain.obj : error LNK2019: 无法解析的外部符号 _DllEntryPoint@12,该符号在函数 _DllMain@12 中被引用
1>E:\Work\SourceCode\TestCode\FilterSample\Debug\FilterSample.dll : fatal error LNK1120: 2 个无法解析的外部命令
链接错误, 一般是库文件没有包含。所以,库输入中包含: strmbasd.lib


1>正在链接...
1>strmbasd.lib(wxdebug.obj) : error LNK2019: 无法解析的外部符号 __imp__timeGetTime@0,该符号在函数 "void __stdcall DbgInitialise(struct HINSTANCE__ *)" (?DbgInitialise@@YGXPAUHINSTANCE__@@@Z) 中被引用
1>strmbasd.lib(wxutil.obj) : error LNK2001: 无法解析的外部符号 __imp__timeGetTime@0
1>strmbasd.lib(wxutil.obj) : error LNK2019: 无法解析的外部符号 __imp__timeSetEvent@20,该符号在函数 "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:\Work\SourceCode\TestCode\FilterSample\Debug\FilterSample.dll : fatal error LNK1120: 4 个无法解析的外部命令
包含  winmm.lib, 可以解决 timeGetTime 链接错误的问题


增加头文件 FilterSample.h,其内容如下:
其中 GUID 的生成,需要用到下面的工具: Microsoft SDKs\Windows\v7.1\Bin\guidgen.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 的内容如下:
// FilterSample.cpp : 定义 DLL 应用程序的导出函数。
//


#include "stdafx.h"
#include "streams.h"
#include "initguid.h"


#include "FilterSample.h"


// Using this pointer in constructor
#pragma warning(disable:4355 4127)


//////////////////////////////////////////////////////////////////////////
// AMOVIESETUP_FILTER 描述一个 Filter
// AMOVIESETUP_PIN 描述 pin
// AMOVIESETUP_MEDIATYPE 描述数据类型
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;
}


编译时,可能遇到如下问题:
1>FilterSample.obj : error LNK2019: 无法解析的外部符号 "public: __thiscall CBaseFilter::CBaseFilter(wchar_t const *,struct IUnknown *,class CCritSec *,struct _GUID const &)" (??0CBaseFilter@@QAE@PB_WPAUIUnknown@@PAVCCritSec@@ABU_GUID@@@Z),该符号在函数 "public: __thiscall CFilterSample::CFilterSample(wchar_t *,struct IUnknown *,long *)" (??0CFilterSample@@QAE@PA_WPAUIUnknown@@PAJ@Z) 中被引用
1>E:\Work\SourceCode\TestCode\FilterSample\Debug\FilterSample.dll : fatal error LNK1120: 1 个无法解析的外部命令
解决方法: 属性页 常规->字符集 修改为“使用多字节字符集”


编译通过后,如何验证此 Filter 呢?
regsvr32 FilterSample.ax
regsvr32 提示错误:
模块“FilterSample.ax”已加载,但找不到入口点 DllRegisterServer。
请确保“FilterSample.ax”为有效的 DLL 或 OCX 文件,然后重试。
出现此错误的原因,是 FilterSample.def 未链接到工程中。在工程属性页,“链接器”/“输入”/“模块定义文件” 中输入: FilterSample.def 后,重新编译。
然后在命令行执行: regsvr32 FilterSample.ax,则提示成功。


如何确认成功了呢?可通过 Microsoft SDKs\Windows\v7.1\Bin\graphedt.exe 来查看。方法如下:
点击 graphedt.exe 的 Graph 菜单,选择 Insert Filters...。在弹出的对话框: Which filters do you want to insert? 中,选择 Directshow Filters/Filter Sample Test Lib。
这个名字看起来是不是很熟悉?看看上面的代码中是不是有这样的字符串,呵呵...。如果还不确认,请看 Filter Sample Test Lib 的 GUID 和 文件名分别是:
@device:sw:{083863F1-70DE-11D0-BD40-00A0C911CE86}\{33B57142-BD07-4A77-AE91-A8F6C24A8F40} 和 E:\Filters\FilterSample.ax(具体的目录与执行 regsvr32 时 FilterSample.ax 所在的路径相关)。
0 0
原创粉丝点击