mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
5372 lines
129 KiB
5372 lines
129 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
confvid.cpp
|
|
|
|
Abstract:
|
|
|
|
This module contains implementation of the video send and receive
|
|
stream implementations.
|
|
|
|
Author:
|
|
|
|
Mu Han (muhan) 15-September-1999
|
|
|
|
--*/
|
|
|
|
#include "stdafx.h"
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CStreamVideoRecv
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
CStreamVideoRecv::CStreamVideoRecv()
|
|
: CIPConfMSPStream()
|
|
{
|
|
m_szName = L"VideoRecv";
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::Init(
|
|
IN HANDLE hAddress,
|
|
IN CMSPCallBase * pMSPCall,
|
|
IN IMediaEvent * pIGraphBuilder,
|
|
IN DWORD dwMediaType,
|
|
IN TERMINAL_DIRECTION Direction
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Init our substream array and then call the base class' Init.
|
|
|
|
Arguments:
|
|
|
|
hAddress - a handle to the address, used in identify terminals.
|
|
|
|
pMSPCall - the call object that owns the stream.
|
|
|
|
pIGraphBuilder - the filter graph object.
|
|
|
|
dwMediaType - the mediatype of this stream.
|
|
|
|
Direction - the direction of this stream.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_OUTOFMEMORY
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "CStreamVideoRecvVideoSend::Init - enter"));
|
|
|
|
// initialize the stream array so that the array is not NULL.
|
|
if (!m_SubStreams.Grow())
|
|
{
|
|
LOG((MSP_TRACE, "CStreamVideoRecvVideoSend::Init - return out of memory"));
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
return CIPConfMSPStream::Init(
|
|
hAddress, pMSPCall, pIGraphBuilder,dwMediaType, Direction
|
|
);
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ShutDown()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Shut down the stream.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
CLock lock(m_lock);
|
|
|
|
// if there are terminals
|
|
BOOL fHasTerminal = FALSE;
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
fHasTerminal = TRUE;
|
|
}
|
|
|
|
// if graph is running
|
|
HRESULT hr;
|
|
OAFilterState FilterState = State_Stopped;
|
|
if (m_pIMediaControl)
|
|
{
|
|
if (FAILED (hr = m_pIMediaControl->GetState(0, &FilterState)))
|
|
{
|
|
LOG ((MSP_ERROR, "CStreamAudioRecv::ShutDown failed to query filter state. %d", hr));
|
|
FilterState = State_Stopped;
|
|
}
|
|
}
|
|
|
|
// if there are branches and configured, we need to disconnect
|
|
// the terminals and remove the branches.
|
|
if (m_Branches.GetSize() > 0)
|
|
{
|
|
// Stop the graph before disconnecting the terminals.
|
|
hr = CMSPStream::StopStream();
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"stream %ws %p failed to stop, %x", m_szName, this, hr));
|
|
return hr;
|
|
}
|
|
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
RemoveOneBranch(&m_Branches[i]);
|
|
}
|
|
m_Branches.RemoveAll();
|
|
}
|
|
|
|
// release all the substream objects.
|
|
for (int i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
m_SubStreams[i]->Release();
|
|
}
|
|
m_SubStreams.RemoveAll();
|
|
|
|
// fire event
|
|
if (fHasTerminal && FilterState == State_Running)
|
|
{
|
|
SendStreamEvent(CALL_STREAM_INACTIVE, CALL_CAUSE_LOCAL_REQUEST, 0, NULL);
|
|
}
|
|
|
|
return CIPConfMSPStream::ShutDown();
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::InternalCreateSubStream(
|
|
OUT ITSubStream ** ppSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This method creat a substream object and add it into out list.
|
|
|
|
Arguments:
|
|
ppSubStream - the memory location that will store the returned SubStream.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
E_OUTOFMEMORY
|
|
E_NOINTERFACE
|
|
|
|
--*/
|
|
{
|
|
CComObject<CSubStreamVideoRecv> * pCOMSubStream;
|
|
|
|
HRESULT hr;
|
|
|
|
hr = ::CreateCComObjectInstance(&pCOMSubStream);
|
|
|
|
if (NULL == pCOMSubStream)
|
|
{
|
|
LOG((MSP_ERROR, "could not create video recv sub stream:%x", hr));
|
|
return hr;
|
|
}
|
|
|
|
ITSubStream* pSubStream;
|
|
|
|
// get the interface pointer.
|
|
hr = pCOMSubStream->_InternalQueryInterface(
|
|
__uuidof(ITSubStream),
|
|
(void **)&pSubStream
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "Create VideoRecv Substream QueryInterface failed: %x", hr));
|
|
delete pCOMSubStream;
|
|
return hr;
|
|
}
|
|
|
|
// Initialize the object.
|
|
hr = pCOMSubStream->Init(this);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CreateMSPSubStream:call init failed: %x", hr));
|
|
pSubStream->Release();
|
|
|
|
return hr;
|
|
}
|
|
|
|
// Add the SubStream into our list of SubStreams. This takes a refcount.
|
|
if (!m_SubStreams.Add(pSubStream))
|
|
{
|
|
pSubStream->Release();
|
|
|
|
LOG((MSP_ERROR, "out of memory in adding a SubStream."));
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
// AddRef the interface pointer and return it.
|
|
pSubStream->AddRef();
|
|
*ppSubStream = pSubStream;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// ITStream method
|
|
STDMETHODIMP CStreamVideoRecv::StopStream ()
|
|
{
|
|
ENTER_FUNCTION ("CStreamVideoRecv::StopStream");
|
|
|
|
HRESULT hr;
|
|
|
|
CLock lock (m_lock);
|
|
|
|
// copy stopstream from ipconfmsp because
|
|
// we want to generate unmap event before stream inactive event
|
|
|
|
// if there is no terminal selected
|
|
if (m_Terminals.GetSize() == 0)
|
|
{
|
|
LOG((MSP_INFO, "stream %ws %p needs terminal", m_szName, this));
|
|
|
|
// Enter stopped state. (SO)
|
|
m_dwState = STRM_STOPPED;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
if (!m_fIsConfigured)
|
|
{
|
|
LOG((MSP_INFO, "stream %ws %p is not configured yet", m_szName, this));
|
|
|
|
// Enter stopped state. (SO, ST)
|
|
m_dwState = STRM_STOPPED;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// Stop the graph.
|
|
if (FAILED (hr = CMSPStream::StopStream()))
|
|
{
|
|
LOG((MSP_ERROR, "stream %ws %p failed to stop, %x", m_szName, this, hr));
|
|
return hr;
|
|
}
|
|
|
|
// check if we have filter chain
|
|
CComPtr <IFilterChain> pIFilterChain;
|
|
|
|
// Query IFilterChain
|
|
hr = m_pIMediaControl->QueryInterface(
|
|
__uuidof(IFilterChain),
|
|
(void**)&pIFilterChain
|
|
);
|
|
|
|
if (FAILED (hr) && (hr != E_NOINTERFACE))
|
|
{
|
|
LOG ((MSP_ERROR, "stream %ws %p failted to get filter chain. %x", m_szName, this, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (pIFilterChain)
|
|
{
|
|
DWORD dwSSRC = 0;
|
|
ITParticipant *pParticipant = NULL;
|
|
INT count, next;
|
|
|
|
next = m_SubStreams.GetSize ();
|
|
// generate participant leave
|
|
while ((count = next) > 0)
|
|
{
|
|
if (!((CSubStreamVideoRecv*)m_SubStreams[0])->GetCurrentParticipant (&dwSSRC, &pParticipant))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get current participant on %p", __fxName, m_SubStreams[0]));
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
pParticipant->Release ();
|
|
|
|
if (FAILED (hr = ProcessParticipantLeave (dwSSRC)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to process participant leave. ssrc=%x, hr=%x", __fxName, dwSSRC, hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
next = m_SubStreams.GetSize ();
|
|
if (next >= count)
|
|
{
|
|
// no substream was removed. we have big trouble
|
|
LOG ((MSP_ERROR, "%s: not substream was removed", __fxName));
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (!m_Branches[i].pITSubStream) continue;
|
|
|
|
if (FAILED (hr = ProcessPinUnmapEvent (
|
|
m_Branches[i].dwSSRC, m_Branches[i].pIPin)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s (%ws) failed to process pin unmap event. %x", __fxName, m_szName, hr));
|
|
}
|
|
}
|
|
}
|
|
|
|
SendStreamEvent(CALL_STREAM_INACTIVE, CALL_CAUSE_LOCAL_REQUEST, 0, NULL);
|
|
LOG((MSP_INFO, "stream %ws %p stopped", m_szName, this));
|
|
|
|
// Enter stopped state.(ST)
|
|
m_dwState = STRM_STOPPED;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// ITSubStreamControl methods, called by the app.
|
|
STDMETHODIMP CStreamVideoRecv::CreateSubStream(
|
|
IN OUT ITSubStream ** ppSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This method creates a new substream on this video receive stream. Since
|
|
the substreams are created based on the participants, this function
|
|
returns only TAPI_E_NOTSUPPORTED.
|
|
|
|
Arguments:
|
|
ppSubStream - the memory location that will store the returned SubStream.
|
|
|
|
Return Value:
|
|
|
|
TAPI_E_NOTSUPPORTED
|
|
|
|
--*/
|
|
{
|
|
return TAPI_E_NOTSUPPORTED;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::RemoveSubStream(
|
|
IN ITSubStream * pSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This method remove substream on this video receive stream. Since
|
|
the substreams are created based on the participants, this function
|
|
returns only TAPI_E_NOTSUPPORTED.
|
|
|
|
Arguments:
|
|
pSubStream - the SubStream to be removed.
|
|
|
|
Return Value:
|
|
|
|
TAPI_E_NOTSUPPORTED
|
|
--*/
|
|
{
|
|
return TAPI_E_NOTSUPPORTED;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::EnumerateSubStreams(
|
|
OUT IEnumSubStream ** ppEnumSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This method returns an enumerator of the substreams.
|
|
|
|
Arguments:
|
|
ppEnumSubStream - the memory location to store the returned pointer.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
E_POINTER
|
|
E_UNEXPECTED
|
|
E_OUTOFMEMORY
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"EnumerateSubStreams entered. ppEnumSubStream:%x", ppEnumSubStream));
|
|
|
|
//
|
|
// Check parameters.
|
|
//
|
|
|
|
if (IsBadWritePtr(ppEnumSubStream, sizeof(VOID *)))
|
|
{
|
|
LOG((MSP_ERROR, "CMSPCallBase::EnumerateSubStreams - "
|
|
"bad pointer argument - exit E_POINTER"));
|
|
|
|
return E_POINTER;
|
|
}
|
|
|
|
//
|
|
// First see if this call has been shut down.
|
|
// acquire the lock before accessing the SubStream object list.
|
|
//
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_SubStreams.GetData() == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "CMSPCallBase::EnumerateSubStreams - "
|
|
"call appears to have been shut down - exit E_UNEXPECTED"));
|
|
|
|
// This call has been shut down.
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
//
|
|
// Create an enumerator object.
|
|
//
|
|
HRESULT hr;
|
|
|
|
typedef _CopyInterface<ITSubStream> CCopy;
|
|
typedef CSafeComEnum<IEnumSubStream, &__uuidof(IEnumSubStream),
|
|
ITSubStream *, CCopy> CEnumerator;
|
|
CComObject<CEnumerator> *pEnum = NULL;
|
|
|
|
hr = ::CreateCComObjectInstance(&pEnum);
|
|
|
|
if (pEnum == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "CMSPCallBase::EnumerateSubStreams - "
|
|
"Could not create enumerator object, %x", hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// query for the __uuidof(IEnumSubStream) i/f
|
|
//
|
|
|
|
|
|
IEnumSubStream * pEnumSubStream;
|
|
hr = pEnum->_InternalQueryInterface(__uuidof(IEnumSubStream), (void**)&pEnumSubStream);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CMSPCallBase::EnumerateSubStreams - "
|
|
"query enum interface failed, %x", hr));
|
|
|
|
delete pEnum;
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Init the enumerator object. The CSafeComEnum can handle zero-sized array.
|
|
//
|
|
|
|
hr = pEnum->Init(
|
|
m_SubStreams.GetData(), // the begin itor
|
|
m_SubStreams.GetData() + m_SubStreams.GetSize(), // the end itor,
|
|
NULL, // IUnknown
|
|
AtlFlagCopy // copy the data.
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CMSPCallBase::EnumerateSubStreams - "
|
|
"init enumerator object failed, %x", hr));
|
|
|
|
pEnumSubStream->Release();
|
|
return hr;
|
|
}
|
|
|
|
LOG((MSP_TRACE, "CMSPCallBase::EnumerateSubStreams - exit S_OK"));
|
|
|
|
*ppEnumSubStream = pEnumSubStream;
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::get_SubStreams(
|
|
OUT VARIANT * pVariant
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This method returns a collection of the substreams.
|
|
|
|
Arguments:
|
|
pVariant - a variant structure.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
E_POINTER
|
|
E_UNEXPECTED
|
|
E_OUTOFMEMORY
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "CStreamVideoRecv::get_SubStreams - enter"));
|
|
|
|
//
|
|
// Check parameters.
|
|
//
|
|
|
|
if ( IsBadWritePtr(pVariant, sizeof(VARIANT) ) )
|
|
{
|
|
LOG((MSP_ERROR, "CStreamVideoRecv::get_SubStreams - "
|
|
"bad pointer argument - exit E_POINTER"));
|
|
|
|
return E_POINTER;
|
|
}
|
|
|
|
//
|
|
// See if this call has been shut down. Acquire the lock before accessing
|
|
// the SubStream object list.
|
|
//
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_SubStreams.GetData() == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "CStreamVideoRecv::get_SubStreams - "
|
|
"call appears to have been shut down - exit E_UNEXPECTED"));
|
|
|
|
// This call has been shut down.
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
//
|
|
// create the collection object - see mspcoll.h
|
|
//
|
|
|
|
typedef CTapiIfCollection< ITSubStream * > SubStreamCollection;
|
|
CComObject<SubStreamCollection> * pCollection;
|
|
HRESULT hr;
|
|
|
|
hr = ::CreateCComObjectInstance(&pCollection);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "CStreamVideoRecv::get_SubStreams - "
|
|
"can't create collection - exit 0x%08x", hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// get the Collection's IDispatch interface
|
|
//
|
|
|
|
IDispatch * pDispatch;
|
|
|
|
hr = pCollection->_InternalQueryInterface(__uuidof(IDispatch),
|
|
(void **) &pDispatch );
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "CStreamVideoRecv::get_SubStreams - "
|
|
"QI for IDispatch on collection failed - exit 0x%08x", hr));
|
|
|
|
delete pCollection;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Init the collection using an iterator -- pointers to the beginning and
|
|
// the ending element plus one.
|
|
//
|
|
|
|
hr = pCollection->Initialize( m_SubStreams.GetSize(),
|
|
m_SubStreams.GetData(),
|
|
m_SubStreams.GetData() + m_SubStreams.GetSize() );
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CStreamVideoRecv::get_SubStreams - "
|
|
"Initialize on collection failed - exit 0x%08x", hr));
|
|
|
|
pDispatch->Release();
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// put the IDispatch interface pointer into the variant
|
|
//
|
|
|
|
VariantInit(pVariant);
|
|
pVariant->vt = VT_DISPATCH;
|
|
pVariant->pdispVal = pDispatch;
|
|
|
|
LOG((MSP_TRACE, "CStreamVideoRecv::get_SubStreams - exit S_OK"));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::CheckTerminalTypeAndDirection(
|
|
IN ITTerminal * pTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check to see if the terminal is allowed on this stream. Only video
|
|
render terminal is allowed.
|
|
|
|
Arguments:
|
|
|
|
pTerminal - the terminal.
|
|
|
|
Return value:
|
|
|
|
S_OK
|
|
TAPI_E_INVALIDTERMINAL
|
|
*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoRecv.CheckTerminalTypeAndDirection"));
|
|
|
|
// check the media type of this terminal.
|
|
long lMediaType;
|
|
HRESULT hr = pTerminal->get_MediaType(&lMediaType);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal media type. %x", hr));
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
if ((DWORD)lMediaType != m_dwMediaType)
|
|
{
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
// check the direction of this terminal.
|
|
TERMINAL_DIRECTION Direction;
|
|
hr = pTerminal->get_Direction(&Direction);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal direction. %x", hr));
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
if (Direction != TD_BIDIRECTIONAL && Direction != m_Direction)
|
|
{
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::SubStreamSelectTerminal(
|
|
IN ITSubStream * pITSubStream,
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
handle terminals being selected on the sub streams. It gives the terminal
|
|
to one free branch and then sets up a mapping between the branch and the
|
|
substream, so that the participant in the substream is displayed on the
|
|
terminal selected.
|
|
|
|
Arguments:
|
|
|
|
pITSubStream - the Substream that got a terminal selected.
|
|
|
|
pITTerminal - the terminal object.
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoRecv SubStreamSelectTerminal"));
|
|
|
|
HRESULT hr;
|
|
|
|
CLock lock(m_lock);
|
|
|
|
// Call the base class's select terminal first. The terminal will be put
|
|
// into the terminal pool and a branch of filters will be created for it.
|
|
hr = CIPConfMSPStream::SelectTerminal(pITTerminal);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
// Find out which branch got the terminal.
|
|
int i;
|
|
for (i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITTerminal == pITTerminal)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
_ASSERTE(i < m_Branches.GetSize());
|
|
|
|
if (i >= m_Branches.GetSize())
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// Find out the participant on the SubStream.
|
|
ITParticipant *pITParticipant = NULL;
|
|
DWORD dwSSRC;
|
|
|
|
if ((static_cast<CSubStreamVideoRecv*>(pITSubStream))->GetCurrentParticipant(
|
|
&dwSSRC,
|
|
&pITParticipant
|
|
) == FALSE)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
pITParticipant->Release();
|
|
|
|
if (m_pIRTPDemux == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "no demux filter"));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// map the pin to this SSRC only.
|
|
hr = m_pIRTPDemux->SetMappingState(-1, m_Branches[i].pIPin, dwSSRC, TRUE);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "map SSRC %x to pin %p returned %x",
|
|
dwSSRC, m_Branches[i].pIPin, hr));
|
|
return hr;
|
|
}
|
|
|
|
_ASSERTE(m_Branches[i].pITSubStream == NULL);
|
|
|
|
pITSubStream->AddRef();
|
|
m_Branches[i].pITSubStream = pITSubStream;
|
|
m_Branches[i].dwSSRC = dwSSRC;
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ConfigureRTPFormats(
|
|
IN IBaseFilter * pIRTPFilter,
|
|
IN IStreamConfig * pIStreamConfig
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Configure the RTP filter with RTP<-->AM media type mappings.
|
|
|
|
Arguments:
|
|
|
|
pIRTPFilter - The source RTP Filter.
|
|
|
|
pIStreamConfig - The stream config interface that has the media info.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("VideoRecv::ConfigureRTPFormats");
|
|
LOG((MSP_TRACE, "%s enters", __fxName));
|
|
|
|
HRESULT hr;
|
|
|
|
CComPtr<IRtpMediaControl> pIRtpMediaControl;
|
|
hr = pIRTPFilter->QueryInterface(&pIRtpMediaControl);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s adding source filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// find the number of capabilities supported.
|
|
DWORD dwCount;
|
|
hr = pIStreamConfig->GetNumberOfCapabilities(&dwCount);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s GetNumberOfCapabilities. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
BOOL fFound = FALSE;
|
|
for (int i = dwCount - 1; i >= 0; i --)
|
|
{
|
|
// TODO, a new interface is needed to resolve RTP to MediaType.
|
|
AM_MEDIA_TYPE *pMediaType;
|
|
DWORD dwPayloadType;
|
|
|
|
hr = pIStreamConfig->GetStreamCaps(
|
|
i, &pMediaType, NULL, &dwPayloadType
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s GetStreamCaps. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
BITMAPINFOHEADER *pHeader = HEADER(pMediaType->pbFormat);
|
|
if (pHeader == NULL)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
|
|
// check the image size
|
|
if (m_Settings.fCIF)
|
|
{
|
|
if (pHeader->biWidth != CIFWIDTH)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (pHeader->biWidth != QCIFWIDTH)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
for (DWORD dw2 = 0; dw2 < m_Settings.dwNumPayloadTypes; dw2 ++)
|
|
{
|
|
if (dwPayloadType == m_Settings.PayloadTypes[dw2])
|
|
{
|
|
hr = pIRtpMediaControl->SetFormatMapping(
|
|
dwPayloadType,
|
|
90000, // default video clock rate.
|
|
pMediaType
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
|
|
LOG((MSP_ERROR, "%s SetFormatMapping. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "%s Configured payload:%d", __fxName, dwPayloadType));
|
|
}
|
|
}
|
|
}
|
|
MSPDeleteMediaType(pMediaType);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::SetUpInternalFilters()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
set up the filters used in the stream.
|
|
|
|
RTP->DECODER->Render terminal
|
|
|
|
This function only creates the RTP and demux filter and the rest of the
|
|
graph is connected in ConnectTerminal.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoRecv::SetUpInternalFilters");
|
|
LOG((MSP_TRACE, "%s entered.", __fxName));
|
|
|
|
HRESULT hr = S_OK;
|
|
|
|
if (m_pIRTPDemux == NULL)
|
|
{
|
|
CComPtr<IBaseFilter> pSourceFilter;
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
// create and add the source fitler.
|
|
if (FAILED(hr = ::AddFilter(
|
|
m_pIGraphBuilder,
|
|
__uuidof(MSRTPSourceFilter),
|
|
L"RtpSource",
|
|
&pSourceFilter)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, adding source filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED(hr = ConfigureRTPFilter(pSourceFilter)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, configure RTP source filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
if (FAILED (hr = m_pIRTPSession->QueryInterface (&pSourceFilter)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get filter from rtp session. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED (hr = m_pIGraphBuilder->AddFilter ((IBaseFilter *)pSourceFilter, L"RtpSource")))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to add filter to graph. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
// get the Demux interface pointer.
|
|
hr = pSourceFilter->QueryInterface(&m_pIRTPDemux);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s query IRtpDemux failed. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
// hr = m_pIRTPDemux->SetPinCount(m_Terminals.GetSize(), RTPDMXMODE_AUTO);
|
|
|
|
#define DEFAULT_PIN_SIZE 4
|
|
|
|
int isize = m_Terminals.GetSize();
|
|
if (isize < DEFAULT_PIN_SIZE)
|
|
{
|
|
isize = DEFAULT_PIN_SIZE;
|
|
}
|
|
|
|
hr = m_pIRTPDemux->SetPinCount(isize, RTPDMXMODE_AUTO);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s query IRtpDemux failed. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::AddOneBranch(
|
|
BRANCH * pBranch,
|
|
BOOL fFirstBranch,
|
|
BOOL fDirectRTP
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Create a new branch of filters off the demux.
|
|
|
|
Arguments:
|
|
|
|
pBranch - a pointer to a structure that remembers the info about the branch.
|
|
|
|
fFirstBranch - whether this is the first branch.
|
|
|
|
fDirectRTP - whether to output RTP directly.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoRecv::AddOneBranch");
|
|
LOG((MSP_TRACE, "%s entered.", __fxName));
|
|
|
|
HRESULT hr;
|
|
|
|
_ASSERT(m_pIRTPDemux != NULL);
|
|
|
|
CComPtr<IBaseFilter> pRTPFilter;
|
|
hr = m_pIRTPDemux->QueryInterface(
|
|
__uuidof(IBaseFilter), (void**)&pRTPFilter);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, query IBaseFilter failed, %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// Find the next output pin on the demux fitler.
|
|
CComPtr<IPin> pIPinOutput;
|
|
|
|
if (FAILED(hr = ::FindPin(
|
|
(IBaseFilter *)pRTPFilter,
|
|
(IPin**)&pIPinOutput,
|
|
PINDIR_OUTPUT
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, find free pin on demux, %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// create and add the video decoder filter.
|
|
CComPtr<IBaseFilter> pCodecFilter;
|
|
|
|
if (fDirectRTP)
|
|
{
|
|
// only create the decoder and ask questions
|
|
if (FAILED(hr = CoCreateInstance(
|
|
__uuidof(TAPIVideoDecoder),
|
|
NULL,
|
|
CLSCTX_INPROC_SERVER | CLSCTX_NO_CODE_DOWNLOAD,
|
|
__uuidof(IBaseFilter),
|
|
(void **) &pCodecFilter
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, create filter %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// create the decoder and add it into the graph.
|
|
if (FAILED(hr = ::AddFilter(
|
|
m_pIGraphBuilder,
|
|
__uuidof(TAPIVideoDecoder),
|
|
L"codec",
|
|
&pCodecFilter
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, add Codec filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
CComPtr<IPin> pIPinInput;
|
|
if (FAILED(hr = ::FindPin(pCodecFilter, &pIPinInput, PINDIR_INPUT, TRUE)))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, find input pin on pCodecFilter failed. hr=%x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (fFirstBranch)
|
|
{
|
|
CComPtr<IStreamConfig> pIStreamConfig;
|
|
|
|
hr = pIPinInput->QueryInterface(&pIStreamConfig);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, query IStreamConfig failed", __fxName));
|
|
return hr;
|
|
}
|
|
|
|
// configure the format info on the RTP filter
|
|
if (FAILED(hr = ConfigureRTPFormats(pRTPFilter, pIStreamConfig)))
|
|
{
|
|
LOG((MSP_ERROR, "%s configure RTP formats. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
if (!fDirectRTP)
|
|
{
|
|
// Connect the decoder to the output pin of the source filter.
|
|
if (FAILED(hr = ::ConnectFilters(
|
|
m_pIGraphBuilder,
|
|
(IPin *)pIPinOutput,
|
|
(IBaseFilter *)pCodecFilter
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, connect RTP filter and codec. %x", __fxName, hr));
|
|
|
|
m_pIGraphBuilder->RemoveFilter(pCodecFilter);
|
|
|
|
return hr;
|
|
}
|
|
pBranch->pCodecFilter = pCodecFilter;
|
|
pBranch->pCodecFilter->AddRef();
|
|
}
|
|
|
|
pBranch->pIPin = pIPinOutput;
|
|
pBranch->pIPin->AddRef();
|
|
|
|
// retrieve IBitrateControl
|
|
if (FAILED (hr = pIPinInput->QueryInterface (&(pBranch->pBitrateControl))))
|
|
{
|
|
LOG((MSP_ERROR, "%, query IBitrateControl failed. %x", __fxName, hr));
|
|
pBranch->pBitrateControl = NULL;
|
|
// return hr;
|
|
}
|
|
|
|
LOG((MSP_TRACE, "%s, AddOneBranch exits ok.", __fxName));
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::RemoveOneBranch(
|
|
BRANCH * pBranch
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Remove all the filters in a branch and release all the pointers.
|
|
the caller of this function should not use any member of this branch
|
|
after this function call.
|
|
|
|
Arguments:
|
|
|
|
pBranch - a pointer to a structure that has the info about the branch.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("VideoRecv::RemoveOneBranch");
|
|
LOG((MSP_TRACE, "%s entered", __fxName));
|
|
|
|
if (pBranch->pBitrateControl)
|
|
{
|
|
pBranch->pBitrateControl->Release();
|
|
}
|
|
|
|
if (pBranch->pIPin)
|
|
{
|
|
pBranch->pIPin->Release();
|
|
}
|
|
|
|
if (pBranch->pCodecFilter)
|
|
{
|
|
|
|
// #ifdef DYNGRAPH
|
|
HRESULT hr;
|
|
OAFilterState FilterState;
|
|
CComPtr <IFilterChain> pIFilterChain;
|
|
|
|
// Query IFilterChain
|
|
hr = m_pIMediaControl->QueryInterface(
|
|
__uuidof(IFilterChain),
|
|
(void**)&pIFilterChain
|
|
);
|
|
|
|
if (FAILED (hr) && (hr != E_NOINTERFACE))
|
|
{
|
|
LOG ((MSP_ERROR, "stream %ws %p failted to get filter chain. %x", m_szName, this, hr));
|
|
// return hr;
|
|
}
|
|
|
|
if (pIFilterChain)
|
|
{
|
|
hr = m_pIMediaControl->GetState(0, &FilterState);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s get filter graph state failed, %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
// stop the chain before removing filters.
|
|
if (FilterState == State_Running)
|
|
{
|
|
// stop the chain if the graph is in running state.
|
|
hr = pIFilterChain->StopChain(pBranch->pCodecFilter, NULL);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s stop chain failed. hr=%x", __fxName, hr));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// #endif
|
|
|
|
m_pIGraphBuilder->RemoveFilter(pBranch->pCodecFilter);
|
|
pBranch->pCodecFilter->Release();
|
|
}
|
|
|
|
if (pBranch->pITTerminal)
|
|
{
|
|
// get the terminal control interface.
|
|
CComQIPtr<ITTerminalControl, &__uuidof(ITTerminalControl)>
|
|
pTerminal(pBranch->pITTerminal);
|
|
|
|
_ASSERTE(pTerminal != NULL);
|
|
|
|
if (pTerminal != NULL)
|
|
{
|
|
HRESULT hr = pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
LOG((MSP_TRACE,
|
|
"%s, terminal %p is disonnected. hr:%x",
|
|
__fxName, pBranch->pITTerminal, hr));
|
|
}
|
|
pBranch->pITTerminal->Release();
|
|
}
|
|
|
|
if (pBranch->pITSubStream)
|
|
{
|
|
((CSubStreamVideoRecv*)pBranch->pITSubStream)->
|
|
ClearCurrentTerminal();
|
|
pBranch->pITSubStream->Release();
|
|
}
|
|
|
|
LOG((MSP_TRACE, "%s, RemoveOneBranch exits ok.", __fxName));
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ConnectPinToTerminal(
|
|
IN IPin * pOutputPin,
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Connect the codec filter to the render filter inside the terminal.
|
|
|
|
Arguments:
|
|
|
|
pOutputPin - The last pin before the terminal.
|
|
|
|
pITTerminal - the terminal object.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
// get the terminal control interface.
|
|
CComQIPtr<ITTerminalControl, &__uuidof(ITTerminalControl)>
|
|
pTerminal(pITTerminal);
|
|
if (pTerminal == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "can't get Terminal Control interface"));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, E_NOINTERFACE, pITTerminal);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
|
|
const DWORD MAXPINS = 8;
|
|
|
|
DWORD dwNumPins = MAXPINS;
|
|
IPin * Pins[MAXPINS];
|
|
|
|
HRESULT hr = pTerminal->ConnectTerminal(
|
|
m_pIGraphBuilder, TD_RENDER, &dwNumPins, Pins
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't connect to terminal, %x", hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_CONNECT_FAIL, hr, pITTerminal);
|
|
|
|
return hr;
|
|
}
|
|
|
|
// the number of pins should never be 0.
|
|
if (dwNumPins == 0)
|
|
{
|
|
LOG((MSP_ERROR, "terminal has no pins."));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
if (IsBadReadPtr (Pins, dwNumPins * sizeof (IPin*)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin array"));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL, CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_POINTER;
|
|
}
|
|
|
|
for (DWORD i = 0; i < dwNumPins; i++)
|
|
{
|
|
if (IsBadReadPtr (Pins[i], sizeof (IPin)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin. # %d", i));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL, CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_POINTER;
|
|
}
|
|
}
|
|
|
|
// Connect the codec filter to the video render terminal.
|
|
hr = ::ConnectFilters(
|
|
m_pIGraphBuilder,
|
|
(IBaseFilter *)pOutputPin,
|
|
(IPin *)Pins[0],
|
|
FALSE // use Connect instead of ConnectDirect.
|
|
);
|
|
|
|
// release the refcounts on the pins.
|
|
for (DWORD i = 0; i < dwNumPins; i ++)
|
|
{
|
|
Pins[i]->Release();
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "connect the pin to the terminal. %x", hr));
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return hr;
|
|
|
|
}
|
|
|
|
//
|
|
// Now we are actually connected. Update our state and perform postconnection
|
|
// (ignore postconnection error code).
|
|
//
|
|
pTerminal->CompleteConnectTerminal();
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ConnectTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
connect video render terminal.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - The terminal to be connected.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("VideoRecv::ConnectTerminal");
|
|
LOG((MSP_TRACE, "%s enters, pTerminal %p", __fxName, pITTerminal));
|
|
|
|
HRESULT hr;
|
|
|
|
// #ifdef DYNGRAPH
|
|
OAFilterState FilterState;
|
|
CComPtr <IFilterChain> pIFilterChain;
|
|
|
|
// Query IFilterChain
|
|
hr = m_pIMediaControl->QueryInterface(
|
|
__uuidof(IFilterChain),
|
|
(void**)&pIFilterChain
|
|
);
|
|
|
|
if (FAILED (hr) && (hr != E_NOINTERFACE))
|
|
{
|
|
LOG ((MSP_ERROR, "stream %ws %p failted to get filter chain. %x", m_szName, this, hr));
|
|
return hr;
|
|
}
|
|
|
|
hr = m_pIMediaControl->GetState(0, &FilterState);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s get filter graph state failed, %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
// #endif
|
|
|
|
hr = SetUpInternalFilters();
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s Set up internal filter failed, %x", __fxName, hr));
|
|
|
|
CleanUpFilters();
|
|
|
|
return hr;
|
|
}
|
|
|
|
// get the terminal control interface.
|
|
CComQIPtr<ITTerminalControl, &__uuidof(ITTerminalControl)>
|
|
pTerminal(pITTerminal);
|
|
if (pTerminal == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "can't get Terminal Control interface"));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, E_NOINTERFACE, pITTerminal);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
const DWORD MAXPINS = 8;
|
|
|
|
DWORD dwNumPins = MAXPINS;
|
|
IPin * Pins[MAXPINS];
|
|
|
|
hr = pTerminal->ConnectTerminal(
|
|
m_pIGraphBuilder, TD_RENDER, &dwNumPins, Pins
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't connect to terminal, %x", hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_CONNECT_FAIL, hr, pITTerminal);
|
|
|
|
return hr;
|
|
}
|
|
|
|
// the number of pins should never be 0.
|
|
if (dwNumPins == 0)
|
|
{
|
|
LOG((MSP_ERROR, "terminal has no pins."));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
if (IsBadReadPtr (Pins, dwNumPins * sizeof (IPin*)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin array"));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL, CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_POINTER;
|
|
}
|
|
|
|
DWORD i;
|
|
for (i = 0; i < dwNumPins; i++)
|
|
{
|
|
if (IsBadReadPtr (Pins[i], sizeof (IPin)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin. # %d", i));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL, CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return E_POINTER;
|
|
}
|
|
}
|
|
|
|
// check the media type supported by input pin on terminal
|
|
BOOL fDirectRTP = FALSE;
|
|
if (S_OK == ::PinSupportsMediaType (
|
|
Pins[0], __uuidof(MEDIATYPE_RTP_Single_Stream)
|
|
))
|
|
{
|
|
fDirectRTP = TRUE;
|
|
}
|
|
|
|
// first create the branch structure needed before the terminal.
|
|
BRANCH aBranch;
|
|
ZeroMemory(&aBranch, sizeof BRANCH);
|
|
|
|
hr = AddOneBranch(&aBranch, (m_Branches.GetSize() == 0), fDirectRTP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s Set up a new decode branch failed, %x", __fxName, hr));
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
return hr;
|
|
}
|
|
|
|
CComPtr <IPin> pOutputPin;
|
|
|
|
if (fDirectRTP)
|
|
{
|
|
// connect the RTP output pin to the terminal's input pin.
|
|
hr = m_pIGraphBuilder->ConnectDirect(aBranch.pIPin, Pins[0], NULL);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s connecting codec to terminal failed, %x", __fxName, hr));
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
goto cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// connect the codec to the terminal
|
|
hr = ConnectFilters(m_pIGraphBuilder, aBranch.pCodecFilter, Pins[0]);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s connecting codec to terminal failed, %x", __fxName, hr));
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
// #ifdef DYNGRAPH
|
|
if (pIFilterChain)
|
|
{
|
|
if (FilterState == State_Running)
|
|
{
|
|
if (fDirectRTP)
|
|
{
|
|
hr = E_UNEXPECTED;
|
|
LOG((MSP_ERROR, "%s can't support this. %x", __fxName, hr));
|
|
goto cleanup;
|
|
}
|
|
|
|
hr = pIFilterChain->StartChain(aBranch.pCodecFilter, NULL);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s start chain failed. hr=%x", __fxName, hr));
|
|
goto cleanup;
|
|
}
|
|
}
|
|
}
|
|
// #endif
|
|
|
|
pITTerminal->AddRef();
|
|
aBranch.pITTerminal = pITTerminal;
|
|
|
|
if (!m_Branches.Add(aBranch))
|
|
{
|
|
LOG((MSP_ERROR, "%s out of mem.", __fxName));
|
|
hr = E_OUTOFMEMORY;
|
|
goto cleanup;
|
|
}
|
|
|
|
// release the refcounts on the pins.
|
|
for (i = 0; i < dwNumPins; i ++)
|
|
{
|
|
Pins[i]->Release();
|
|
}
|
|
|
|
//
|
|
// Now we are actually connected. Update our state and perform postconnection
|
|
// (ignore postconnection error code).
|
|
//
|
|
pTerminal->CompleteConnectTerminal();
|
|
|
|
return S_OK;
|
|
|
|
cleanup:
|
|
|
|
// release the refcounts on the pins.
|
|
for (i = 0; i < dwNumPins; i ++)
|
|
{
|
|
Pins[i]->Release();
|
|
}
|
|
|
|
// remove the added filters from the graph and disconnect the terminal.
|
|
RemoveOneBranch(&aBranch);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::DisconnectTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Disconnect a terminal. It will remove its filters from the graph and
|
|
also release its references to the graph. A branch of filters is also
|
|
released.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - the terminal.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITTerminal == pITTerminal)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (i < m_Branches.GetSize())
|
|
{
|
|
RemoveOneBranch(&m_Branches[i]);
|
|
m_Branches.RemoveAt(i);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::SetUpFilters()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Insert filters into the graph and connect to the terminals.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoRecv.SetUpFilters"));
|
|
|
|
HRESULT hr = SetUpInternalFilters();
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "Set up internal filter failed, %x", hr));
|
|
|
|
CleanUpFilters();
|
|
|
|
return hr;
|
|
}
|
|
|
|
for (int i = 0; i < m_Terminals.GetSize(); i ++)
|
|
{
|
|
HRESULT hr = ConnectTerminal(m_Terminals[i]);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// ITParticipantSubStreamControl methods, called by the app.
|
|
STDMETHODIMP CStreamVideoRecv::get_SubStreamFromParticipant(
|
|
IN ITParticipant * pITParticipant,
|
|
OUT ITSubStream ** ppITSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find out which substream is rendering the participant.
|
|
|
|
Arguments:
|
|
|
|
pITParticipant - the participant.
|
|
|
|
ppITSubStream - the returned sub stream.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
TAPI_E_NOITEMS,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "get substream from participant:%p", pITParticipant));
|
|
|
|
if (IsBadWritePtr(ppITSubStream, sizeof(VOID *)))
|
|
{
|
|
LOG((MSP_ERROR, "ppITSubStream is a bad pointer"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
CLock lock(m_lock);
|
|
|
|
ITSubStream * pITSubStream = NULL;
|
|
|
|
// find out which substream has the participant.
|
|
for (int i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
ITParticipant *pTempParticipant = NULL;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pTempParticipant
|
|
);
|
|
|
|
_ASSERTE(pTempParticipant != NULL);
|
|
|
|
pTempParticipant->Release(); // we dont' need the ref here.
|
|
|
|
if (pITParticipant == pTempParticipant)
|
|
{
|
|
pITSubStream = m_SubStreams[i];
|
|
pITSubStream->AddRef();
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pITSubStream == NULL)
|
|
{
|
|
return TAPI_E_NOITEMS;
|
|
}
|
|
|
|
*ppITSubStream = pITSubStream;
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::get_ParticipantFromSubStream(
|
|
IN ITSubStream * pITSubStream,
|
|
OUT ITParticipant ** ppITParticipant
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find out which participant the substream is rendering.
|
|
|
|
Arguments:
|
|
|
|
pITSubStream - the sub stream.
|
|
|
|
ppITParticipant - the returned participant
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
TAPI_E_NOITEMS,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "get participant from substream:%p", pITSubStream));
|
|
|
|
if (IsBadWritePtr(ppITParticipant, sizeof(VOID *)))
|
|
{
|
|
LOG((MSP_ERROR, "ppITParticipant is a bad pointer"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
CLock lock(m_lock);
|
|
|
|
int i;
|
|
|
|
// check to see if the substream is in our list.
|
|
if ((i = m_SubStreams.Find(pITSubStream)) < 0)
|
|
{
|
|
LOG((MSP_ERROR, "wrong SubStream handle %p", pITSubStream));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
ITParticipant *pITParticipant;
|
|
DWORD dwSSRC;
|
|
|
|
if (((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pITParticipant
|
|
) == FALSE)
|
|
{
|
|
return TAPI_E_NOITEMS;
|
|
}
|
|
|
|
*ppITParticipant = pITParticipant;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::SwitchTerminalToSubStream(
|
|
IN ITTerminal * pITTerminal,
|
|
IN ITSubStream * pITSubStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Switch terminal to a substream to display the participant that is on the
|
|
substream.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - the terminal.
|
|
|
|
pITSubStream - the sub stream.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_INVALIDARG,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "switch terminal %p to substream:%p",
|
|
pITTerminal, pITSubStream));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPDemux == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the demux filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// first, find out which branch has the terminal now.
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITTerminal == pITTerminal)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (i >= m_Branches.GetSize())
|
|
{
|
|
LOG((MSP_TRACE, "terminal %p doesn't exist", pITTerminal));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
// second, find out if the substream exists.
|
|
if (m_SubStreams.Find(pITSubStream) < 0)
|
|
{
|
|
LOG((MSP_TRACE, "SubStream %p doesn't exist", pITSubStream));
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
|
|
// thrid, find the participant on the substream and configure the demux
|
|
// filter to render the participant on the chosen branch.
|
|
ITParticipant *pITParticipant = NULL;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->GetCurrentParticipant(
|
|
&dwSSRC, &pITParticipant
|
|
) ;
|
|
|
|
_ASSERTE(pITParticipant != NULL);
|
|
|
|
// we don't need the reference here.
|
|
pITParticipant->Release();
|
|
|
|
// map the pin to this SSRC only.
|
|
HRESULT hr = m_pIRTPDemux->SetMappingState(
|
|
-1, m_Branches[i].pIPin, dwSSRC, TRUE
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "map SSRC %x to pin %p returned %x",
|
|
dwSSRC, m_Branches[i].pIPin, hr));
|
|
return hr;
|
|
}
|
|
|
|
DWORD dwOldSSRC = 0;
|
|
|
|
// Finally, set up the mappings among the branch, the substream and
|
|
// the terminal
|
|
|
|
// release the refcount on the old branch that the substream was on.
|
|
for (int j = 0; j < m_Branches.GetSize(); j ++)
|
|
{
|
|
if (m_Branches[j].pITSubStream == pITSubStream)
|
|
{
|
|
m_Branches[j].pITSubStream->Release();
|
|
m_Branches[j].pITSubStream = NULL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (m_Branches[i].pITSubStream != NULL)
|
|
{
|
|
((CSubStreamVideoRecv*)m_Branches[i].pITSubStream)->
|
|
ClearCurrentTerminal();
|
|
|
|
m_Branches[i].pITSubStream->Release();
|
|
dwOldSSRC = m_Branches[i].dwSSRC;
|
|
}
|
|
|
|
pITSubStream->AddRef();
|
|
m_Branches[i].pITSubStream = pITSubStream;
|
|
m_Branches[i].dwSSRC = dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->ClearCurrentTerminal();
|
|
((CSubStreamVideoRecv*)pITSubStream)->SetCurrentTerminal(
|
|
m_Branches[i].pITTerminal
|
|
);
|
|
|
|
|
|
// After all the steps, we still have to change QOS reservation.
|
|
if (dwOldSSRC != 0)
|
|
{
|
|
// cancel QOS for the old participant.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwOldSSRC, FALSE)))
|
|
{
|
|
LOG((MSP_ERROR, "disabling QOS for %x. hr:%x", dwOldSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "disabled video QOS for %x.", dwOldSSRC));
|
|
}
|
|
}
|
|
|
|
// reserve QOS for the new participant.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, TRUE)))
|
|
{
|
|
LOG((MSP_ERROR, "enabling video QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "enabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT CStreamVideoRecv::ProcessTalkingEvent(
|
|
IN DWORD dwSSRC
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A sender has just joined. A substream needs to be created for the
|
|
participant.
|
|
|
|
A pin mapped event might have happended when we didn't have the
|
|
participant's name so it was queued in a list. Now that we have a new
|
|
participant, let's check if this is the same participant. If it is,
|
|
we complete the pin mapped event by sending the app an notification.
|
|
|
|
Arguments:
|
|
|
|
dwSSRC - the SSRC of the participant.
|
|
|
|
pITParticipant - the participant object.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoRecv::ProcessTalkingEvent");
|
|
|
|
LOG((MSP_TRACE, "%s entered. %x", __fxName, dwSSRC));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the network filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// first find out if this participant object exists.
|
|
ITParticipant * pITParticipant = NULL;
|
|
|
|
int i;
|
|
// find the SSRC in our participant list.
|
|
for (i = 0; i < m_Participants.GetSize(); i ++)
|
|
{
|
|
if (((CParticipant *)m_Participants[i])->
|
|
HasSSRC((ITStream *)this, dwSSRC))
|
|
{
|
|
pITParticipant = m_Participants[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if the participant is not there yet, just return. It will be checked
|
|
// later when CName is available.
|
|
if (!pITParticipant)
|
|
{
|
|
LOG((MSP_TRACE, "%s participant not exist", __fxName));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
// Find out if a substream has been created for this participant when we
|
|
// processed PinMapped event and receiver reports.
|
|
for (i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
ITParticipant *pTempParticipant;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pTempParticipant
|
|
);
|
|
|
|
_ASSERTE(pTempParticipant != NULL);
|
|
|
|
pTempParticipant->Release(); // we dont' need the ref here.
|
|
|
|
if (pITParticipant == pTempParticipant)
|
|
{
|
|
// the participant has been created.
|
|
return S_OK;
|
|
}
|
|
}
|
|
|
|
ITSubStream * pITSubStream;
|
|
HRESULT hr = InternalCreateSubStream(&pITSubStream);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%ls can't create a SubStream, %x", m_szName, hr));
|
|
return hr;
|
|
}
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->SetCurrentParticipant(
|
|
dwSSRC, pITParticipant
|
|
);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_NEW_SUBSTREAM,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
// look at the pending SSRC list and find out if this report
|
|
// fits in the list.
|
|
IPin *pIPin = NULL;
|
|
|
|
for (i = 0; i < m_PinMappedEvents.GetSize(); i ++)
|
|
{
|
|
if (m_PinMappedEvents[i].dwSSRC == dwSSRC)
|
|
{
|
|
pIPin = m_PinMappedEvents[i].pIPin;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!pIPin)
|
|
{
|
|
// the SSRC is not in the list of pending PinMappedEvents.
|
|
LOG((MSP_TRACE, "the SSRC %x is not in the pending list", dwSSRC));
|
|
|
|
pITSubStream->Release();
|
|
|
|
return S_OK;;
|
|
}
|
|
|
|
// get rid of the peding event.
|
|
m_PinMappedEvents.RemoveAt(i);
|
|
|
|
// reserve QOS since we are rendering this sender.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, TRUE)))
|
|
{
|
|
LOG((MSP_ERROR, "enabling video QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "enabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
// tell the app about the newly mapped sender.
|
|
for (i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pIPin == pIPin)
|
|
{
|
|
if (m_Branches[i].pITSubStream != NULL)
|
|
{
|
|
((CSubStreamVideoRecv*)m_Branches[i].pITSubStream)
|
|
->ClearCurrentTerminal();
|
|
|
|
m_Branches[i].pITSubStream->Release();
|
|
}
|
|
|
|
m_Branches[i].dwSSRC = dwSSRC;
|
|
m_Branches[i].pITSubStream = pITSubStream;
|
|
pITSubStream->AddRef();
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->
|
|
SetCurrentTerminal(m_Branches[i].pITTerminal);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_MAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
pITSubStream->Release();
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ProcessSilentEvent(
|
|
IN DWORD dwSSRC
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
When participant left the session, remove the stream from the participant
|
|
object's list of streams. If all streams are removed, remove the
|
|
participant from the call object's list too.
|
|
|
|
Arguments:
|
|
|
|
dwSSRC - the SSRC of the participant left.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "%ls ProcessSilentEvent, SSRC: %x", m_szName, dwSSRC));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the network filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// first find out if this participant object exists.
|
|
ITParticipant * pITParticipant = NULL;
|
|
|
|
int i;
|
|
// find the SSRC in our participant list.
|
|
for (i = 0; i < m_Participants.GetSize(); i ++)
|
|
{
|
|
if (((CParticipant *)m_Participants[i])->
|
|
HasSSRC((ITStream *)this, dwSSRC))
|
|
{
|
|
pITParticipant = m_Participants[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if the participant is not there, just return.
|
|
if (!pITParticipant)
|
|
{
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
HRESULT hr;
|
|
// cancel QOS for this participant.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, FALSE)))
|
|
{
|
|
LOG((MSP_ERROR, "disabling QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "disabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
// find out which substream is going away.
|
|
ITSubStream * pITSubStream = NULL;
|
|
for (i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
// Find out the participant on the SubStream.
|
|
ITParticipant *pTempParticipant;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pTempParticipant
|
|
);
|
|
|
|
_ASSERTE(pTempParticipant != NULL);
|
|
|
|
pTempParticipant->Release(); // we dont' need the ref here.
|
|
|
|
if (pTempParticipant == pITParticipant)
|
|
{
|
|
pITSubStream = m_SubStreams[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pITSubStream)
|
|
{
|
|
// remove the mapping if the substream was mapped to a branch.
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITSubStream == pITSubStream)
|
|
{
|
|
m_Branches[i].pITSubStream->Release();
|
|
m_Branches[i].pITSubStream = NULL;
|
|
m_Branches[i].dwSSRC = 0;
|
|
|
|
// fire an event to tell the app that the substream is not used.
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_UNMAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_REMOVED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
if (m_SubStreams.Remove(pITSubStream))
|
|
{
|
|
pITSubStream->Release();
|
|
}
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::NewParticipantPostProcess(
|
|
IN DWORD dwSSRC,
|
|
IN ITParticipant *pITParticipant
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A pin mapped event might have happended when we didn't have the
|
|
participant's name so it was queued in a list. Now that we have a new
|
|
participant, let's check if this is the same participant. If it is,
|
|
we complete the pin mapped event by creating a substream and send
|
|
the app a notification.
|
|
|
|
Arguments:
|
|
|
|
dwSSRC - the SSRC of the participant.
|
|
|
|
pITParticipant - the participant object.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "%ls Check pending mapped event, dwSSRC: %x", m_szName, dwSSRC));
|
|
|
|
// look at the pending SSRC list and find out if this report
|
|
// fits in the list.
|
|
IPin *pIPin = NULL;
|
|
|
|
for (int i = 0; i < m_PinMappedEvents.GetSize(); i ++)
|
|
{
|
|
if (m_PinMappedEvents[i].dwSSRC == dwSSRC)
|
|
{
|
|
pIPin = m_PinMappedEvents[i].pIPin;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!pIPin)
|
|
{
|
|
// the SSRC is not in the list of pending PinMappedEvents.
|
|
LOG((MSP_TRACE, "the SSRC %x is not in the pending list", dwSSRC));
|
|
|
|
// Find out if the participant is talking.
|
|
// if (ParticipantIsNotTalking)
|
|
{
|
|
return S_OK;;
|
|
}
|
|
}
|
|
|
|
ITSubStream * pITSubStream;
|
|
HRESULT hr = InternalCreateSubStream(&pITSubStream);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%ls can't create a SubStream, %x", m_szName, hr));
|
|
return hr;
|
|
}
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->SetCurrentParticipant(
|
|
dwSSRC, pITParticipant
|
|
);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_NEW_SUBSTREAM,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
if (pIPin)
|
|
{
|
|
// we got here because we had a pending mapped event.
|
|
|
|
// get rid of the peding event.
|
|
m_PinMappedEvents.RemoveAt(i);
|
|
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, TRUE)))
|
|
{
|
|
LOG((MSP_ERROR, "enabling video QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "enabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
// Now we get the participant, the substream, and the pin. Establish a mapping
|
|
// between the decoding branch and the substream.
|
|
for (i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pIPin == pIPin)
|
|
{
|
|
if (m_Branches[i].pITSubStream != NULL)
|
|
{
|
|
((CSubStreamVideoRecv*)m_Branches[i].pITSubStream)
|
|
->ClearCurrentTerminal();
|
|
|
|
m_Branches[i].pITSubStream->Release();
|
|
}
|
|
|
|
m_Branches[i].dwSSRC = dwSSRC;
|
|
m_Branches[i].pITSubStream = pITSubStream;
|
|
pITSubStream->AddRef();
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->
|
|
SetCurrentTerminal(m_Branches[i].pITTerminal);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_MAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
break;
|
|
}
|
|
}
|
|
_ASSERT(i < m_Branches.GetSize());
|
|
}
|
|
|
|
pITSubStream->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ProcessPinMappedEvent(
|
|
IN DWORD dwSSRC,
|
|
IN IPin * pIPin
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A pin just got a new SSRC mapped to it. If the participant doesn't exist,
|
|
put the event in a pending queue and wait for a RTCP report that has the
|
|
participant's name. If the participant exists, check to see if a SubStream
|
|
has been created for the stream. If not, a SubStream is created. Then a
|
|
Particiapnt substream event is fired.
|
|
|
|
Arguments:
|
|
|
|
dwSSRC - the SSRC of the participant.
|
|
|
|
pIPin - the output pin of the demux filter that just got a new SSRC.
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "%ls Process pin mapped event, pIPin: %p", m_szName, pIPin));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPDemux == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the demux filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
for (int iBranch = 0; iBranch < m_Branches.GetSize(); iBranch ++)
|
|
{
|
|
if (m_Branches[iBranch].pIPin == pIPin)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
LOG((MSP_INFO, "Branch %d has the pin", iBranch));
|
|
|
|
if (iBranch >= m_Branches.GetSize())
|
|
{
|
|
LOG((MSP_ERROR, "Wrong pin is mapped. %p", pIPin));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// sometimes we might get a mapped event for branches that are still
|
|
// in use.
|
|
if (m_Branches[iBranch].pITSubStream != NULL)
|
|
{
|
|
LOG((MSP_ERROR, "ProcessPinMappedEvent: Branch still in use"));
|
|
|
|
// sometimes we might get duplicated map events
|
|
if (m_Branches[iBranch].dwSSRC == dwSSRC)
|
|
{
|
|
// LOG((MSP_WARNING, "ProcessPinMappedEvent: Branch still in use"));
|
|
|
|
LOG((MSP_ERROR, "The same pin mapped twice. %p", pIPin));
|
|
return E_UNEXPECTED;
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_ERROR, "The branch is in use. Cleaning up."));
|
|
|
|
((CSubStreamVideoRecv*)m_Branches[iBranch].pITSubStream)->
|
|
ClearCurrentTerminal();
|
|
|
|
// cancel QOS for the old participant.
|
|
m_pIRTPSession->SetQosState(m_Branches[iBranch].dwSSRC, FALSE);
|
|
|
|
m_Branches[iBranch].pITSubStream->Release();
|
|
m_Branches[iBranch].pITSubStream = NULL;
|
|
m_Branches[iBranch].dwSSRC = 0;
|
|
}
|
|
}
|
|
|
|
ITParticipant * pITParticipant = NULL;
|
|
|
|
// find the SSRC in our participant list.
|
|
for (int i = 0; i < m_Participants.GetSize(); i ++)
|
|
{
|
|
if (((CParticipant *)m_Participants[i])->
|
|
HasSSRC((ITStream *)this, dwSSRC))
|
|
{
|
|
pITParticipant = m_Participants[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if the participant is not there yet, put the event in a queue and it
|
|
// will be fired when we have the CName for the participant.
|
|
if (!pITParticipant)
|
|
{
|
|
LOG((MSP_INFO, "can't find a participant that has SSRC %x", dwSSRC));
|
|
|
|
PINMAPEVENT Event;
|
|
Event.pIPin = pIPin;
|
|
Event.dwSSRC = dwSSRC;
|
|
|
|
m_PinMappedEvents.Add(Event);
|
|
|
|
LOG((MSP_INFO, "added the event to pending list, new list size:%d",
|
|
m_PinMappedEvents.GetSize()));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT hr;
|
|
|
|
// Enable QOS for the participant since it is being rendered.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, TRUE)))
|
|
{
|
|
LOG((MSP_ERROR, "enabling vidoe QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "enabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
// Find out if a substream has been created for this participant who might
|
|
// have been a receiver only and hasn't got a substream.
|
|
ITSubStream * pITSubStream = NULL;
|
|
for (i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
ITParticipant *pTempParticipant;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pTempParticipant
|
|
);
|
|
|
|
_ASSERTE(pTempParticipant != NULL);
|
|
|
|
pTempParticipant->Release(); // we dont' need the ref here.
|
|
|
|
if (pITParticipant == pTempParticipant)
|
|
{
|
|
pITSubStream = m_SubStreams[i];
|
|
pITSubStream->AddRef();
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pITSubStream == NULL)
|
|
{
|
|
// we need to create a substream for this participant since he has
|
|
// started sending.
|
|
hr = InternalCreateSubStream(&pITSubStream);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%ls can't create a SubStream, %x", m_szName, hr));
|
|
return hr;
|
|
}
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->SetCurrentParticipant(
|
|
dwSSRC, pITParticipant
|
|
);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_NEW_SUBSTREAM,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
}
|
|
|
|
if (((CSubStreamVideoRecv*)pITSubStream)->ClearCurrentTerminal())
|
|
{
|
|
// The substrem has a terminal before. This is an error.
|
|
// _ASSERT(!"SubStream has a terminal already");
|
|
|
|
LOG((MSP_ERROR, "SubStream %p has already got a terminal", pITSubStream));
|
|
|
|
// remove the mapping if the substream was mapped to a branch.
|
|
for (i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITSubStream == pITSubStream)
|
|
{
|
|
// cancel QOS for the old participant.
|
|
m_pIRTPSession->SetQosState(m_Branches[i].dwSSRC, FALSE);
|
|
|
|
m_Branches[i].pITSubStream->Release();
|
|
m_Branches[i].pITSubStream = NULL;
|
|
m_Branches[i].dwSSRC = 0;
|
|
|
|
LOG((MSP_ERROR, "SubStream %p was mapped to branch %d", i));
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Now we get the participant, the substream, and the pin. Establish a mapping
|
|
// between the decoding branch and the substream.
|
|
m_Branches[iBranch].dwSSRC = dwSSRC;
|
|
m_Branches[iBranch].pITSubStream = pITSubStream;
|
|
pITSubStream->AddRef();
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->
|
|
SetCurrentTerminal(m_Branches[iBranch].pITTerminal);
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_MAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
pITSubStream->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ProcessPinUnmapEvent(
|
|
IN DWORD dwSSRCOnPin,
|
|
IN IPin * pIPin
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
A pin just got unmapped by the demux. Notify the app which substream
|
|
is not going to have any data.
|
|
|
|
Arguments:
|
|
|
|
dwSSRCOnPin - the SSRC of the participant.
|
|
|
|
pIPin - the output pin of the demux filter
|
|
|
|
Return Value:
|
|
|
|
S_OK,
|
|
E_UNEXPECTED
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "%ls Proces pin unmapped event, pIPin: %p", m_szName, pIPin));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the demux filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// look at the pending SSRC list and find out if the pin is in the
|
|
// pending list.
|
|
for (int i = 0; i < m_PinMappedEvents.GetSize(); i ++)
|
|
{
|
|
if (m_PinMappedEvents[i].pIPin == pIPin)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if the pin is in the pending list, just remove it.
|
|
if (i < m_PinMappedEvents.GetSize())
|
|
{
|
|
m_PinMappedEvents.RemoveAt(i);
|
|
return S_OK;
|
|
}
|
|
|
|
// find out which substream got unmapped.
|
|
ITSubStream * pITSubStream = NULL;
|
|
for (i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pIPin == pIPin)
|
|
{
|
|
pITSubStream = m_Branches[i].pITSubStream;
|
|
|
|
if (pITSubStream)
|
|
{
|
|
// Don't release the ref until the end of this function.
|
|
m_Branches[i].pITSubStream = NULL;
|
|
m_Branches[i].dwSSRC = 0;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!pITSubStream)
|
|
{
|
|
LOG((MSP_ERROR, "can't find a substream that got unmapped."));
|
|
return TAPI_E_NOITEMS;
|
|
}
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->ClearCurrentTerminal();
|
|
|
|
ITParticipant *pITParticipant = NULL;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)pITSubStream)->GetCurrentParticipant(
|
|
&dwSSRC, &pITParticipant
|
|
) ;
|
|
|
|
_ASSERTE(pITParticipant != NULL);
|
|
|
|
if (dwSSRCOnPin != dwSSRC)
|
|
{
|
|
LOG((MSP_ERROR, "SSRCs don't match, pin's SSRC:%x, mine:%x",
|
|
dwSSRCOnPin, dwSSRC));
|
|
}
|
|
|
|
if (pITParticipant != NULL)
|
|
{
|
|
// fire an event to tell the app that the substream is not used.
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_UNMAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
pITParticipant->Release();
|
|
|
|
// cancel QOS for this participant.
|
|
HRESULT hr = m_pIRTPSession->SetQosState(dwSSRC, FALSE);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "disabling QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "disabled video QOS for %x.", dwSSRC));
|
|
}
|
|
}
|
|
|
|
pITSubStream->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoRecv::ProcessParticipantLeave(
|
|
IN DWORD dwSSRC
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
When participant left the session, remove the stream from the participant
|
|
object's list of streams. If all streams are removed, remove the
|
|
participant from the call object's list too.
|
|
|
|
Arguments:
|
|
|
|
dwSSRC - the SSRC of the participant left.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "%ls ProcessParticipantLeave, SSRC: %x", m_szName, dwSSRC));
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "the network filter doesn't exist."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
CParticipant *pParticipant;
|
|
BOOL fLast = FALSE;
|
|
|
|
HRESULT hr = E_FAIL;
|
|
|
|
// first try to find the SSRC in our participant list.
|
|
for (int iParticipant = 0;
|
|
iParticipant < m_Participants.GetSize(); iParticipant ++)
|
|
{
|
|
pParticipant = (CParticipant *)m_Participants[iParticipant];
|
|
hr = pParticipant->RemoveStream(
|
|
(ITStream *)this,
|
|
dwSSRC,
|
|
&fLast
|
|
);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// if the participant is not found
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_TRACE, "SSRC:%x had been removed.", dwSSRC));
|
|
return hr;
|
|
}
|
|
|
|
ITParticipant *pITParticipant = m_Participants[iParticipant];
|
|
|
|
// cancel QOS for this participant.
|
|
if (FAILED(hr = m_pIRTPSession->SetQosState(dwSSRC, FALSE)))
|
|
{
|
|
// the stream might already been stopped
|
|
// so we just put a warning here
|
|
LOG((MSP_WARN, "disabling QOS for %x. hr:%x", dwSSRC, hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "disabled video QOS for %x.", dwSSRC));
|
|
}
|
|
|
|
// find out which substream is going away.
|
|
ITSubStream * pITSubStream = NULL;
|
|
for (int i = 0; i < m_SubStreams.GetSize(); i ++)
|
|
{
|
|
// Find out the participant on the SubStream.
|
|
ITParticipant *pTempParticipant;
|
|
DWORD dwSSRC;
|
|
|
|
((CSubStreamVideoRecv*)m_SubStreams[i])->GetCurrentParticipant(
|
|
&dwSSRC, &pTempParticipant
|
|
);
|
|
|
|
_ASSERTE(pTempParticipant != NULL);
|
|
|
|
pTempParticipant->Release(); // we dont' need the ref here.
|
|
|
|
if (pTempParticipant == pITParticipant)
|
|
{
|
|
pITSubStream = m_SubStreams[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (pITSubStream)
|
|
{
|
|
// remove the mapping if the substream was mapped to a branch.
|
|
for (int i = 0; i < m_Branches.GetSize(); i ++)
|
|
{
|
|
if (m_Branches[i].pITSubStream == pITSubStream)
|
|
{
|
|
m_Branches[i].pITSubStream->Release();
|
|
m_Branches[i].pITSubStream = NULL;
|
|
m_Branches[i].dwSSRC = 0;
|
|
|
|
// fire an event to tell the app that the substream is not used.
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_UNMAPPED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
((CIPConfMSPCall *)m_pMSPCall)->SendParticipantEvent(
|
|
PE_SUBSTREAM_REMOVED,
|
|
pITParticipant,
|
|
pITSubStream
|
|
);
|
|
|
|
if (m_SubStreams.Remove(pITSubStream))
|
|
{
|
|
pITSubStream->Release();
|
|
}
|
|
}
|
|
|
|
|
|
m_Participants.RemoveAt(iParticipant);
|
|
|
|
// if this stream is the last stream that the participant is on,
|
|
// tell the call object to remove it from its list.
|
|
if (fLast)
|
|
{
|
|
((CIPConfMSPCall *)m_pMSPCall)->ParticipantLeft(pITParticipant);
|
|
}
|
|
|
|
pITParticipant->Release();
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//
|
|
// ITStreamQualityControl methods
|
|
//
|
|
STDMETHODIMP CStreamVideoRecv::Set (
|
|
IN StreamQualityProperty Property,
|
|
IN long lValue,
|
|
IN TAPIControlFlags lFlags
|
|
)
|
|
{
|
|
return E_NOTIMPL;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoRecv::Get(
|
|
IN InnerStreamQualityProperty property,
|
|
OUT LONG *plValue,
|
|
OUT TAPIControlFlags *plFlags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get the value for a quality control property. Delegated to the quality
|
|
controller.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoRecv::Get(QualityControl)");
|
|
|
|
HRESULT hr;
|
|
int i;
|
|
LONG totalbps, bps;
|
|
|
|
CLock lock(m_lock);
|
|
|
|
switch (property)
|
|
{
|
|
case InnerStreamQuality_MaxBitrate:
|
|
|
|
bps = 0;
|
|
totalbps = 0;
|
|
|
|
for (i=0; i<m_Branches.GetSize (); i++)
|
|
{
|
|
if (NULL == m_Branches[i].pBitrateControl)
|
|
continue;
|
|
|
|
if (FAILED (hr = m_Branches[i].pBitrateControl->Get (BitrateControl_Maximum, &bps, plFlags, LAYERID)))
|
|
LOG ((MSP_ERROR, "%s failed to get maximum bitrate, %x", __fxName, hr));
|
|
else
|
|
totalbps += bps;
|
|
}
|
|
|
|
*plValue = totalbps;
|
|
|
|
hr = S_OK;
|
|
break;
|
|
|
|
case InnerStreamQuality_CurrBitrate:
|
|
|
|
bps = 0;
|
|
totalbps = 0;
|
|
|
|
for (i=0; i<m_Branches.GetSize (); i++)
|
|
{
|
|
if (NULL == m_Branches[i].pBitrateControl)
|
|
continue;
|
|
|
|
if (FAILED (hr = m_Branches[i].pBitrateControl->Get (BitrateControl_Current, &bps, plFlags, LAYERID)))
|
|
LOG ((MSP_ERROR, "%s failed to get current bitrate, %x", __fxName, hr));
|
|
else
|
|
totalbps += bps;
|
|
}
|
|
|
|
*plValue = totalbps;
|
|
|
|
hr = S_OK;
|
|
break;
|
|
|
|
default:
|
|
hr = CIPConfMSPStream::Get (property, plValue, plFlags);
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CStreamVideoSend
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
CStreamVideoSend::CStreamVideoSend()
|
|
: CIPConfMSPStream(),
|
|
m_pCaptureTerminal(NULL),
|
|
m_pPreviewTerminal(NULL),
|
|
m_pCaptureFilter(NULL),
|
|
m_pCapturePin(NULL),
|
|
m_pPreviewPin(NULL),
|
|
m_pRTPPin(NULL),
|
|
m_pCaptureBitrateControl(NULL),
|
|
m_pCaptureFrameRateControl(NULL),
|
|
m_pPreviewFrameRateControl(NULL)
|
|
{
|
|
m_szName = L"VideoSend";
|
|
}
|
|
|
|
CStreamVideoSend::~CStreamVideoSend()
|
|
{
|
|
CleanupCachedInterface();
|
|
}
|
|
|
|
void CStreamVideoSend::CleanupCachedInterface()
|
|
{
|
|
if (m_pCaptureFilter)
|
|
{
|
|
m_pCaptureFilter->Release();
|
|
m_pCaptureFilter = NULL;
|
|
}
|
|
|
|
if (m_pCapturePin)
|
|
{
|
|
m_pCapturePin->Release();
|
|
m_pCapturePin = NULL;
|
|
}
|
|
|
|
if (m_pIStreamConfig)
|
|
{
|
|
m_pIStreamConfig->Release();
|
|
m_pIStreamConfig = NULL;
|
|
}
|
|
|
|
if (m_pPreviewPin)
|
|
{
|
|
m_pPreviewPin->Release();
|
|
m_pPreviewPin = NULL;
|
|
}
|
|
|
|
if (m_pRTPPin)
|
|
{
|
|
m_pRTPPin->Release();
|
|
m_pRTPPin = NULL;
|
|
}
|
|
|
|
if (m_pCaptureFrameRateControl)
|
|
{
|
|
m_pCaptureFrameRateControl->Release();
|
|
m_pCaptureFrameRateControl = NULL;
|
|
}
|
|
|
|
if (m_pCaptureBitrateControl)
|
|
{
|
|
m_pCaptureBitrateControl->Release();
|
|
m_pCaptureBitrateControl = NULL;
|
|
}
|
|
|
|
if (m_pPreviewFrameRateControl)
|
|
{
|
|
m_pPreviewFrameRateControl->Release();
|
|
m_pPreviewFrameRateControl = NULL;
|
|
}
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ShutDown()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Shut down the stream. Release our members and then calls the base class's
|
|
ShutDown method.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
S_OK
|
|
--*/
|
|
{
|
|
CLock lock(m_lock);
|
|
|
|
// if there are terminals
|
|
BOOL fHasTerminal = FALSE;
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
fHasTerminal = TRUE;
|
|
}
|
|
|
|
// if graph is running
|
|
HRESULT hr;
|
|
OAFilterState FilterState = State_Stopped;
|
|
if (m_pIMediaControl)
|
|
{
|
|
if (FAILED (hr = m_pIMediaControl->GetState(0, &FilterState)))
|
|
{
|
|
LOG ((MSP_ERROR, "CStreamAudioRecv::ShutDown failed to query filter state. %d", hr));
|
|
FilterState = State_Stopped;
|
|
}
|
|
}
|
|
|
|
if (m_pCaptureTerminal)
|
|
{
|
|
m_pCaptureTerminal->Release();
|
|
m_pCaptureTerminal = NULL;
|
|
}
|
|
|
|
if (m_pPreviewTerminal)
|
|
{
|
|
m_pPreviewTerminal->Release();
|
|
m_pPreviewTerminal = NULL;
|
|
}
|
|
|
|
CleanupCachedInterface();
|
|
|
|
// fire event
|
|
if (fHasTerminal && FilterState == State_Running)
|
|
{
|
|
SendStreamEvent(CALL_STREAM_INACTIVE, CALL_CAUSE_LOCAL_REQUEST, 0, NULL);
|
|
}
|
|
|
|
return CIPConfMSPStream::ShutDown();
|
|
}
|
|
|
|
HRESULT
|
|
SetVideoFormat(
|
|
IN IUnknown * pIUnknown,
|
|
IN BOOL bCIF,
|
|
IN DWORD dwFramesPerSecond
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set the video format to be CIF or QCIF and also set the frames per second.
|
|
|
|
Arguments:
|
|
|
|
pIUnknown - a capture terminal.
|
|
|
|
bCIF - CIF or QCIF.
|
|
|
|
dwFramesPerSecond - Frames per second.
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "SetVideoFormat"));
|
|
|
|
HRESULT hr;
|
|
|
|
// first get eht IAMStreamConfig interface.
|
|
CComPtr<IAMStreamConfig> pIAMStreamConfig;
|
|
|
|
if (FAILED(hr = pIUnknown->QueryInterface(
|
|
__uuidof(IAMStreamConfig),
|
|
(void **)&pIAMStreamConfig
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "Can't get IAMStreamConfig interface.%8x", hr));
|
|
return hr;
|
|
}
|
|
|
|
// get the current format of the video capture terminal.
|
|
AM_MEDIA_TYPE *pmt;
|
|
if (FAILED(hr = pIAMStreamConfig->GetFormat(&pmt)))
|
|
{
|
|
LOG((MSP_ERROR, "GetFormat returns error: %8x", hr));
|
|
return hr;
|
|
}
|
|
|
|
VIDEOINFO *pVideoInfo = (VIDEOINFO *)pmt->pbFormat;
|
|
if (pVideoInfo == NULL)
|
|
{
|
|
MSPDeleteMediaType(pmt);
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
BITMAPINFOHEADER *pHeader = HEADER(pmt->pbFormat);
|
|
if (pHeader == NULL)
|
|
{
|
|
MSPDeleteMediaType(pmt);
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
LOG((MSP_INFO,
|
|
"Video capture: Format BitRate: %d, TimePerFrame: %d",
|
|
pVideoInfo->dwBitRate,
|
|
pVideoInfo->AvgTimePerFrame));
|
|
|
|
LOG((MSP_INFO, "Video capture: Format Compression:%c%c%c%c %dbit %dx%d",
|
|
(DWORD)pHeader->biCompression & 0xff,
|
|
((DWORD)pHeader->biCompression >> 8) & 0xff,
|
|
((DWORD)pHeader->biCompression >> 16) & 0xff,
|
|
((DWORD)pHeader->biCompression >> 24) & 0xff,
|
|
pHeader->biBitCount,
|
|
pHeader->biWidth,
|
|
pHeader->biHeight));
|
|
|
|
// The time is in 100ns unit.
|
|
pVideoInfo->AvgTimePerFrame = (DWORD) 1e7 / dwFramesPerSecond;
|
|
|
|
if (bCIF)
|
|
{
|
|
pHeader->biWidth = CIFWIDTH;
|
|
pHeader->biHeight = CIFHEIGHT;
|
|
}
|
|
else
|
|
{
|
|
pHeader->biWidth = QCIFWIDTH;
|
|
pHeader->biHeight = QCIFHEIGHT;
|
|
}
|
|
|
|
#if defined(ALPHA)
|
|
// update bmiSize with new Width/Height
|
|
pHeader->biSizeImage = DIBSIZE( ((VIDEOINFOHEADER *)pmt->pbFormat)->bmiHeader );
|
|
#endif
|
|
|
|
if (FAILED(hr = pIAMStreamConfig->SetFormat(pmt)))
|
|
{
|
|
LOG((MSP_ERROR, "putMediaFormat returns error: %8x", hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO,
|
|
"Video capture: Format BitRate: %d, TimePerFrame: %d",
|
|
pVideoInfo->dwBitRate,
|
|
pVideoInfo->AvgTimePerFrame));
|
|
|
|
LOG((MSP_INFO, "Video capture: Format Compression:%c%c%c%c %dbit %dx%d",
|
|
(DWORD)pHeader->biCompression & 0xff,
|
|
((DWORD)pHeader->biCompression >> 8) & 0xff,
|
|
((DWORD)pHeader->biCompression >> 16) & 0xff,
|
|
((DWORD)pHeader->biCompression >> 24) & 0xff,
|
|
pHeader->biBitCount,
|
|
pHeader->biWidth,
|
|
pHeader->biHeight));
|
|
}
|
|
|
|
MSPDeleteMediaType(pmt);
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT
|
|
SetVideoBufferSize(
|
|
IN IUnknown *pIUnknown
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set the video capture terminal's buffersize.
|
|
|
|
Arguments:
|
|
|
|
pIUnknown - a capture terminal.
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
// The number of capture buffers is four for now.
|
|
#define NUMCAPTUREBUFFER 4
|
|
|
|
LOG((MSP_TRACE, "SetVideoBufferSize"));
|
|
|
|
HRESULT hr;
|
|
|
|
CComPtr<IAMBufferNegotiation> pBN;
|
|
if (FAILED(hr = pIUnknown->QueryInterface(
|
|
__uuidof(IAMBufferNegotiation),
|
|
(void **)&pBN
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "Can't get buffer negotiation interface.%8x", hr));
|
|
return hr;
|
|
}
|
|
|
|
ALLOCATOR_PROPERTIES prop;
|
|
|
|
#if 0 // Get allocator property is not working.
|
|
if (FAILED(hr = pBN->GetAllocatorProperties(&prop)))
|
|
{
|
|
LOG((MSP_ERROR, "GetAllocatorProperties returns error: %8x", hr));
|
|
return hr;
|
|
}
|
|
|
|
// Set the number of buffers.
|
|
if (prop.cBuffers > NUMCAPTUREBUFFER)
|
|
{
|
|
prop.cBuffers = NUMCAPTUREBUFFER;
|
|
}
|
|
#endif
|
|
|
|
DWORD dwBuffers = NUMCAPTUREBUFFER;
|
|
GetRegValue(gszNumVideoCaptureBuffers, &dwBuffers);
|
|
|
|
prop.cBuffers = dwBuffers;
|
|
prop.cbBuffer = -1;
|
|
prop.cbAlign = -1;
|
|
prop.cbPrefix = -1;
|
|
|
|
if (FAILED(hr = pBN->SuggestAllocatorProperties(&prop)))
|
|
{
|
|
LOG((MSP_ERROR, "SuggestAllocatorProperties returns error: %8x", hr));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO,
|
|
"SetVidedobuffersize"
|
|
" buffers: %d, buffersize: %d, align: %d, Prefix: %d",
|
|
prop.cBuffers,
|
|
prop.cbBuffer,
|
|
prop.cbAlign,
|
|
prop.cbPrefix
|
|
));
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::FindPreviewInputPin(
|
|
IN ITTerminalControl* pTerminal,
|
|
OUT IPin ** ppIPin
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Find the input pin on a preview terminal.
|
|
|
|
Arguments:
|
|
|
|
pTerminal - a video render terminal.
|
|
|
|
ppIPin - the address to store a pointer to a IPin interface.
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoSend.FindPreviewInputPin, pTerminal %x", pTerminal));
|
|
|
|
/*
|
|
// try to disable DDraw because we want to use DDraw for receive stream.
|
|
HRESULT hr2;
|
|
IDrawVideoImage *pIDrawVideoImage;
|
|
hr2 = pTerminal->QueryInterface(__uuidof(IDrawVideoImage), (void **)&pIDrawVideoImage);
|
|
if (SUCCEEDED(hr2))
|
|
{
|
|
hr2 = pIDrawVideoImage->DrawVideoImageBegin();
|
|
if (FAILED(hr2))
|
|
{
|
|
LOG((MSP_WARN, "Can't disable DDraw. %x", hr2));
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "DDraw disabled."));
|
|
}
|
|
|
|
pIDrawVideoImage->Release();
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_WARN, "Can't get IDrawVideoImage. %x", hr2));
|
|
}
|
|
*/
|
|
|
|
// Get the pins from the first terminal because we only use on terminal
|
|
// on this stream.
|
|
const DWORD MAXPINS = 8;
|
|
|
|
DWORD dwNumPins = MAXPINS;
|
|
IPin * Pins[MAXPINS];
|
|
|
|
HRESULT hr = pTerminal->ConnectTerminal(
|
|
m_pIGraphBuilder, TD_RENDER, &dwNumPins, Pins
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't connect to terminal, %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
if (dwNumPins == 0)
|
|
{
|
|
LOG((MSP_ERROR, "terminal has no pins."));
|
|
return hr;
|
|
}
|
|
|
|
if (IsBadReadPtr (Pins, dwNumPins * sizeof (IPin*)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin array"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
for (DWORD i = 0; i < dwNumPins; i++)
|
|
{
|
|
if (IsBadReadPtr (Pins[i], sizeof (IPin)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin. # %d", i));
|
|
return E_POINTER;
|
|
}
|
|
}
|
|
|
|
// Save the first pin and release the others.
|
|
CComPtr <IPin> pIPin = Pins[0];
|
|
for (DWORD i = 0; i < dwNumPins; i ++)
|
|
{
|
|
Pins[i]->Release();
|
|
}
|
|
|
|
pIPin->AddRef();
|
|
*ppIPin = pIPin;
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::CheckTerminalTypeAndDirection(
|
|
IN ITTerminal * pTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check if the terminal is allowed on this stream.
|
|
VideoSend allows both a capture terminal and a preivew terminal.
|
|
|
|
Arguments:
|
|
|
|
pTerminal - the terminal.
|
|
|
|
Return value:
|
|
|
|
HRESULT.
|
|
S_OK means the terminal is OK.
|
|
*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoSend.CheckTerminalTypeAndDirection"));
|
|
|
|
// This stream only support one capture + one preview terminal
|
|
if (m_Terminals.GetSize() > 1)
|
|
{
|
|
return TAPI_E_MAXTERMINALS;
|
|
}
|
|
|
|
// check the media type of this terminal.
|
|
long lMediaType;
|
|
HRESULT hr = pTerminal->get_MediaType(&lMediaType);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal media type. %x", hr));
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
if ((DWORD)lMediaType != m_dwMediaType)
|
|
{
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
// check the direction of this terminal.
|
|
TERMINAL_DIRECTION Direction;
|
|
hr = pTerminal->get_Direction(&Direction);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal direction. %x", hr));
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
// check the direction of this terminal.
|
|
TERMINAL_DIRECTION Direction2;
|
|
hr = m_Terminals[0]->get_Direction(&Direction2);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal direction. %x", hr));
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
if (Direction == Direction2)
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"can't have two terminals with the same direction. %x", hr));
|
|
return TAPI_E_MAXTERMINALS;
|
|
}
|
|
}
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::SetUpFilters()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Insert filters into the graph and connect to the terminals.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "VideoSend.SetUpFilters"));
|
|
|
|
// we only support one capture terminal and one preview
|
|
// window on this stream.
|
|
if (m_Terminals.GetSize() > 2)
|
|
{
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
int iCaptureIndex = -1, iPreviewIndex = -1;
|
|
|
|
// Find out which terminal is capture and which is preview.
|
|
HRESULT hr;
|
|
for (int i = 0; i < m_Terminals.GetSize(); i ++)
|
|
{
|
|
TERMINAL_DIRECTION Direction;
|
|
hr = m_Terminals[i]->get_Direction(&Direction);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "can't get terminal direction. %x", hr));
|
|
SendStreamEvent(CALL_TERMINAL_FAIL, CALL_CAUSE_BAD_DEVICE, hr, m_Terminals[i]);
|
|
|
|
return hr;
|
|
}
|
|
|
|
if (Direction == TD_CAPTURE || Direction == TD_BIDIRECTIONAL)
|
|
{
|
|
iCaptureIndex = i;
|
|
}
|
|
else
|
|
{
|
|
iPreviewIndex = i;
|
|
}
|
|
}
|
|
|
|
// the stream will not work without a capture terminal.
|
|
if (iCaptureIndex == -1)
|
|
{
|
|
LOG((MSP_ERROR, "no capture terminal selected."));
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
// Connect the capture filter to the terminal.
|
|
if (FAILED(hr = ConnectTerminal(
|
|
m_Terminals[iCaptureIndex]
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "connect the codec filter to terminal. %x", hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
if (iPreviewIndex != -1)
|
|
{
|
|
// Connect the preview filter to the terminal.
|
|
if (FAILED(hr = ConnectTerminal(
|
|
m_Terminals[iPreviewIndex]
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR, "connect the codec filter to terminal. %x", hr));
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::GetVideoCapturePins(
|
|
IN ITTerminalControl* pTerminal,
|
|
OUT BOOL *pfDirectRTP
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Given a video capture terminal, find all the pins we need, which will be
|
|
the capture pin, preview pin, and the RTP packetization pin.
|
|
|
|
Side effect: It changes the m_pCapturePin, m_pPreviewPin, m_pRTPPin
|
|
members, which needs to be cleaned up if the terminal is disconnected.
|
|
|
|
Arguments:
|
|
|
|
pTerminal - a pointer to the ITTerminalControl interface.
|
|
|
|
pfDirectRTP - whether this terminal support RTP directly.
|
|
|
|
Return Value:
|
|
|
|
HRESULT
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::GetVideoCapturePins");
|
|
LOG((MSP_TRACE, "%s enters", __fxName));
|
|
|
|
const DWORD MAXPINS = 4;
|
|
|
|
DWORD dwNumPins = MAXPINS;
|
|
IPin * Pins[MAXPINS];
|
|
|
|
HRESULT hr = pTerminal->ConnectTerminal(
|
|
m_pIGraphBuilder, TD_CAPTURE, &dwNumPins, Pins
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, can't connect to terminal, hr=%x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
_ASSERT(m_pCapturePin == NULL && m_pPreviewPin == NULL && m_pRTPPin == NULL);
|
|
|
|
if (IsBadReadPtr (Pins, dwNumPins * sizeof (IPin*)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin array"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
|
|
// find the pins we need.
|
|
for (DWORD i = 0; i < dwNumPins; i ++)
|
|
{
|
|
if (IsBadReadPtr (Pins[i], sizeof (IPin)))
|
|
{
|
|
LOG((MSP_ERROR, "terminal returned bad pin. # %d", i));
|
|
hr = E_POINTER;
|
|
break;
|
|
}
|
|
|
|
PIN_INFO PinInfo;
|
|
hr = Pins[i]->QueryPinInfo(&PinInfo);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, can't get pin info, hr=%x", __fxName, hr));
|
|
break;
|
|
}
|
|
|
|
if (lstrcmpW(PinInfo.achName, PNAME_CAPTURE) == 0)
|
|
{
|
|
m_pCapturePin = Pins[i];
|
|
|
|
// remember the capture filter as well.
|
|
m_pCaptureFilter = PinInfo.pFilter;
|
|
m_pCaptureFilter->AddRef();
|
|
|
|
}
|
|
else if (lstrcmpW(PinInfo.achName, PNAME_PREVIEW) == 0)
|
|
{
|
|
m_pPreviewPin = Pins[i];
|
|
}
|
|
else if (lstrcmpW(PinInfo.achName, PNAME_RTPPD) == 0)
|
|
{
|
|
m_pRTPPin = Pins[i];
|
|
}
|
|
else if (PinInfo.dir == PINDIR_OUTPUT)
|
|
{
|
|
// this must be the capture filter of some third party terminal.
|
|
m_pCapturePin = Pins[i];
|
|
|
|
// remember the capture filter as well.
|
|
m_pCaptureFilter = PinInfo.pFilter;
|
|
m_pCaptureFilter->AddRef();
|
|
|
|
}
|
|
else
|
|
{
|
|
Pins[i]->Release();
|
|
}
|
|
|
|
// we don't need the filter here.
|
|
PinInfo.pFilter->Release();
|
|
}
|
|
|
|
|
|
// check if we have got all the pins we need.
|
|
if (m_pCapturePin == NULL ||
|
|
m_pPreviewPin == NULL ||
|
|
m_pRTPPin == NULL)
|
|
{
|
|
if ((m_pCapturePin != NULL)
|
|
&& (hr = ::PinSupportsMediaType(
|
|
m_pCapturePin, __uuidof(MEDIATYPE_RTP_Single_Stream))) == S_OK)
|
|
{
|
|
// This terminal generates RTP directly.
|
|
*pfDirectRTP = TRUE;
|
|
return S_OK;
|
|
}
|
|
|
|
LOG((MSP_ERROR,
|
|
"%s, can't find all the pins, Capture:%p, Preview:%p, RTP:%P",
|
|
__fxName, m_pCapturePin, m_pPreviewPin, m_pRTPPin));
|
|
|
|
hr = E_UNEXPECTED;
|
|
}
|
|
|
|
if (hr != S_OK)
|
|
{
|
|
// something is wrong, clean up
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
CleanupCachedInterface();
|
|
|
|
return hr;
|
|
}
|
|
|
|
// now get the optional video interfaces.
|
|
_ASSERT(m_pIStreamConfig == NULL);
|
|
|
|
hr = m_pCapturePin->QueryInterface(&m_pIStreamConfig);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_WARN, "%s, queryinterface failed", __fxName));
|
|
}
|
|
|
|
hr = m_pCapturePin->QueryInterface(&m_pCaptureFrameRateControl);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_WARN,
|
|
"%s:query capture pin's IFrameRateControl failed, hr=%x",
|
|
__fxName, hr));
|
|
}
|
|
|
|
hr = m_pCapturePin->QueryInterface(&m_pCaptureBitrateControl);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_WARN,
|
|
"%s:query capture pin's IBitRateControl failed, hr=%x",
|
|
__fxName, hr));
|
|
}
|
|
|
|
hr = m_pPreviewPin->QueryInterface(&m_pPreviewFrameRateControl);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_WARN,
|
|
"%s:query preview pin's IFrameRateControl failed, hr=%x",
|
|
__fxName, hr));
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ConnectCaptureTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
The stream needs its capture pin, preview pin, and RTP packetization pin.
|
|
|
|
The capture pin and the preview pin are connected to the RTP sink filter
|
|
and the preview pin is connected to the preivew terminal. If the preview
|
|
terminal doesn't exist yet, the preview pin is remembered and used later
|
|
when the preview terminal is selected.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - the terminal being connected.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::ConnectCaptureTerminal");
|
|
LOG((MSP_TRACE, "%s enters, pITTerminal:%p", __fxName, pITTerminal));
|
|
|
|
// Get the TerminalControl interface on the terminal
|
|
CComPtr<ITTerminalControl> pTerminal;
|
|
HRESULT hr = pITTerminal->QueryInterface(&pTerminal);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, can't get Terminal Control interface", __fxName));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, E_NOINTERFACE, pITTerminal);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
// Find the pins on the capture terminal. The pins will be stored in
|
|
// m_pCapturePin, m_pPreviewPin, m_pRTPPin
|
|
BOOL fDirectRTP = FALSE;
|
|
hr = GetVideoCapturePins(pTerminal, &fDirectRTP);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, Get capture pins failed. hr=%x", __fxName, hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
return hr;
|
|
}
|
|
|
|
hr = CreateSendFilters(m_pCapturePin, m_pRTPPin, fDirectRTP);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, Create video send filters failed. hr=%x", __fxName, hr));
|
|
|
|
goto cleanup;
|
|
}
|
|
|
|
//
|
|
// Now we are actually connected. Update our state and perform
|
|
// postconnection.
|
|
//
|
|
hr = pTerminal->CompleteConnectTerminal();
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, Create video send filters failed. hr=%x", __fxName, hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (m_pPreviewTerminal != NULL)
|
|
{
|
|
// errors will be fired as events.
|
|
ConnectPreviewTerminal(m_pPreviewTerminal);
|
|
}
|
|
|
|
return S_OK;
|
|
|
|
cleanup:
|
|
// disconnect the terminal.
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
CleanupCachedInterface();
|
|
|
|
// clean up internal filters as well.
|
|
CleanUpFilters();
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ConnectPreviewTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
If the capture terminal has been connected, this function connects the
|
|
capture terminal's preview pin with the preview terminal. Otherwise, the
|
|
preview terminal is just remembered and wait for the capture terminal.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - the terminal being connected.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::ConnectPreviewTerminal");
|
|
LOG((MSP_TRACE, "%s enters, pITTerminal:%p", __fxName, pITTerminal));
|
|
|
|
if (!m_pCapturePin)
|
|
{
|
|
LOG ((MSP_TRACE, "%s capture pin is null.", __fxName));
|
|
return E_FAIL;
|
|
}
|
|
|
|
if (!m_pPreviewPin)
|
|
{
|
|
// the capture terminal is not selected yet. We will just wait.
|
|
LOG((MSP_TRACE, "%s, capture is not ready yet.", __fxName));
|
|
return S_OK;
|
|
}
|
|
|
|
// Get the TerminalControl interface on the terminal
|
|
CComPtr<ITTerminalControl> pTerminal;
|
|
HRESULT hr = pITTerminal->QueryInterface(&pTerminal);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, can't get Terminal Control interface", __fxName));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, E_NOINTERFACE, pITTerminal);
|
|
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
// find the input pin on the preview window.
|
|
CComPtr<IPin> pPreviewInputPin;
|
|
|
|
hr = FindPreviewInputPin(pTerminal, &pPreviewInputPin);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, find preview input pin failed. hr=%x", __fxName, hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
return hr;
|
|
}
|
|
|
|
// connect the pins together.
|
|
hr = m_pIGraphBuilder->Connect(m_pPreviewPin, pPreviewInputPin);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, connect preview pins failed. hr=%x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Now we are actually connected, perform postconnection.
|
|
//
|
|
hr = pTerminal->CompleteConnectTerminal();
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, complete connect terminal failed. hr=%x", __fxName, hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
// disconnect the terminal.
|
|
pTerminal->DisconnectTerminal(m_pIGraphBuilder, 0);
|
|
|
|
return hr;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ConnectTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
connect the video terminals to the stream.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::ConnectTerminal");
|
|
LOG((MSP_TRACE, "%s enters, pITTerminal:%p", __fxName, pITTerminal));
|
|
|
|
// Find out the direction of the terminal.
|
|
TERMINAL_DIRECTION Direction;
|
|
HRESULT hr = pITTerminal->get_Direction(&Direction);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, can't get terminal direction. hr=%x", __fxName, hr));
|
|
|
|
SendStreamEvent(CALL_TERMINAL_FAIL,
|
|
CALL_CAUSE_BAD_DEVICE, hr, pITTerminal);
|
|
|
|
return hr;
|
|
}
|
|
|
|
if (Direction != TD_RENDER)
|
|
{
|
|
hr = ConnectCaptureTerminal(pITTerminal);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// save the capture terminal.
|
|
_ASSERT(m_pCaptureTerminal == NULL);
|
|
|
|
m_pCaptureTerminal = pITTerminal;
|
|
m_pCaptureTerminal->AddRef();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
hr = ConnectPreviewTerminal(pITTerminal);
|
|
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
// save the preview terminal.
|
|
_ASSERT(m_pPreviewTerminal == NULL);
|
|
|
|
m_pPreviewTerminal = pITTerminal;
|
|
m_pPreviewTerminal->AddRef();
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::DisconnectTerminal(
|
|
IN ITTerminal * pITTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Disconnect a terminal. It will remove its filters from the graph and
|
|
also release its references to the graph.
|
|
|
|
If it is the capture terminal being disconnected, all the pins that the
|
|
stream cached need to be released too.
|
|
|
|
Arguments:
|
|
|
|
pITTerminal - the terminal.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::DisconnectTerminal");
|
|
LOG((MSP_TRACE, "%s enters, pITTerminal:%p", __fxName, pITTerminal));
|
|
|
|
HRESULT hr = CIPConfMSPStream::DisconnectTerminal(pITTerminal);
|
|
|
|
if (pITTerminal == m_pCaptureTerminal)
|
|
{
|
|
// release all the capture pins we cached.
|
|
CleanupCachedInterface();
|
|
|
|
m_pCaptureTerminal->Release();
|
|
m_pCaptureTerminal = NULL;
|
|
|
|
CleanUpFilters ();
|
|
|
|
// disconnect preview term as well
|
|
// when we connect capture,
|
|
// we always try to connect preview if one is available
|
|
|
|
if (m_pPreviewTerminal)
|
|
{
|
|
CIPConfMSPStream::DisconnectTerminal(m_pPreviewTerminal);
|
|
}
|
|
}
|
|
else if (pITTerminal == m_pPreviewTerminal)
|
|
{
|
|
m_pPreviewTerminal->Release();
|
|
m_pPreviewTerminal = NULL;
|
|
}
|
|
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ConnectRTPFilter(
|
|
IN IGraphBuilder *pIGraphBuilder,
|
|
IN IPin *pCapturePin,
|
|
IN IPin *pRTPPin,
|
|
IN IBaseFilter *pRTPFilter
|
|
)
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::ConnectRTPFilters");
|
|
LOG((MSP_TRACE, "%s enters", __fxName));
|
|
|
|
HRESULT hr;
|
|
|
|
// find the capture pin on the RTP filter.
|
|
CComPtr <IPin> pRTPCapturePin;
|
|
hr = pRTPFilter->FindPin(PNAME_CAPTURE, &pRTPCapturePin);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, find capture pin on rtp filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// Connect the capture pin of the video capture filter with the capture pin
|
|
// of the rTP filter.
|
|
hr = pIGraphBuilder->ConnectDirect(pCapturePin, pRTPCapturePin, NULL);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, can't connect capture pins. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (pRTPPin)
|
|
{
|
|
// find the packetization pin on the RTP filter.
|
|
CComPtr <IPin> pRTPRTPPin;
|
|
hr = pRTPFilter->FindPin(PNAME_RTPPD, &pRTPRTPPin);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, find capture pin on rtp filter. %x", __fxName, hr));
|
|
|
|
pIGraphBuilder->Disconnect(pRTPPin);
|
|
|
|
return hr;
|
|
}
|
|
|
|
// Connect the RTP pin of the video capture filter with the RTP pin
|
|
// of the rTP filter.
|
|
hr = pIGraphBuilder->ConnectDirect(pRTPPin, pRTPRTPPin, NULL);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, can't connect capture pins. %x", __fxName, hr));
|
|
|
|
pIGraphBuilder->Disconnect(pRTPPin);
|
|
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::ConfigureRTPFormats(
|
|
IN IBaseFilter * pIRTPFilter,
|
|
IN IStreamConfig * pIStreamConfig
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Configure the RTP filter with RTP<-->AM media type mappings.
|
|
|
|
Arguments:
|
|
|
|
pIRTPFilter - The source RTP Filter.
|
|
|
|
pIStreamConfig - The stream config interface that has the media info.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("VideoSend::ConfigureRTPFormats");
|
|
LOG((MSP_TRACE, "%s enters", __fxName));
|
|
|
|
HRESULT hr;
|
|
|
|
CComPtr<IRtpMediaControl> pIRtpMediaControl;
|
|
hr = pIRTPFilter->QueryInterface(&pIRtpMediaControl);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s adding source filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// find the number of capabilities supported.
|
|
DWORD dwCount;
|
|
hr = pIStreamConfig->GetNumberOfCapabilities(&dwCount);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s GetNumberOfCapabilities. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
BOOL fFound = FALSE;
|
|
BOOL fFormatSet = FALSE;
|
|
|
|
for (DWORD dw = 0; dw < dwCount; dw ++)
|
|
{
|
|
// TODO, a new interface is needed to resolve RTP to MediaType.
|
|
AM_MEDIA_TYPE *pMediaType;
|
|
DWORD dwPayloadType;
|
|
|
|
hr = pIStreamConfig->GetStreamCaps(
|
|
dw, &pMediaType, NULL, &dwPayloadType
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s GetStreamCaps. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
BITMAPINFOHEADER *pHeader = HEADER(pMediaType->pbFormat);
|
|
if (pHeader == NULL)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
|
|
// check the image size
|
|
if (m_Settings.fCIF)
|
|
{
|
|
if (pHeader->biWidth != CIFWIDTH)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (pHeader->biWidth != QCIFWIDTH)
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
for (DWORD dw2 = 0; dw2 < m_Settings.dwNumPayloadTypes; dw2 ++)
|
|
{
|
|
if (dwPayloadType == m_Settings.PayloadTypes[dw2])
|
|
{
|
|
hr = pIRtpMediaControl->SetFormatMapping(
|
|
dwPayloadType,
|
|
90000, // default video clock rate.
|
|
pMediaType
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
|
|
LOG((MSP_ERROR, "%s SetFormatMapping. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_INFO, "%s Configured payload:%d", __fxName, dwPayloadType));
|
|
}
|
|
|
|
if (dw2 == 0 && !fFormatSet)
|
|
{
|
|
// tell the encoder to use this format.
|
|
// TODO, cache all the allowed mediatypes in the conference for
|
|
// future enumerations. It would be nice that we can get the SDP blob
|
|
// when the call object is created.
|
|
hr = pIStreamConfig->SetFormat(dwPayloadType, pMediaType);
|
|
if (FAILED(hr))
|
|
{
|
|
MSPDeleteMediaType(pMediaType);
|
|
|
|
LOG((MSP_ERROR, "%s SetFormat. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
fFormatSet = TRUE;
|
|
}
|
|
}
|
|
}
|
|
MSPDeleteMediaType(pMediaType);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
HRESULT CStreamVideoSend::CreateSendFilters(
|
|
IN IPin *pCapturePin,
|
|
IN IPin *pRTPPin,
|
|
IN BOOL fDirectRTP
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Insert filters into the graph and connect to the capture pin.
|
|
|
|
Capturepin->[Encoder]->RTPRender
|
|
|
|
Arguments:
|
|
|
|
pCapturePin - the capture pin on the capture filter.
|
|
|
|
pRTPPin - the RTP packetization pin.
|
|
|
|
fDirectRTP - the capture pin supports RTP directly.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::CreateSendFilters");
|
|
LOG((MSP_TRACE, "%s enters", __fxName));
|
|
|
|
HRESULT hr;
|
|
|
|
// Create the RTP render filter and add it into the graph.
|
|
CComPtr<IBaseFilter> pRenderFilter;
|
|
|
|
if (m_pIRTPSession == NULL)
|
|
{
|
|
if (FAILED(hr = ::AddFilter(
|
|
m_pIGraphBuilder,
|
|
__uuidof(MSRTPRenderFilter),
|
|
L"RtpRender",
|
|
&pRenderFilter)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, adding render filter. hr=%x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED(hr = ConfigureRTPFilter(pRenderFilter)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, configure RTP render filter failed. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (FAILED (hr = m_pIRTPSession->QueryInterface (&pRenderFilter)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to get filter from rtp session. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
if (FAILED (hr = m_pIGraphBuilder->AddFilter ((IBaseFilter *)pRenderFilter, L"RtpRender")))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to add filter to graph. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
|
|
if (!fDirectRTP)
|
|
{
|
|
CComPtr<IStreamConfig> pIStreamConfig;
|
|
hr = pCapturePin->QueryInterface(&pIStreamConfig);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "%s, query IStreamConfig. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
// configure the format info on the RTP filter
|
|
if (FAILED(hr = ConfigureRTPFormats(pRenderFilter, pIStreamConfig)))
|
|
{
|
|
LOG((MSP_ERROR, "%s, configure RTP formats. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// configure RTP_SINGLE_STREAM to on the RTP filter.
|
|
}
|
|
|
|
if (FAILED(hr = ConnectRTPFilter(
|
|
m_pIGraphBuilder,
|
|
pCapturePin,
|
|
pRTPPin,
|
|
pRenderFilter
|
|
)))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"%s, connect capture pin and the Render filter. %x", __fxName, hr));
|
|
return hr;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//
|
|
// IInnerStreamQualityControl methods.
|
|
//
|
|
STDMETHODIMP CStreamVideoSend::GetRange(
|
|
IN InnerStreamQualityProperty property,
|
|
OUT LONG *plMin,
|
|
OUT LONG *plMax,
|
|
OUT LONG *plSteppingDelta,
|
|
OUT LONG *plDefault,
|
|
OUT TAPIControlFlags *plFlags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get the range for a quality control property. Delegated to capture filter
|
|
for now.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::GetRange (InnerStreamQualityControl)");
|
|
|
|
HRESULT hr;
|
|
static BOOL fReported = FALSE;
|
|
|
|
CLock lock(m_lock);
|
|
|
|
switch (property)
|
|
{
|
|
case InnerStreamQuality_MinFrameInterval:
|
|
|
|
if (m_pCaptureFrameRateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureFrameRateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureFrameRateControl->GetRange(
|
|
FrameRateControl_Maximum, plMin, plMax, plSteppingDelta, plDefault, plFlags
|
|
);
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_AvgFrameInterval:
|
|
|
|
if (m_pCaptureFrameRateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureFrameRateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureFrameRateControl->GetRange(
|
|
FrameRateControl_Current, plMin, plMax, plSteppingDelta, plDefault, plFlags
|
|
);
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_MaxBitrate:
|
|
|
|
if (m_pCaptureBitrateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureBitrateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureBitrateControl->GetRange(
|
|
BitrateControl_Maximum, plMin, plMax, plSteppingDelta, plDefault, plFlags, LAYERID
|
|
);
|
|
|
|
if (S_OK == hr)
|
|
{
|
|
if (*plMax < QCLIMIT_MIN_BITRATE)
|
|
{
|
|
LOG ((MSP_WARN, "%s: max bitrate %d too low", __fxName, *plMax));
|
|
hr = E_UNEXPECTED;
|
|
}
|
|
else
|
|
{
|
|
// adjust the min and default value
|
|
if (*plMin < QCLIMIT_MIN_BITRATE)
|
|
*plMin = QCLIMIT_MIN_BITRATE;
|
|
if (*plDefault < QCLIMIT_MIN_BITRATE)
|
|
*plDefault = QCLIMIT_MIN_BITRATE;
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_CurrBitrate:
|
|
|
|
if (m_pCaptureBitrateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureBitrateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureBitrateControl->GetRange(
|
|
BitrateControl_Current, plMin, plMax, plSteppingDelta, plDefault, plFlags, LAYERID
|
|
);
|
|
|
|
if (S_OK == hr)
|
|
{
|
|
if (*plMax < QCLIMIT_MIN_BITRATE)
|
|
{
|
|
LOG ((MSP_WARN, "%s: max bitrate %d too low", __fxName, *plMax));
|
|
hr = E_UNEXPECTED;
|
|
}
|
|
else
|
|
{
|
|
// adjust the min and default value
|
|
if (*plMin < QCLIMIT_MIN_BITRATE)
|
|
*plMin = QCLIMIT_MIN_BITRATE;
|
|
if (*plDefault < QCLIMIT_MIN_BITRATE)
|
|
*plDefault = QCLIMIT_MIN_BITRATE;
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
hr = CIPConfMSPStream::GetRange (property, plMin, plMax, plSteppingDelta, plDefault, plFlags);
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoSend::Get(
|
|
IN InnerStreamQualityProperty property,
|
|
OUT LONG *plValue,
|
|
OUT TAPIControlFlags *plFlags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get the value for a quality control property. Delegated to the quality
|
|
controller.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::Get(QualityControl)");
|
|
|
|
HRESULT hr;
|
|
static BOOL fReported = FALSE;
|
|
|
|
CLock lock(m_lock);
|
|
|
|
switch (property)
|
|
{
|
|
case InnerStreamQuality_MinFrameInterval:
|
|
|
|
if (m_pCaptureFrameRateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureFrameRateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureFrameRateControl->Get(FrameRateControl_Maximum, plValue, plFlags);
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_AvgFrameInterval:
|
|
|
|
if (m_pCaptureFrameRateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureFrameRateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureFrameRateControl->Get(FrameRateControl_Current, plValue, plFlags);
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_MaxBitrate:
|
|
|
|
if( m_pCaptureBitrateControl == NULL )
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pICaptureBitrateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureBitrateControl->Get(BitrateControl_Maximum, plValue, plFlags, LAYERID);
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_CurrBitrate:
|
|
|
|
if (m_pCaptureBitrateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, m_pCaptureBitrateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
}
|
|
else
|
|
{
|
|
hr = m_pCaptureBitrateControl->Get(BitrateControl_Current, plValue, plFlags, LAYERID);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
hr = CIPConfMSPStream::Get (property, plValue, plFlags);
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CStreamVideoSend::Set(
|
|
IN InnerStreamQualityProperty property,
|
|
IN LONG lValue,
|
|
IN TAPIControlFlags lFlags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set the value for a quality control property. Delegated to the quality
|
|
controller.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
ENTER_FUNCTION("CStreamVideoSend::Set(InnerStreamQualityControl)");
|
|
|
|
CLock lock(m_lock);
|
|
|
|
HRESULT hr;
|
|
LONG l;
|
|
static BOOL fReported = FALSE;
|
|
|
|
LONG min, max, delta, Default;
|
|
TAPIControlFlags flags;
|
|
|
|
switch (property)
|
|
{
|
|
// adjusted frame rate by call qc
|
|
case InnerStreamQuality_AdjMinFrameInterval:
|
|
|
|
if (m_pCaptureFrameRateControl == NULL &&
|
|
m_pPreviewFrameRateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, Capture/Preview FrameRateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
break;
|
|
}
|
|
|
|
// set capture frame rate control
|
|
if (m_pCaptureFrameRateControl)
|
|
{
|
|
// get valid range
|
|
if (FAILED (hr = m_pCaptureFrameRateControl->GetRange (
|
|
FrameRateControl_Current,
|
|
&min, &max, &delta, &Default, &flags)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to getrange on capture frame rate control. %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
// adjust value
|
|
l = lValue;
|
|
// use current value - max - if input value not set
|
|
if (l==QCDEFAULT_QUALITY_UNSET) l = max;
|
|
else if (l<min) l = min;
|
|
else if (l>max) l = max;
|
|
|
|
// remember the value
|
|
m_pStreamQCRelay->Set (property, l, lFlags);
|
|
|
|
if (FAILED (hr = m_pCaptureFrameRateControl->Set(FrameRateControl_Maximum, l, lFlags)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to set on capture frame rate control. value %d, hr %x", __fxName, l, hr));
|
|
}
|
|
}
|
|
}
|
|
|
|
// set Preview frame rate control
|
|
if (m_pPreviewFrameRateControl)
|
|
{
|
|
// get valid range
|
|
if (FAILED (hr = m_pPreviewFrameRateControl->GetRange (
|
|
FrameRateControl_Current,
|
|
&min, &max, &delta, &Default, &flags)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to getrange on Preview frame rate control. %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
// adjust value
|
|
l = lValue;
|
|
// use current value - max - if input value not set
|
|
if (l==QCDEFAULT_QUALITY_UNSET) l = max;
|
|
else if (l<min) l = min;
|
|
else if (l>max) l = max;
|
|
|
|
// remember the value
|
|
m_pStreamQCRelay->Set (property, l, lFlags);
|
|
|
|
if (FAILED (hr = m_pPreviewFrameRateControl->Set(FrameRateControl_Maximum, l, lFlags)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to set on Preview frame rate control. value %d, hr %x", __fxName, l, hr));
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
// adjusted bitrate by call qc
|
|
case InnerStreamQuality_AdjMaxBitrate:
|
|
|
|
if (m_pCaptureBitrateControl == NULL)
|
|
{
|
|
if (!fReported)
|
|
{
|
|
LOG((MSP_WARN, "%s, Capture BitrateControl is NULL", __fxName));
|
|
fReported = TRUE;
|
|
}
|
|
hr = E_NOTIMPL;
|
|
break;
|
|
}
|
|
|
|
// set capture bitrate control
|
|
if (m_pCaptureBitrateControl)
|
|
{
|
|
// get valid range
|
|
if (FAILED (hr = m_pCaptureBitrateControl->GetRange (
|
|
BitrateControl_Current,
|
|
&min, &max, &delta, &Default, &flags, LAYERID)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to getrange on capture bitrate control. %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
// adjust value
|
|
l = lValue;
|
|
if (!m_pStreamQCRelay->m_fQOSAllowedToSend)
|
|
if (l > QCLIMIT_MAX_QOSNOTALLOWEDTOSEND)
|
|
l = QCLIMIT_MAX_QOSNOTALLOWEDTOSEND;
|
|
// use current value - max - if input value not set
|
|
if (l==QCDEFAULT_QUALITY_UNSET) l = max;
|
|
else if (l<min) l = min;
|
|
else if (l>max) l = max;
|
|
|
|
// remember the value
|
|
m_pStreamQCRelay->Set (property, l, lFlags);
|
|
|
|
if (FAILED (hr = m_pCaptureBitrateControl->Set(BitrateControl_Maximum, l, lFlags, LAYERID)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to set on capture bit rate control. value %d, hr %x", __fxName, l, hr));
|
|
}
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_PrefMaxBitrate:
|
|
|
|
// check input value
|
|
if (m_pCaptureBitrateControl)
|
|
{
|
|
// get valid range
|
|
if (FAILED (hr = m_pCaptureBitrateControl->GetRange (
|
|
BitrateControl_Current,
|
|
&min, &max, &delta, &Default, &flags, LAYERID)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to getrange on capture bitrate control. %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
if (lValue < min || lValue > max)
|
|
return E_INVALIDARG;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_WARN, "%s no bitratecontrol to check bitrate input.", __fxName));
|
|
}
|
|
|
|
hr = CIPConfMSPStream::Set (property, lValue, lFlags);
|
|
|
|
break;
|
|
|
|
case InnerStreamQuality_PrefMinFrameInterval:
|
|
|
|
// check input value
|
|
if (m_pCaptureFrameRateControl)
|
|
{
|
|
// get valid range
|
|
if (FAILED (hr = m_pCaptureFrameRateControl->GetRange (
|
|
FrameRateControl_Current,
|
|
&min, &max, &delta, &Default, &flags)))
|
|
{
|
|
LOG ((MSP_ERROR, "%s failed to getrange on capture frame rate control. %x", __fxName, hr));
|
|
}
|
|
else
|
|
{
|
|
if (lValue < min || lValue > max)
|
|
return E_INVALIDARG;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LOG((MSP_WARN, "%s no framerate cntl to check input.", __fxName));
|
|
}
|
|
|
|
hr = CIPConfMSPStream::Set (property, lValue, lFlags);
|
|
|
|
break;
|
|
|
|
default:
|
|
hr = CIPConfMSPStream::Set (property, lValue, lFlags);
|
|
break;
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CSubStreamVideoRecv
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
CSubStreamVideoRecv::CSubStreamVideoRecv()
|
|
: m_pFTM(NULL),
|
|
m_pStream(NULL),
|
|
m_pCurrentParticipant(NULL)
|
|
{
|
|
}
|
|
|
|
// methods called by the videorecv object.
|
|
HRESULT CSubStreamVideoRecv::Init(
|
|
IN CStreamVideoRecv * pStream
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Initialize the substream object.
|
|
|
|
Arguments:
|
|
|
|
pStream - The pointer to the stream that owns this substream.
|
|
|
|
Return Value:
|
|
|
|
HRESULT.
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"CSubStreamVideoRecv::Init, pStream %p", pStream));
|
|
|
|
// This method is called only once when the object is created. No other
|
|
// method will be called until this function succeeds. No need to lock.
|
|
_ASSERTE(m_pStream == NULL);
|
|
|
|
// initialize the terminal array so that the array is not NULL. Used for
|
|
// generating an empty enumerator if no terminal is selected.
|
|
if (!m_Terminals.Grow())
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::Init - exit E_OUTOFMEMORY"));
|
|
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
// create the marshaler.
|
|
HRESULT hr;
|
|
hr = CoCreateFreeThreadedMarshaler(GetControllingUnknown(), &m_pFTM);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "create marshaler failed, %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
// save the stream reference.
|
|
m_pStream = pStream;
|
|
(pStream->GetControllingUnknown())->AddRef();
|
|
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::Init returns S_OK"));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
#ifdef DEBUG_REFCOUNT
|
|
ULONG CSubStreamVideoRecv::InternalAddRef()
|
|
{
|
|
ULONG lRef = CComObjectRootEx<CComMultiThreadModelNoCS>::InternalAddRef();
|
|
|
|
LOG((MSP_TRACE, "SubStreamVideoRecv %p Addref, ref = %d", this, lRef));
|
|
|
|
return lRef;
|
|
}
|
|
|
|
ULONG CSubStreamVideoRecv::InternalRelease()
|
|
{
|
|
ULONG lRef = CComObjectRootEx<CComMultiThreadModelNoCS>::InternalRelease();
|
|
|
|
LOG((MSP_TRACE, "SubStreamVideoRecv %p Release, ref = %d", this, lRef));
|
|
|
|
return lRef;
|
|
}
|
|
#endif
|
|
|
|
void CSubStreamVideoRecv::FinalRelease()
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
release everything before being deleted.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::FinalRelease - enter"));
|
|
|
|
if (m_pCurrentParticipant)
|
|
{
|
|
m_pCurrentParticipant->Release();
|
|
}
|
|
|
|
for ( int i = 0; i < m_Terminals.GetSize(); i ++ )
|
|
{
|
|
m_Terminals[i]->Release();
|
|
}
|
|
m_Terminals.RemoveAll();
|
|
|
|
if (m_pStream)
|
|
{
|
|
(m_pStream->GetControllingUnknown())->Release();
|
|
}
|
|
|
|
if (m_pFTM)
|
|
{
|
|
m_pFTM->Release();
|
|
}
|
|
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::FinalRelease - exit"));
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::SelectTerminal(
|
|
IN ITTerminal * pTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Select a terminal on this substream. This method calls the same method
|
|
on the stream object to handle that.
|
|
|
|
Arguments:
|
|
pTerminal - the terminal to be selected.
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"CSubStreamVideoRecv::SelectTerminal, pTerminal %p", pTerminal));
|
|
|
|
HRESULT hr;
|
|
|
|
m_lock.Lock();
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
m_lock.Unlock();
|
|
return TAPI_E_MAXTERMINALS;
|
|
}
|
|
|
|
BOOL bFlag = m_Terminals.Add(pTerminal);
|
|
|
|
_ASSERTE(bFlag);
|
|
|
|
m_lock.Unlock();
|
|
|
|
if (!bFlag)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
// This is the refcount for the pointer in m_Terminals.
|
|
pTerminal->AddRef();
|
|
|
|
// Call the stream's select terminal to handle the state changes and also
|
|
// make sure that locking happens only from the stream to substream.
|
|
hr = m_pStream->SubStreamSelectTerminal(this, pTerminal);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"CSubStreamVideoRecv::SelectTerminal failed, hr:%x", hr));
|
|
|
|
m_lock.Lock();
|
|
|
|
m_Terminals.Remove(pTerminal);
|
|
pTerminal->Release();
|
|
|
|
m_lock.Unlock();
|
|
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::UnselectTerminal(
|
|
IN ITTerminal * pTerminal
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Unselect a terminal on this substream. This method calls the same method
|
|
on the stream object to handle that.
|
|
|
|
Arguments:
|
|
pTerminal - the terminal to be unselected.
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"CSubStreamVideoRecv::UnSelectTerminal, pTerminal %p", pTerminal));
|
|
|
|
m_lock.Lock();
|
|
if (!m_Terminals.Remove(pTerminal))
|
|
{
|
|
m_lock.Unlock();
|
|
LOG((MSP_ERROR, "SubStreamVideoRecv::UnselectTerminal, invalid terminal."));
|
|
|
|
return TAPI_E_INVALIDTERMINAL;
|
|
}
|
|
pTerminal->Release();
|
|
|
|
m_lock.Unlock();
|
|
|
|
HRESULT hr;
|
|
|
|
// Call the stream's unselect terminal to handle the state changes and also
|
|
// make sure that locking happens only from the stream to substream.
|
|
hr = m_pStream->UnselectTerminal(pTerminal);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR,
|
|
"CSubStreamVideoRecv::UnSelectTerminal failed, hr:%x", hr));
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::EnumerateTerminals(
|
|
OUT IEnumTerminal ** ppEnumTerminal
|
|
)
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"EnumerateTerminals entered. ppEnumTerminal:%x", ppEnumTerminal));
|
|
|
|
if (IsBadWritePtr(ppEnumTerminal, sizeof(VOID *)))
|
|
{
|
|
LOG((MSP_ERROR, "ppEnumTerminal is a bad pointer"));
|
|
return E_POINTER;
|
|
}
|
|
|
|
// acquire the lock before accessing the Terminal object list.
|
|
CLock lock(m_lock);
|
|
|
|
if (m_Terminals.GetData() == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::EnumerateTerminals - "
|
|
"stream appears to have been shut down - exit E_UNEXPECTED"));
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
typedef _CopyInterface<ITTerminal> CCopy;
|
|
typedef CSafeComEnum<IEnumTerminal, &__uuidof(IEnumTerminal),
|
|
ITTerminal *, CCopy> CEnumerator;
|
|
|
|
HRESULT hr;
|
|
|
|
CMSPComObject<CEnumerator> *pEnum = NULL;
|
|
|
|
hr = ::CreateCComObjectInstance(&pEnum);
|
|
|
|
if (pEnum == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "Could not create enumerator object, %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
// query for the __uuidof(IEnumTerminal) i/f
|
|
IEnumTerminal * pEnumTerminal;
|
|
hr = pEnum->_InternalQueryInterface(__uuidof(IEnumTerminal), (void**)&pEnumTerminal);
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "query enum interface failed, %x", hr));
|
|
delete pEnum;
|
|
return hr;
|
|
}
|
|
|
|
// The CSafeComEnum can handle zero-sized array.
|
|
hr = pEnum->Init(
|
|
m_Terminals.GetData(), // the begin itor
|
|
m_Terminals.GetData() + m_Terminals.GetSize(), // the end itor,
|
|
NULL, // IUnknown
|
|
AtlFlagCopy // copy the data.
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "init enumerator object failed, %x", hr));
|
|
pEnumTerminal->Release();
|
|
return hr;
|
|
}
|
|
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::EnumerateTerminals - exit S_OK"));
|
|
|
|
*ppEnumTerminal = pEnumTerminal;
|
|
|
|
return hr;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::get_Terminals(
|
|
OUT VARIANT * pVariant
|
|
)
|
|
{
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::get_Terminals - enter"));
|
|
|
|
//
|
|
// Check parameters.
|
|
//
|
|
|
|
if ( IsBadWritePtr(pVariant, sizeof(VARIANT) ) )
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"bad pointer argument - exit E_POINTER"));
|
|
|
|
return E_POINTER;
|
|
}
|
|
|
|
//
|
|
// See if this stream has been shut down. Acquire the lock before accessing
|
|
// the terminal object list.
|
|
//
|
|
|
|
CLock lock(m_lock);
|
|
|
|
if (m_Terminals.GetData() == NULL)
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"stream appears to have been shut down - exit E_UNEXPECTED"));
|
|
|
|
return E_UNEXPECTED;
|
|
}
|
|
|
|
|
|
//
|
|
// create the collection object - see mspcoll.h
|
|
//
|
|
|
|
HRESULT hr;
|
|
typedef CTapiIfCollection< ITTerminal * > TerminalCollection;
|
|
CComObject<TerminalCollection> * pCollection;
|
|
|
|
hr = ::CreateCComObjectInstance(&pCollection);
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"can't create collection - exit 0x%08x", hr));
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// get the Collection's IDispatch interface
|
|
//
|
|
|
|
IDispatch * pDispatch;
|
|
|
|
hr = pCollection->_InternalQueryInterface(__uuidof(IDispatch),
|
|
(void **) &pDispatch );
|
|
|
|
if ( FAILED(hr) )
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"QI for IDispatch on collection failed - exit 0x%08x", hr));
|
|
|
|
delete pCollection;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// Init the collection using an iterator -- pointers to the beginning and
|
|
// the ending element plus one.
|
|
//
|
|
|
|
hr = pCollection->Initialize( m_Terminals.GetSize(),
|
|
m_Terminals.GetData(),
|
|
m_Terminals.GetData() + m_Terminals.GetSize() );
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"Initialize on collection failed - exit 0x%08x", hr));
|
|
|
|
pDispatch->Release();
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// put the IDispatch interface pointer into the variant
|
|
//
|
|
|
|
LOG((MSP_ERROR, "CSubStreamVideoRecv::get_Terminals - "
|
|
"placing IDispatch value %08x in variant", pDispatch));
|
|
|
|
VariantInit(pVariant);
|
|
pVariant->vt = VT_DISPATCH;
|
|
pVariant->pdispVal = pDispatch;
|
|
|
|
LOG((MSP_TRACE, "CSubStreamVideoRecv::get_Terminals - exit S_OK"));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::get_Stream (
|
|
OUT ITStream ** ppITStream
|
|
)
|
|
{
|
|
LOG((MSP_TRACE,
|
|
"VideoRecvSubStream.get_Stream, ppITStream %x", ppITStream));
|
|
|
|
if (IsBadWritePtr(ppITStream, sizeof (VOID *)))
|
|
{
|
|
LOG((MSP_ERROR, "Bad pointer, ppITStream:%x",ppITStream));
|
|
return E_POINTER;
|
|
}
|
|
|
|
ITStream * pITStream;
|
|
HRESULT hr = m_pStream->_InternalQueryInterface(
|
|
__uuidof(ITStream),
|
|
(void **)&pITStream
|
|
);
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
LOG((MSP_ERROR, "get_Stream:QueryInterface failed: %x", hr));
|
|
return hr;
|
|
}
|
|
|
|
*ppITStream = pITStream;
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::StartSubStream()
|
|
{
|
|
return TAPI_E_NOTSUPPORTED;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::PauseSubStream()
|
|
{
|
|
return TAPI_E_NOTSUPPORTED;
|
|
}
|
|
|
|
STDMETHODIMP CSubStreamVideoRecv::StopSubStream()
|
|
{
|
|
return TAPI_E_NOTSUPPORTED;
|
|
}
|
|
|
|
BOOL CSubStreamVideoRecv::GetCurrentParticipant(
|
|
DWORD * pdwSSRC,
|
|
ITParticipant** ppITParticipant
|
|
)
|
|
{
|
|
CLock lock(m_lock);
|
|
if (m_pCurrentParticipant)
|
|
{
|
|
m_pCurrentParticipant->AddRef();
|
|
*ppITParticipant = m_pCurrentParticipant;
|
|
|
|
((CParticipant *)m_pCurrentParticipant)->GetSSRC(
|
|
(ITStream*)m_pStream,
|
|
pdwSSRC
|
|
);
|
|
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
VOID CSubStreamVideoRecv::SetCurrentParticipant(
|
|
DWORD dwSSRC,
|
|
ITParticipant * pParticipant
|
|
)
|
|
{
|
|
CLock lock(m_lock);
|
|
|
|
if (m_pCurrentParticipant)
|
|
{
|
|
m_pCurrentParticipant->Release();
|
|
}
|
|
|
|
m_pCurrentParticipant = pParticipant;
|
|
|
|
if (m_pCurrentParticipant)
|
|
{
|
|
m_pCurrentParticipant->AddRef();
|
|
}
|
|
}
|
|
|
|
BOOL CSubStreamVideoRecv::ClearCurrentTerminal()
|
|
{
|
|
CLock lock(m_lock);
|
|
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
m_Terminals[0]->Release();
|
|
m_Terminals.RemoveAt(0);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL CSubStreamVideoRecv::SetCurrentTerminal(ITTerminal * pTerminal)
|
|
{
|
|
CLock lock(m_lock);
|
|
|
|
if (m_Terminals.GetSize() > 0)
|
|
{
|
|
_ASSERTE(FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
BOOL bFlag = m_Terminals.Add(pTerminal);
|
|
|
|
// This should never fail since the terminal array has been grown
|
|
// at the init time.
|
|
_ASSERTE(bFlag);
|
|
|
|
if (bFlag)
|
|
{
|
|
pTerminal->AddRef();
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|