Samplegrabber在AVI / MPEG文件上运行良好,但与WMV不同

时间:2010-04-15 21:44:37

标签: c# directshow wmv samplegrabber

我一直在使用最新版本的WPFMediaKit。我要做的是编写一个示例应用程序,它将使用Samplegrabber捕获视频文件的视频帧,以便将它们作为单独的位图。

到目前为止,在构建和渲染图形时,我对以下代码运气不错。但是,当我使用此代码播放.wmv视频文件时,如果连接了samplegrabber,它将播放跳跃或波涛汹涌。如果我注释掉我添加samplegrabber过滤器的行,它可以正常工作。同样,它可以使用AVI / MPEG等正确使用samplegrabber。

 protected virtual void OpenSource()
    {
        FrameCount = 0;
        /* Make sure we clean up any remaining mess */
        FreeResources();

        if (m_sourceUri == null)
            return;

        string fileSource = m_sourceUri.OriginalString;

        if (string.IsNullOrEmpty(fileSource))
            return;

        try
        {
            /* Creates the GraphBuilder COM object */
            m_graph = new FilterGraphNoThread() as IGraphBuilder;

            if (m_graph == null)
                throw new Exception("Could not create a graph");


            /* Add our prefered audio renderer */
            InsertAudioRenderer(AudioRenderer);

            var filterGraph = m_graph as IFilterGraph2;

            if (filterGraph == null)
                throw new Exception("Could not QueryInterface for the IFilterGraph2");

            IBaseFilter renderer = CreateVideoMixingRenderer9(m_graph, 1);                                

            IBaseFilter sourceFilter;

            /* Have DirectShow find the correct source filter for the Uri */
            var hr = filterGraph.AddSourceFilter(fileSource, fileSource, out sourceFilter);
            DsError.ThrowExceptionForHR(hr);

            /* We will want to enum all the pins on the source filter */
            IEnumPins pinEnum;

            hr = sourceFilter.EnumPins(out pinEnum);
            DsError.ThrowExceptionForHR(hr);

            IntPtr fetched = IntPtr.Zero;
            IPin[] pins = { null };

            /* Counter for how many pins successfully rendered */
            int pinsRendered = 0;                

            m_sampleGrabber = (ISampleGrabber)new SampleGrabber();
            SetupSampleGrabber(m_sampleGrabber);
            hr = m_graph.AddFilter(m_sampleGrabber as IBaseFilter, "SampleGrabber");
            DsError.ThrowExceptionForHR(hr);

            /* Loop over each pin of the source filter */
            while (pinEnum.Next(pins.Length, pins, fetched) == 0)
            {
                if (filterGraph.RenderEx(pins[0],
                                         AMRenderExFlags.RenderToExistingRenderers,
                                         IntPtr.Zero) >= 0)
                pinsRendered++;

                Marshal.ReleaseComObject(pins[0]);
            }

            Marshal.ReleaseComObject(pinEnum);
            Marshal.ReleaseComObject(sourceFilter);

            if (pinsRendered == 0)
                throw new Exception("Could not render any streams from the source Uri");

            /* Configure the graph in the base class */
            SetupFilterGraph(m_graph);

            HasVideo = true;
            /* Sets the NaturalVideoWidth/Height */
            //SetNativePixelSizes(renderer);
        }
        catch (Exception ex)
        {
            /* This exection will happen usually if the media does
             * not exist or could not open due to not having the
             * proper filters installed */
            FreeResources();

            /* Fire our failed event */
            InvokeMediaFailed(new MediaFailedEventArgs(ex.Message, ex));
        }

        InvokeMediaOpened();
    }

 private void SetupSampleGrabber(ISampleGrabber sampleGrabber)
    {
        FrameCount = 0;
        var mediaType = new AMMediaType
        {
            majorType = MediaType.Video,
            subType = MediaSubType.RGB24,
            formatType = FormatType.VideoInfo
        };

        int hr = sampleGrabber.SetMediaType(mediaType);

        DsUtils.FreeAMMediaType(mediaType);
        DsError.ThrowExceptionForHR(hr);

        hr = sampleGrabber.SetCallback(this, 0);
        DsError.ThrowExceptionForHR(hr);
    }

我已经阅读了一些说明.wmv或.asf格式是异步的东西。我试图插入一个WMAsfReader来解码哪个有效,但是一旦它进入VMR9就会产生相同的行为。此外,当我注释掉IBaseFilter renderer = CreateVideoMixingRenderer9(m_graph, 1);行并且filterGraph.Render(pins[0]);时,我已经让它正常工作 - 唯一的缺点是它现在呈现在自己的Activemovie寡妇而不是我的控制中,但是samplegrabber功能正常,没有任何跳过。所以我认为这个bug存在于VMR9 / samplegrabbing中。

有任何帮助吗?我是新手。

2 个答案:

答案 0 :(得分:1)

有些解码器会使用DXVA进行硬件加速。这是通过协商部分解码的格式,并将该部分解码的数据传递给渲染器以完成解码和渲染来实现的。如果在解码器和渲染器之间插入配置为RGB24的样本采集器,则将禁用硬件加速。

我确信,这是问题的症结所在。我担心细节仍然有点模糊,例如当你使用默认的VMR-7时它的工作原理,但是当你使用VMR-9时它会失败。我猜想解码器正在尝试使用dxva并在vmr-9情况下失败,但是它有一个合理的纯软件备份,在vmr-7中运行良好。

我不熟悉WPFMediaKit,但我认为最简单的解决方案是使用显式vmr-7创建替换显式vmr-9创建。也就是说,如果解码器仅使用vmr-7进行软件操作,那么请使用它并集中精力修复窗口重新定位问题。

答案 1 :(得分:0)

最终我发布的代码(这本身就是我从Jeremiah Morrill的WPFMediakit源代码中稍微修改过一些无耻的)实际上足以呈现.WMV文件并被样本抓取。

看起来波动与VS调试器或VS2008本身有关。在可视化编辑器中使用XAML一段时间后,然后运行应用程序,我将引入这种不稳定的行为。关闭VS2008似乎可以解决它。 :P

所以没有多少答案,但至少有一个(烦人的 - 重启VS2008)解决了它的问题。