c++ - 如何访问 USB 电视调谐卡等 BDA 设备

标签 c++ directshow

我在 Windows 10 上工作,想从我的电视调谐器 USB 设备访问视频流。

首先,我尝试使用媒体基础 API,经过多次搜索后才知道该 API 不支持电视调谐卡。

然后我切换到 directshow 但枚举视频设备没有列出我的电视调谐器设备。仅列出我的网络摄像头。我发现我应该使用过滤器和引脚,但没有此类操作的指南。

任何帮助、建议或示例都会有所帮助。

谢谢。

K

最佳答案

这个周末我正在解决同样的问题 :-)。我想做一个简单的电视调谐器/播放器。直接显示似乎是开发的最后一部分,因为您必须从 USB 加密狗获取数据。

我搜索了很多,但正如您所写,不幸的是几乎没有指南。但是,我认为,您可以从 MS 页面开始 - Microsoft TV Technologies: https://msdn.microsoft.com/en-us/library/windows/desktop/dd695086%28v=vs.85%29.aspx?f=255&MSPPError=-2147217396 并专注于统一调优模型,这导致 Microsoft 统一调优模型,其中有一些提示,如何操作。但是,对于代码示例和详细信息,我闭口不谈……这是基于带有 OLE 的 MFC(我对这项技术还很陌生)。

目前,我正在研究部分 Tuning Spaces,不幸的是,存在一些陷阱,如注册表权限(我还没有解决这个问题)。 稍后,您应该在编写时了解有关过滤器和引脚的知识。参见 https://msdn.microsoft.com/en-us/library/dd377601(v=vs.85).aspx . 不幸的是,我不知道如何为 DVB-T 创建图表。无论如何,在我的开发中。使用 Win 10 的桌面,当我打开 Direct Show Filters 时,GraphEdit 崩溃(可能与 DVB-T HW 的当前 SW 崩溃的原因相同)。我发现有趣的博客与替代图形编辑器: https://en.wikipedia.org/wiki/GraphStudio - 检查外部链接。较新的 GraphStudioNext 看起来也很有前途(我还没有尝试过)。

这是我刚刚开始的方式,如果您仍然感兴趣,希望这对您有所帮助。如果您有更多经验,欢迎任何建议! :-)

我的测试代码(还不能正常工作!)如下所示。灵感来自 ITuner::put_TuneRequest() call ignored

hCo = CoInitializeEx(nullptr, COINIT::COINIT_APARTMENTTHREADED);

if (hCo == S_OK)
{
    IErrorInfo* pIErrorInfo = nullptr;
    HRESULT hr = S_FALSE;

    // Create the SystemTuningSpaces container.
    PQTuningSpaceContainer pTuningSpaceContainer; // PQTuningSpaceContaineris typedef of CComQIPtr<ITuningSpace
    hr = pTuningSpaceContainer.CoCreateInstance(__uuidof(SystemTuningSpaces));
    if (SUCCEEDED(hr))
    {
        // Get the enumerator for the collection.
        CComPtr<IEnumTuningSpaces> pTuningSpaceEnum;
        hr = pTuningSpaceContainer->get_EnumTuningSpaces(&pTuningSpaceEnum);
        if (SUCCEEDED(hr))
        {
            // Loop through the collection.
            PQTuningSpace pTuningSpace;
            while (S_OK == pTuningSpaceEnum->Next(1, &pTuningSpace, nullptr))
            {
                BSTR bstrDVBValue;
                pTuningSpace->get_UniqueName(&bstrDVBValue);
                printf("%S\n", bstrDVBValue);
                pTuningSpace->get_FriendlyName(&bstrDVBValue);
                printf("%S\n", bstrDVBValue);
                pTuningSpace->get_NetworkType(&bstrDVBValue);
                printf("%S\n\n", bstrDVBValue);
                // pTuningSpace points to the next tuning space.
                // You can query this pointer for derived interfaces.
                pTuningSpace.Release();
            }
        }
    }

    PQDVBTuningSpace pDVBTuningSpace;
    WCHAR szDVBNameU[64] = L"DVB-T";
    WCHAR szDVBNameF[64] = L"Local DVB-T Digital Antenna";
    // WCHAR szDVBNetworkType[64] = L"{B2F3A67C-29DA-4C78-8831-091ED509A475}";
    WCHAR szDVBNetworkType[64] = L"{216C62DF-6D7F-4e9a-8571-05F14EDB766A}"; // DVB-T Network type

    hr = pDVBTuningSpace.CoCreateInstance(__uuidof(DVBTuningSpace));
    if (FAILED(hr))
    {
        printf("Failed to create system tuning spaces object.");
        return;
    }

    hr = pDVBTuningSpace->put_SystemType(DVBSystemType::DVB_Terrestrial);
    if (FAILED(hr))
    {
        printf("Failed to put system type of tuning space.");
        return;
    }

    hr = pDVBTuningSpace->put_NetworkType(szDVBNetworkType);
    if (FAILED(hr))
    {
        printf("Failed to put network type of tuning space.");
        return;
    }

    BSTR bstrDVBNetworkType = SysAllocString(szDVBNetworkType);
    hr = pDVBTuningSpace->put_NetworkType(bstrDVBNetworkType);
    SysFreeString(bstrDVBNetworkType);
    if (FAILED(hr))
    {
        printf("Failed to put network type of tuning space.");
        return;
    }

    BSTR bstrDVBName = nullptr;
    bstrDVBName = SysAllocString(szDVBNameU);
    hr = pDVBTuningSpace->put_UniqueName(bstrDVBName);
    SysFreeString(bstrDVBName);

    bstrDVBName = SysAllocString(szDVBNameF);
    hr = pDVBTuningSpace->put_FriendlyName(bstrDVBName);
    SysFreeString(bstrDVBName);

    if (FAILED(hr))
    {
        printf("Failed to put name of tuning space.");
        return;
    }

    PQDVBTLocator pDVBTLocator;
    hr = pDVBTLocator.CoCreateInstance(__uuidof(DVBTLocator));
    hr = pDVBTLocator->put_CarrierFrequency(538000);
    hr = pDVBTLocator->put_Bandwidth(8);
    hr = pDVBTuningSpace->put_DefaultLocator(pDVBTLocator);
    if (FAILED(hr))
    {
        printf("Failed to put locator of tuning space.");
        return;
    }

    VARIANT tiIndex = {};
    //PQTuningSpaceContainer pTuningSpaceContainer;
    //hr = pTuningSpaceContainer.CoCreateInstance(__uuidof(SystemTuningSpaces));
    hr = pTuningSpaceContainer->Add(pDVBTuningSpace, &tiIndex); // #fix needed: registry permissions
    if (hr == HRESULT_FROM_WIN32(ERROR_DUP_NAME))
    {
        CComPtr<IEnumTuningSpaces> pTuningSpaceEnum;
        hr = pTuningSpaceContainer->get_EnumTuningSpaces(&pTuningSpaceEnum);
        if (SUCCEEDED(hr))
        {
            // Loop through the collection.
            PQTuningSpace pTuningSpace;
            tiIndex.intVal = 0;
            while (S_OK == pTuningSpaceEnum->Next(1, &pTuningSpace, nullptr))
            {
                CComBSTR name;
                hr = pTuningSpace->get_UniqueName(&name);
                if (SUCCEEDED(hr))
                {
                    if (name == szDVBNameU)
                    {
                        hr = pTuningSpaceContainer->put_Item(tiIndex, pDVBTuningSpace); // #fix needed: E_INVALIDARG One or more arguments are invalid.
                        break;
                    }
                }

                tiIndex.intVal++;
                pTuningSpace.Release();
            }
        }
    }

    PQTuneRequest pTuneRequest;
    hr = pDVBTuningSpace->CreateTuneRequest(&pTuneRequest);
    if (FAILED(hr))
    {
        printf("Failed to create tune request.");
        return;
    }

    PQDVBTuneRequest pDVBTuneRequest(pTuneRequest);
    if (pDVBTuneRequest) 
    {
        hr = pDVBTuneRequest->put_SID(-1);
        hr = pDVBTuneRequest->put_TSID(-1);
        hr = pDVBTuneRequest->put_ONID(-1);
    }

    CComPtr<IGraphBuilder> pGraph;
    hr = CoCreateInstance(CLSID_FilterGraph, NULL, CLSCTX::CLSCTX_INPROC_SERVER, IID_IGraphBuilder, (void **)&pGraph);

    CComPtr<IBaseFilter> pNetworkProvider;
    GUID CLSIDNetworkType;
    hr = pDVBTuningSpace->get__NetworkType(&CLSIDNetworkType);
    hr = CoCreateInstance(CLSIDNetworkType, NULL, CLSCTX::CLSCTX_INPROC_SERVER, IID_IBaseFilter, (void **)&pNetworkProvider);
    hr = pGraph->AddFilter(pNetworkProvider, L"Network Provider");

    //// Query for ITuner.
    CComQIPtr<ITuner> pTuner(pNetworkProvider);
    if (pTuner) 
    {
        // Submit the tune request to the network provider.
        hr = pTuner->put_TuneRequest(pTuneRequest);
    }

    // TODO: What next???

    CoUninitialize();

编辑:

这会有很大帮助: https://github.com/dgis/CodeTV (检查并为我工作)

关于c++ - 如何访问 USB 电视调谐卡等 BDA 设备,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37069301/

相关文章:

c++ - 在条件中评估字符串

c++ - 将项目添加到队列时运行线程

video - DirectShow 音频/视频 PTS 时钟计算

c++ - VMR9 对捕获的文件施加了错误的视频大小

c++ - DirectShow - 无法创建新线程

c++ - 如何在 Windows 上使用 DirectShow 或其他工具充当麦克风?

c++ - 如何在 C++ 源文件之间共享静态变量?

c++ - 在 C 或 C++ 中使用分布生成数字

c++ - 将代码从 OpenGL 2.1 转换为 OpenGL 3.2

winapi - 如何在 Windows 10 中创建虚拟网络摄像头?