简体   繁体   English

ISampleGrabber :: GetCurrentBuffer()返回E_OUTOFMEMORY

[英]ISampleGrabber::GetCurrentBuffer() returns E_OUTOFMEMORY

I am new to directshow and using DirectShow Sample "FrameGrabberDemo" and facing issue in getting the image. 我是DirectShow的新手,并使用DirectShow示例“ FrameGrabberDemo”,并且在获取图像时遇到问题。 I tried with .avi and .mpg, both are giving same issue. 我尝试使用.avi和.mpg,两者都给出相同的问题。

First issue might be the S_FALSE return value from IMediaControl::Run(). 第一个问题可能是IMediaControl :: Run()的S_FALSE返回值。 However, it is not an error and states that: 但是,这不是错误,并指出:

The graph is preparing to run, but some filters have not completed the transition to a running state. 该图正在准备运行,但是某些过滤器尚未完成向运行状态的转换。

Second observation is ISampleGrabber::GetCurrentBuffer() returns E_OUTOFMEMORY code, which states that "The specified buffer is not large enough." 第二个观察结果是ISampleGrabber :: GetCurrentBuffer()返回E_OUTOFMEMORY代码,该代码指出“指定的缓冲区不够大”。 However, the BitmapInfo has biImageSize = 1244160 and also MediaType has ISampleSize = 1244160. 但是,BitmapInfo的biImageSize = 1244160,MediaType的ISampleSize = 1244160。

HRESULT CFrameGrabberDemoDlg::DoExtractFrame()
{
WCHAR wFile[MAX_PATH];
MultiByteToWideChar( CP_ACP, 0, m_FilePath, -1, wFile, MAX_PATH );

// Create the graph builder
CComPtr<IGraphBuilder> pGraphBuilder;
HRESULT hr = ::CoCreateInstance(CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, 
                                IID_IGraphBuilder, (void**)&pGraphBuilder);
if (FAILED(hr))
    return hr;
ASSERT(pGraphBuilder != NULL);

// Create the "Grabber filter"
CComPtr<IBaseFilter>    pGrabberBaseFilter;
CComPtr<ISampleGrabber> pSampleGrabber;
AM_MEDIA_TYPE   mt;
hr = ::CoCreateInstance(CLSID_SampleGrabber, NULL, CLSCTX_INPROC_SERVER, 
                        IID_IBaseFilter, (LPVOID *)&pGrabberBaseFilter);
if (FAILED(hr))
    return hr;
pGrabberBaseFilter->QueryInterface(IID_ISampleGrabber, (void**)&pSampleGrabber);
if (pSampleGrabber == NULL)
    return E_NOINTERFACE;
hr = pGraphBuilder->AddFilter(pGrabberBaseFilter,L"Grabber");
if (FAILED(hr))
    return hr;

ZeroMemory(&mt, sizeof(AM_MEDIA_TYPE));
mt.majortype = MEDIATYPE_Video;
mt.subtype = MEDIASUBTYPE_RGB24;
mt.formattype = FORMAT_VideoInfo; 
hr = pSampleGrabber->SetMediaType(&mt);        
if (FAILED(hr))
    return hr;
hr = pGraphBuilder->RenderFile(wFile,NULL);
if (FAILED(hr))
    return hr;

CComPtr<IMediaControl> pMediaControl;
CComPtr<IMediaEvent> pMediaEventEx;
// QueryInterface for some basic interfaces
pGraphBuilder->QueryInterface(IID_IMediaControl, (void **)&pMediaControl);
pGraphBuilder->QueryInterface(IID_IMediaEvent, (void **)&pMediaEventEx);

if (pMediaControl == NULL || pMediaEventEx == NULL)
    return E_NOINTERFACE;

// Set up one-shot mode.
hr = pSampleGrabber->SetBufferSamples(TRUE);
if (FAILED(hr))
    return hr;

hr = pSampleGrabber->SetOneShot(TRUE);
if (FAILED(hr))
    return hr;

CComQIPtr<IMediaSeeking> pSeek = pMediaControl;
if (pSeek == NULL)
    return E_NOINTERFACE;
LONGLONG Duration;
hr = pSeek->GetDuration(&Duration);
if (FAILED(hr))
    return hr;
int NumSecs = int(Duration/10000000);

REFERENCE_TIME rtStart = 1 * 10000000;
if (NumSecs < 1)
    rtStart = 0;
REFERENCE_TIME rtStop = rtStart; 

hr = pSeek->SetPositions(&rtStart, AM_SEEKING_AbsolutePositioning, 
                         &rtStop, AM_SEEKING_AbsolutePositioning);
if (FAILED(hr))
    return hr;

CComQIPtr<IVideoWindow> pVideoWindow = pGraphBuilder;

hr = pVideoWindow->put_AutoShow(OAFALSE);
if (FAILED(hr))
    return hr;

// Run the graph and wait for completion.
hr = pMediaControl->Run();
if (FAILED(hr))
    return hr;

long evCode;
hr = pMediaEventEx->WaitForCompletion(INFINITE, &evCode);
if (FAILED(hr))
    return hr;

AM_MEDIA_TYPE MediaType;
ZeroMemory(&MediaType,sizeof(MediaType));
hr = pSampleGrabber->GetConnectedMediaType(&MediaType); 
if (FAILED(hr))
    return hr;

// Get a pointer to the video header.
VIDEOINFOHEADER *pVideoHeader = (VIDEOINFOHEADER*)MediaType.pbFormat;
if (pVideoHeader == NULL)
    return E_FAIL;

// The video header contains the bitmap information. 
// Copy it into a BITMAPINFO structure.
BITMAPINFO BitmapInfo;
ZeroMemory(&BitmapInfo, sizeof(BitmapInfo));
CopyMemory(&BitmapInfo.bmiHeader, &(pVideoHeader->bmiHeader), sizeof(BITMAPINFOHEADER));

// Create a DIB from the bitmap header, and get a pointer to the buffer.
void *buffer = NULL;
HBITMAP hBitmap = ::CreateDIBSection(0, &BitmapInfo, DIB_RGB_COLORS, &buffer, NULL, 0);
GdiFlush(); 

// Copy the image into the buffer.
long size = 0;
hr = pSampleGrabber->GetCurrentBuffer(&size, (long *)buffer);
if (FAILED(hr))
    return hr;

long Width = pVideoHeader->bmiHeader.biWidth;
long Height = pVideoHeader->bmiHeader.biHeight;

HBITMAP hOldBitmap = m_Image.SetBitmap(hBitmap);
if (hOldBitmap != NULL)
    ::DeleteObject(hOldBitmap);

return S_OK;
}

The ::CreateDIBSection is also not returning NULL and buffer also got initialized. :: CreateDIBSection也未返回NULL,并且缓冲区也已初始化。

How can this be resolved? 如何解决?

You are requesting data into zero-length buffer: 您正在将数据请求到零长度缓冲区中:

long size = 0;
hr = pSampleGrabber->GetCurrentBuffer(&size, (long *)buffer);

The error code looks relevant: 错误代码看起来很相关:

If pBuffer is not NULL, set this parameter equal to the size of the buffer, in bytes. 如果pBuffer不为NULL,则将此参数设置为等于缓冲区的大小(以字节为单位)。

E_OUTOFMEMORY The specified buffer is not large enough. E_OUTOFMEMORY指定的缓冲区不够大。

You simple need proper arguments in the call in question (proper buffer length). 您只需在所涉及的调用中使用适当的参数(适当的缓冲区长度)即可。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM