创建一个directshow应用程序一

原创 2014年12月26日 09:28:00

创建dshow应用程序一般有三个步骤:

1.创建一个Filter Graph Manager组件。

IGraphBuilder * pGraph = NULL;

HRESULT hr = CoCreateInstance(CLSID_FilterGraph, NULL,

CLSCTX_INPROC_SERVER, IID_IGraphBuilder, (void**)&pGraph);,

2.根据实际的应用,创建一个filter链,比如播放一个本地文件,最简单快速的代码如下:

hr = pGraph->RenderFile(L"path", NULL);

3.调用Filter Graph Manager上的各个接口进行控制,并且完成Filter Graph Manager与应用程序的交互,比如调用IMediaControl接口方法控制Filter Graph的状态转换。

IMediaControl *pControl = NULL;

hr = pGraph->QueryInterface(IID_IMediaControl, (void**)pControl);

hr = pControl->Run();


通用的Filter Graph构建技术:

1.加入一个指定CLSID的Filter

给定了一个Filter的CLSID,就可以调用CoCreateInstance来创建它。并使用IFilterGraph::AddFilter接口将其加入到Filter Graph中。

代码如下:

HRESULT AddFilterByCLSID(
    IGraphBuilder *pGraph,  // Pointer to the Filter Graph Manager.
    const GUID& clsid,      // CLSID of the filter to create.
    LPCWSTR wszName,        // A name for the filter.
    IBaseFilter **ppF)      // Receives a pointer to the filter.
{
    if (!pGraph || ! ppF) return E_POINTER;
    *ppF = 0;
    IBaseFilter *pF = 0;
    HRESULT hr = CoCreateInstance(clsid, 0, CLSCTX_INPROC_SERVER,
        IID_IBaseFilter, reinterpret_cast<void**>(&pF));
    if (SUCCEEDED(hr))
    {
        hr = pGraph->AddFilter(pF, wszName);
        if (SUCCEEDED(hr))
            *ppF = pF;
        else
            pF->Release();
    }
    return hr;
}



要在Filter Graph中加入一个AVI Mux Filter

IBaseFilter *pMux;

hr = AddFilterByCLSID(pGraph, CLSID_AviDest, L"AVI Mux", &pMux);

if(SUCCEEDED(hr))

{

/*...*/

pMux->Release();

}


2.得到Filter上的未连接的Pin

当在程序中连接filter的时候,首先要取得filter上未连接的Pin或者输出Pin。方法是,枚举Filter上的所有的Pin,调用IPin::QueryDirection查询Pin的方向。然后调用IPin::ConnectedTo查看Pin的连接状态。

HRESULT GetUnconnectedPin(
    IBaseFilter *pFilter,   // Pointer to the filter.
    PIN_DIRECTION PinDir,   // Direction of the pin to find.
    IPin **ppPin)           // Receives a pointer to the pin.
{
    *ppPin = 0;
    IEnumPins *pEnum = 0;
    IPin *pPin = 0;
    HRESULT hr = pFilter->EnumPins(&pEnum);
    if (FAILED(hr))
    {
        return hr;
    }
    while (pEnum->Next(1, &pPin, NULL) == S_OK)
    {
        PIN_DIRECTION ThisPinDir;
        pPin->QueryDirection(&ThisPinDir);
        if (ThisPinDir == PinDir)
        {
            IPin *pTmp = 0;
            hr = pPin->ConnectedTo(&pTmp);
            if (SUCCEEDED(hr))  // Already connected, not the pin we want.
            {
                pTmp->Release();
            }
            else  // Unconnected, this is the pin we want.
            {
                pEnum->Release();
                *ppPin = pPin;
                return S_OK;
            }
        }
        pPin->Release();
    }
    pEnum->Release();
    // Did not find a matching pin.
    return E_FAIL;
}


比如现在要得到一个未连接的输出Pin

IPin *pOut = NULL;

HRESULT hr = GetUnConnectedPin(pFilter, PINDIR_OUTPUT, &pOut);

if(SUCCEEDED(hr))

{

/*...*/

pOut->Release();

}


3.连接两个Filter

Filter Graph中连接两个Filter的函数有IFilterGraph::ConnectDirect和IFilterGraph::Connect

HRESULT ConnectFilters(
    IGraphBuilder *pGraph, // Filter Graph Manager.
    IPin *pOut,            // Output pin on the upstream filter.
    IBaseFilter *pDest)    // Downstream filter.
{
    if ((pGraph == NULL) || (pOut == NULL) || (pDest == NULL))
    {
        return E_POINTER;
    }
#ifdef debug
        PIN_DIRECTION PinDir;
        pOut->QueryDirection(&PinDir);
        _ASSERTE(PinDir == PINDIR_OUTPUT);
#endif

    // Find an input pin on the downstream filter.
    IPin *pIn = 0;
    HRESULT hr = GetUnconnectedPin(pDest, PINDIR_INPUT, &pIn);
    if (FAILED(hr))
    {
        return hr;
    }
    // Try to connect them.
    hr = pGraph->Connect(pOut, pIn);
    pIn->Release();
    return hr;
}

不同参数的ConnectFilters 的函数重载形式:

HRESULT ConnectFilters(
    IGraphBuilder *pGraph, 
    IBaseFilter *pSrc, 
    IBaseFilter *pDest)
{
    if ((pGraph == NULL) || (pSrc == NULL) || (pDest == NULL))
    {
        return E_POINTER;
    }

    // Find an output pin on the first filter.
    IPin *pOut = 0;
    HRESULT hr = GetUnconnectedPin(pSrc, PINDIR_OUTPUT, &pOut);
    if (FAILED(hr)) 
    {
        return hr;
    }
    hr = ConnectFilters(pGraph, pOut, pDest);
    pOut->Release();
    return hr;
}
例如现在要加入一个AVI Mux Filter 和 一个File Writer Filter,然后将他们相连起来。代码如下:
IBaseFilter *pMux, *pWrite;

hr = AddFilterByCLSID(pGraph, CLSID_AviDest, L"AVI Mux", &pMux);

if(SUCCEEDED(hr))

{

hr = AddFilterByCLSID(pGraph, CLSID_FileWriter, L"File Writer", &pWrite);

if(SUCCEEDED(hr))

{

hr = ConnectFilters(pGraph, pMux, pWrite);

pWrite->Release();

}

pMux->Release();

}


4.查找Filter和Pin上的接口


我们需要枚举Graph中的所有的Filter,或者枚举Filter上所有的Pin,一个一个的找接口。

//查找Filter上实现的某个接口

HRESULT FindFilterInterface(
    IGraphBuilder *pGraph, // Pointer to the Filter Graph Manager.
    REFGUID iid,           // IID of the interface to retrieve.
    void **ppUnk)          // Receives the interface pointer.
{
    if (!pGraph || !ppUnk) return E_POINTER;

    HRESULT hr = E_FAIL;
    IEnumFilters *pEnum = NULL;
    IBaseFilter *pF = NULL;
    if (FAILED(pGraph->EnumFilters(&pEnum)))
    {
        return E_FAIL;
    }
    // Query every filter for the interface.
    while (S_OK == pEnum->Next(1, &pF, 0))
    {
        hr = pF->QueryInterface(iid, ppUnk);
        pF->Release();
        if (SUCCEEDED(hr))
        {
            break;
        }
    }
    pEnum->Release();
    return hr;
}

//查找给定filter的Pin上实现的某个接口

HRESULT FindPinInterface(
    IBaseFilter *pFilter,  // Pointer to the filter to search.
    REFGUID iid,           // IID of the interface.
    void **ppUnk)          // Receives the interface pointer.
{
    if (!pFilter || !ppUnk) return E_POINTER;


    HRESULT hr = E_FAIL;
    IEnumPins *pEnum = 0;
    if (FAILED(pFilter->EnumPins(&pEnum)))
    {
        return E_FAIL;
    }
    // Query every pin for the interface.
    IPin *pPin = 0;
    while (S_OK == pEnum->Next(1, &pPin, 0))
    {
        hr = pPin->QueryInterface(iid, ppUnk);
        pPin->Release();
        if (SUCCEEDED(hr))
        {
            break;
        }
    }
    pEnum->Release();
    return hr;
}
//综合了FindFilterInterface和FindPinInterfae两个函数的综合功能

HRESULT FindInterfaceAnywhere(
    IGraphBuilder *pGraph, 
    REFGUID iid, 
    void **ppUnk)
{
    if (!pGraph || !ppUnk) return E_POINTER;
    HRESULT hr = E_FAIL;
    IEnumFilters *pEnum = 0;
    if (FAILED(pGraph->EnumFilters(&pEnum)))
    {
        return E_FAIL;
    }
    // Loop through every filter in the graph.
    IBaseFilter *pF = 0;
    while (S_OK == pEnum->Next(1, &pF, 0))
    {
        hr = pF->QueryInterface(iid, ppUnk);
        if (FAILED(hr))
        {
            // The filter does not expose the interface, but maybe
            // one of its pins does.
            hr = FindPinInterface(pF, iid, ppUnk);
        }
        pF->Release();
        if (SUCCEEDED(hr))
        {
            break;
        }
    }
    pEnum->Release();
    return hr;
}

比如现在通过IGraphBuilder::RenderFile函数,构建了一条内含DV格式数据的AVI文件的回放链路。我们想要得到其中DV视频解码Filter上的IIPDVDec接口,以设置DV解码输出的图像大小为原始图像的四分之一,代码如下:
hr = pGraph->RenderFile(L"C:\\example.avi", 0);

if(SUCCEEDED(hr))

{

IIPDVDec* pDvDec;

hr = FindFilterInterface(pGraph, IID_IIPDVDec, (void**)&pDvDec);

if(SUCCEEDED(hr))

{

........

}

}


5.遍历Filter链路

给定一个Filter链路上的某个Filter,我们可以向上或者向下得到所有的其他的Filter。

