Directshow 捕捉程序的两个问题

Directshow 捕捉程序的两个问题 (2009-04-22 14:41:09)

     之前写的程序里,一直存在两个问题。一个是程序运行时,点击打开视频后,关闭视频。再点击打开视频,就会弹出对话框“can’t build graph"。二是,无法改变采集的图像的大小,总是固定的320*240.注意,此处不是指显示窗口的大小。

     结合查找的一些资料,终于解决了。

    问题一:是Capturevidieo 类析构函数里,少释放了pVW。晕。网上找的类里,都漏了释放了。

    问题二:

               结合以下资料,解决了:

              1

 (1)获得IAMStreamConfig接口  
  hr   =   gcap.pBuilder->FindInterface(&PIN_CATEGORY_CAPTURE,  
                  &MEDIATYPE_Interleaved,  
                  gcap.pVCap,   IID_IAMStreamConfig,   (void   **)&gcap.pVSC);  
  (2)控制  
  if(gcap.pVSC)//   &&   gcap.fUseFrameRate)    
  {  
  hr   =   gcap.pVSC->GetFormat(&pmt);  
  //   DV   capture   does   not   use   a   VIDEOINFOHEADER  
  if(hr   ==   NOERROR)    
  {  
  if(pmt->formattype   ==   FORMAT_VideoInfo)    
  {  
  VIDEOINFOHEADER   *pvi   =   (VIDEOINFOHEADER   *)pmt->pbFormat;  
  pvi->bmiHeader.biWidth=320;  
  pvi->bmiHeader.biHeight   =240;  
  pvi->bmiHeader.biSizeImage=320*240*3;  
  hr   =   gcap.pVSC->SetFormat(pmt);  
  }  
  DeleteMediaType(pmt);  
  }  
  }  
2

调整视频输出格式
我们知道视频流可以有多种输出格式,一个设备可以支持16-bit RGB, 32-bit RGB, and YUYV,在每一种格式下,设备还可以调整视频桢的大小。
在WDM驱动设备上,IAMStreamConfig 接口用来报告设备输出视频的格式的,VFW设备,可以采用对话框的方式来设置,参见前面的内容。
捕捉Filter的捕捉pin和预览pin都支持IAMStreamConfig 接口,可以通过ICaptureGraphBuilder2::FindInterface获得IAMStreamConfig接口。

 IAMStreamConfig *pConfig = NULL;
  hr = pBuild->FindInterface(
  &PIN_CATEGORY_PREVIEW, // Preview pin.
  0, // Any media type.
  pCap, // Pointer to the capture filter.
  IID_IAMStreamConfig, (void**)&pConfig);

设备还支持一系列的媒体类型,对于每一个媒体类型,设备都要支持一系列的属性,比如,桢的大小,图像如何缩放,桢率的范围等。
通过IAMStreamConfig::GetNumberOfCapabilities获得设备所支持的媒体类型的数量。这个方法返回两个值,一个是媒体类型的数量,二是属性所需结构的大小。
这个结构的大小很重要,因为这个方法是用于视频和音频的,视频采用的是VIDEO_STREAM_CONFIG_CAPS结构,音频用AUDIO_STREAM_CONFIG_CAPS结构。
通过函数IAMStreamConfig::GetStreamCaps来枚举媒体类型,要给这个函数传递一个序号作为参数,这个函数返回媒体类型和相应的属性结构体。看代码把

 int iCount = 0, iSize = 0;
  hr = pConfig->GetNumberOfCapabilities(&iCount, &iSize);
