Commongrabber работает нормально на файлах AVI / MPEG, но измельчает WMV
-
27-09-2019 - |
Вопрос
Я использовал последнюю версию Wpfmediakit.. Отказ То, что я пытаюсь сделать, это напишите образец приложения, которое будет использовать образец Grabber для захвата видеокадров видеофайлов, чтобы я мог иметь их как отдельные растровые изображения.
Пока что я удачи в следующем коде при строительстве и рендеринг моего графа. Однако, когда я использую этот код для воспроизведения видеофайла .wmv, когда прикреплен образец, он будет воспроизводить Joughy или Chartpy. Если я прокомментую линию, где добавляю фильтр образецзыра, он работает нормально. Опять же, он работает с помощью образец правильно с AVI / MPEG и т. Д.
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 вдове своего собственного вместо моего управления, однако образецGrabber функционирует правильно и без пропуска. Поэтому я думаю, что ошибка находится в VMR9 / Commungrabbing где-то.
Любая помощь? Я новый к этому.
Решение
Некоторые декодеры будут использовать аппаратное ускорение с использованием DXVA. Это реализуется путем переговоров о частичном декодированном формате и передача этого частично декодированных данных для рендерера для завершения декодирования и рендера. Если вы вставите образец GRABBER, сконфигурированный на RGB24 между декодером и рендерером, вы отключите аппаратное ускорение.
Это, я уверен, это суть проблемы. Детали по-прежнему немного расплывчаты, я боюсь, например, почему это работает, когда вы используете VMR-7 по умолчанию, но не удается при использовании VMR-9. Я догадаю, что декодер пытается использовать DXVA и неспособность, в случае VMR-9, но имеет разумное программное обеспечение только резервное копирование, которое хорошо работает в VMR-7.
Я не знаком с WPFMediakit, но я думаю, что самое простое решение - заменить явное создание VMR-9 с явным созданием VMR-7. То есть, если декодер работает программным обеспечением - только с VMR-7, затем используйте это и сосредоточиться на фиксировании репаратории окна.
Другие советы
Закончил код, который я опубликовал (который сам по себе был довольно бесстыдно, слегка модифицирован мной из исходного кода jeremiah Morrill's WPFMediakit) был на самом деле адекватен файлы .wmv и быть схваченным.
Кажется, что измельчена есть что делать с пробегом через VS отладчик или сам VS2008. После вощения с XAML на некоторое время в визуальном редакторе, то запустите приложение, у меня будет введено это измельченное поведение. Выключение VS2008, кажется, исправляет это. :П
Таким образом, не большая часть ответа, но, по крайней мере, A (раздражает - перезапустить VS2008) исправить, когда он вырос.