HRESULT GetNextFilter(
    IBaseFilter *pFilter, // Pointer to the starting filter
    PIN_DIRECTION Dir,    // Direction to search (upstream or downstream)
    IBaseFilter **ppNext) // Receives a pointer to the next filter.
{
    if (!pFilter || !ppNext) return E_POINTER;

    IEnumPins *pEnum = 0;
    IPin *pPin = 0;
    HRESULT hr = pFilter->EnumPins(&pEnum);
    if (FAILED(hr)) return hr;
    while (S_OK == pEnum->Next(1, &pPin, 0))
    {
        // See if this pin matches the specified direction.
        PIN_DIRECTION ThisPinDir;
        hr = pPin->QueryDirection(&ThisPinDir);
        if (FAILED(hr))
        {
            // Something strange happened.
            hr = E_UNEXPECTED;
            pPin->Release();
            break;
        }
        if (ThisPinDir == Dir)
        {
            // Check if the pin is connected to another pin.
            IPin *pPinNext = 0;
            hr = pPin->ConnectedTo(&pPinNext);
            if (SUCCEEDED(hr))
            {
                // Get the filter that owns that pin.
                PIN_INFO PinInfo;
                hr = pPinNext->QueryPinInfo(&PinInfo);
                pPinNext->Release();
                pPin->Release();
                pEnum->Release();
                if (FAILED(hr) || (PinInfo.pFilter == NULL))
                {
                    // Something strange happened.
                    return E_UNEXPECTED;
                }
                // This is the filter we're looking for.
                *ppNext = PinInfo.pFilter; // Client must release.
                return S_OK;
            }
        }
        pPin->Release();
    }
    pEnum->Release();
    // Did not find a matching filter.
    return E_FAIL;
}







































版权声明:本文为博主原创文章,未经博主允许不得转载。

创建一个directshow应用程序二2.

事件交互的是实现 directshow有一种机制,实现应用程序和Filter Graph的交互控制,该机制叫做事件通知(Event Notification)。 当Filter状态转换,运...
  • rootusers
  • rootusers
  • 2014年12月26日 10:31
  • 8309

DirectShow应用——音频捕捉

现在的所谓多媒体电脑一般都会有声卡(软声卡或硬声卡),有声卡就能进行音频的捕捉。大家一定熟悉Windows自带的附件“录音机”程序,可以通过麦克风进行录音,最终生成一个Wave文件。读完本文之后,你就...
  • markman101
  • markman101
  • 2010年06月30日 10:55
  • 2703

如何用myEclipse建立一个Java应用程序

myEclipse 是一个开放源代码的、基于Java的可扩展开发平台。就其本身而言,它只是一个框架和一组服务,用于通过插件组件构建开发环境。myEclipse 附带了一个标准的插件集,包括Java开发...
  • yhf19881015
  • yhf19881015
  • 2017年12月27日 13:39
  • 127

基于 VC+OpenCV+DirectShow 的多个摄像头同步工作

因项目需要采集2个摄像头的数据进行双目检测,一开始采用以下代码来测试: #include "stdafx.h" #include #include #include int main(...
  • axingxingzp
  • axingxingzp
  • 2013年10月29日 16:24
  • 1660

DirectShow下视频显示窗口设置

显示视频 DirectShow 提供了如下过滤器来显示视频: l         Video Renderer 过滤器. 该过滤器可用于所有的支持DirectX的平台,它对平台没有其它特殊的要求。可...
  • CtiyMan
  • CtiyMan
  • 2015年09月11日 11:11
  • 1124

最简单的基于DirectShow的示例:视频播放器自定义版

本文记录一个简单的基于DirectShow的自定义的视频播放器。这里所说的“自定义播放器”,实际上指的是自己在Filter Graph中手动逐个添加Filter,并且连接这些Filter的后运行的播放...
  • leixiaohua1020
  • leixiaohua1020
  • 2015年01月11日 18:05
  • 10139

面向对象,掌握Windows应用程序的创建过程:一个窗口的诞生

一、Windows应用程序简介          现在的Windows支持32位和64位的优先权式多任务(preemptive multitasking)及多线程的图形操作系统。Windows拥...
  • QQrenzai
  • QQrenzai
  • 2015年11月30日 08:49
  • 619

华南理工大学2013级计科一班操作系统实验

实验一:进程和线程的创建 1. 在linux下编写一个应用程序,命名为an_ch2_1b。这个程序不断地输出如下行: Those output come from child,[系统时间] 另外写一个...
  • u013497977
  • u013497977
  • 2015年10月10日 00:25
  • 2738

根据一个表的结构创建另一个表

在sqlserver 下想复制一张表的,想到oracle下直接create table xxx as select * from ....即可。 sqlserver的语法是 : select * i...
  • u012307002
  • u012307002
  • 2014年05月22日 09:14
  • 1575

创建struts2支持的第一个应用(二)

下载struts2 网址:http://struts.apache.org/download.cgi 一般下载Full Distribution完整all版本,该版本目录结构: apps里...
  • csdnyoyo
  • csdnyoyo
  • 2016年04月23日 21:15
  • 265
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:创建一个directshow应用程序一
举报原因:
原因补充:

(最多只允许输入30个字)