// Check the size to make sure we pass in the correct structure.
  if (iSize == sizeof(VIDEO_STREAM_CONFIG_CAPS)
  {
  // Use the video capabilities structure.
  for (int iFormat = 0; iFormat < iCount; iFormat++)
  {
  VIDEO_STREAM_CONFIG_CAPS scc;
  AM_MEDIA_TYPE *pmtConfig;
  hr = pConfig->GetStreamCaps(iFormat, &pmtConfig, (BYTE*)&scc);
  if (SUCCEEDED(hr))
  {
 
  // Delete the media type when you are done.
  hr = pConfig->SetFormat(pmtConfig);//重新设置视频格式
  DeleteMediaType(pmtConfig);
  }
  }

你可以调用IAMStreamConfig::SetFormat设置新的媒体类型
hr = pConfig->SetFormat(pmtConfig);
如果pin没有连接,当连接的时候就试图用新的格式,如果pin已经在连接了,它就会用的新的媒体格式重新连接。在任何一种情况下,下游的filter都有可能拒绝新的媒体格式。
在SetFormat前你可以修改VIDEO_STREAM_CONFIG_CAPS结构来重新设置媒体类型。
例如:
如果GetStreamCaps返回的是24-bit RGB format,桢的大小是320 x 240 像素,你可以通过检查媒体类型的major type,subtpye,和format等值

 if ((pmtConfig.majortype == MEDIATYPE_Video) &&
  (pmtConfig.subtype == MEDIASUBTYPE_RGB24) &&
  (pmtConfig.formattype == FORMAT_VideoInfo) &&
  (pmtConfig.cbFormat >= sizeof (VIDEOINFOHEADER)) &&
  (pmtConfig.pbFormat != NULL))
  {
  VIDEOINFOHEADER *pVih = (VIDEOINFOHEADER*)pmtConfig.pbFormat;
  // pVih contains the detailed format information.
  LONG lWidth = pVih->bmiHeader.biWidth;
  LONG lHeight = pVih->bmiHeader.biHeight;
  }

VIDEO_STREAM_CONFIG_CAPS结构里包含了该媒体类型的视频长度和宽度的最大值和最小值,还有递增的幅度值,就是每次调整视频size的幅度,例如,设备可能返回如下的值
? MinOutputSize: 160 x 120
? MaxOutputSize: 320 x 240
? OutputGranularityX: 8 pixels (horizontal step size)
? OutputGranularityY: 8 pixels (vertical step size)
这样你可以在(160, 168, 176, ... 304, 312, 320) 范围内设置宽度,在 (120, 128, 136, ... 104, 112, 120).设置高度值,

图6
如果想设置新的值,直接修改在GetStreamCaps函数中返回的值即可,

 pVih->bmiHeader.biWidth = 160;
  pVih->bmiHeader.biHeight = 120;
  pVih->bmiHeader.biSizeImage = DIBSIZE(pVih->bmiHeader);

然后将媒体类型传递给SetFormat函数,就可修改视频格式了。

 

三 自己修改后的程序如下:

cpp

CCaptureVideo::~CCaptureVideo()
{
 // Stop media playback
if (bIsVideoOpen)
{ if(m_pMC)m_pMC->Stop();
 if(m_pVW)
 {
  m_pVW->put_Visible(OAFALSE);
  m_pVW->put_Owner(NULL);
 }
    srelease(m_pBF);

    srelease(m_pCapture);
 srelease(m_pMC);
 srelease(m_pGB);
 srelease(m_pVW);//加上这句,就可以了,晕,否则在程序里没法连续打开两次视频

 
 CoUninitialize( );

}
HRESULT CCaptureVideo::Init(int iDeviceID, HWND hWnd)
{
 HRESULT hr;
 hr = InitCaptureGraphBuilder();
 if (FAILED(hr))
 {
  AfxMessageBox("Failed to get video interfaces!");
  return hr;
 }
 // Bind Device Filter. We know the device because the id was passed in
 if(!BindFilter(iDeviceID, &m_pBF))
 {
  AfxMessageBox("未找到USB摄像头!/n请检查设备后重试!");
        //PostQuitMessage(0);
        return S_FALSE;
 }
 hr = m_pGB->AddFilter(m_pBF, L"Capture Filter");

 // hr = m_pCapture->RenderStream(&PIN_CATEGORY_PREVIEW, &MEDIATYPE_Video,
 // m_pBF, NULL, NULL);

 // create a sample grabber
 hr = m_pGrabber.CoCreateInstance( CLSID_SampleGrabber );
 if( !m_pGrabber )
 {
  AfxMessageBox("Fail to create SampleGrabber, maybe qedit.dll is not registered?");
  return hr;
 }
 CComQIPtr< IBaseFilter, &IID_IBaseFilter > pGrabBase( m_pGrabber );


 //设置视频格式
  
    IAMStreamConfig *pConfig = NULL;
 hr = m_pCapture->FindInterface(&PIN_CATEGORY_CAPTURE,  0, // Any media type.
   m_pBF, // Pointer to the capture filter.
   IID_IAMStreamConfig, (void**)&pConfig);
 
    AM_MEDIA_TYPE *pmt;
 hr   =   pConfig ->GetFormat(&pmt); 
 
    //ZeroMemory(pmt, sizeof(AM_MEDIA_TYPE));
    pmt->majortype = MEDIATYPE_Video;
    pmt->subtype = MEDIASUBTYPE_RGB24;
 pmt->formattype =  FORMAT_VideoInfo;
 VIDEOINFOHEADER   *pvi   =   (VIDEOINFOHEADER   *)pmt->pbFormat;  
 pvi->bmiHeader.biWidth=640;  
 pvi->bmiHeader.biHeight   =480;  
 pvi->bmiHeader.biSizeImage=640*480*3;   //设置帧大小,而不是仅仅改变预览窗口大小
 hr   =   pConfig->SetFormat(pmt);  
   
 FreeMediaType(*pmt);   
 
 AM_MEDIA_TYPE mt;
 //ZeroMemory(&mt, sizeof(AM_MEDIA_TYPE));
 //mt.majortype = MEDIATYPE_Video;
 //mt.subtype = MEDIASUBTYPE_RGB24;
 // mt.formattype =  FORMAT_VideoInfo;
 // VIDEOINFOHEADER * vih = (VIDEOINFOHEADER*) mt.pbFormat;//不能在这里设置图像文件信息
    // 
// hr = m_pGrabber->SetMediaType(&mt);

 if( FAILED( hr ) )
 {
  AfxMessageBox("Fail to set media type!");
  return hr;
 }

 hr = m_pGB->AddFilter( pGrabBase, L"Grabber" );

 if( FAILED( hr ) )
 {
  AfxMessageBox("Fail to put sample grabber in graph");
  return hr;
 }

 // try to render preview/capture pin
 hr = m_pCapture->RenderStream(&PIN_CATEGORY_PREVIEW, &MEDIATYPE_Video,m_pBF,pGrabBase,NULL);
 if( FAILED( hr ) )
    hr = m_pCapture->RenderStream(&PIN_CATEGORY_CAPTURE, &MEDIATYPE_Video,m_pBF,pGrabBase,NULL);
 if( FAILED( hr ) )
 {
  AfxMessageBox("Can’t build the graph");
  return hr;
 }
 
 

 hr = m_pGrabber->GetConnectedMediaType( &mt );

 if ( FAILED( hr) )
 {
  AfxMessageBox("Failt to read the connected media type");
  return hr;
 }
   
 VIDEOINFOHEADER * vihh = (VIDEOINFOHEADER*) mt.pbFormat;
     // vihh->bmiHeader.biWidth=320;
     // vihh->bmiHeader.biHeight=240;
     // vihh->bmiHeader.biSizeImage=320*240*3; 
// hr = m_pGrabber->SetMediaType(&mt);
 mCB.lWidth =vihh->bmiHeader.biWidth;
     
 mCB.lHeight =vihh->bmiHeader.biHeight;
 
 FreeMediaType(mt);
    

 hr = m_pGrabber->SetBufferSamples( false );
 hr = m_pGrabber->SetOneShot( FALSE );
 hr = m_pGrabber->SetCallback( &mCB, 1 );//影响回调

 //设置视频捕捉窗口
 m_hWnd = hWnd ;
 SetupVideoWindow();
 hr = m_pMC->Run();//开始视频捕捉
 if(FAILED(hr))
 {
  AfxMessageBox("Couldn’t run the graph!");
  return hr;
 }

    bIsVideoOpen = TRUE;

 return S_OK;
}

你可能感兴趣的:(Directshow 捕捉程序的两个问题)