|
|
#include "precomp.h"
#ifndef OLDSTUFF
extern IRTP *g_pIRTP; #endif
STDMETHODIMP ImpICommChan::StandbyInit(LPGUID lpMID, LPIH323PubCap pCapObject, IMediaChannel* pMediaStreamSend) { if((!lpMID) || (!pCapObject)) return CHAN_E_INVALID_PARAM; m_MediaID = *lpMID; bIsSendDirection = TRUE; m_pMediaStream = pMediaStreamSend; m_pMediaStream->AddRef(); // keeps a cap object ref
pCapObject->AddRef(); m_pCapObject = pCapObject; return hrSuccess; }
STDMETHODIMP ImpICommChan::QueryInterface( REFIID iid, void ** ppvObject) { // this breaks the rules for the official COM QueryInterface because
// the interfaces that are queried for are not necessarily real COM
// interfaces. The reflexive property of QueryInterface would be broken in
// that case.
HRESULT hr = E_NOINTERFACE; if(!ppvObject) return hr; *ppvObject = 0; if(iid == IID_IUnknown) { *ppvObject = this; hr = hrSuccess; AddRef(); } else if((iid == IID_ICommChannel)) { *ppvObject = (ICommChannel *)this; hr = hrSuccess; AddRef(); } else if((iid == IID_ICtrlCommChannel)) { *ppvObject = (ICtrlCommChan *)this; hr = hrSuccess; AddRef(); } else if((iid == IID_IStreamSignal)) { *ppvObject = (IStreamSignal *)this; hr = hrSuccess; AddRef(); } else if((iid == IID_IAppAudioCap ) && m_pCapObject) { hr = m_pCapObject->QueryInterface(iid, ppvObject); } else if((iid == IID_IAppVidCap ) && m_pCapObject) { hr = m_pCapObject->QueryInterface(iid, ppvObject); } else if((iid == IID_IDualPubCap) && m_pCapObject) { hr = m_pCapObject->QueryInterface(iid, ppvObject); } else if(iid == IID_IVideoRender) { hr=hrSuccess; if(!m_pMediaStream && m_pH323ConfAdvise) { hr = m_pH323ConfAdvise->GetMediaChannel(&m_MediaID, bIsSendDirection, &m_pMediaStream); } if(HR_SUCCEEDED(hr)) { hr = m_pMediaStream->QueryInterface(iid, ppvObject); } } else if(iid == IID_IVideoChannel) { hr=hrSuccess; if(!m_pMediaStream && m_pH323ConfAdvise) { hr = m_pH323ConfAdvise->GetMediaChannel(&m_MediaID, bIsSendDirection, &m_pMediaStream); } if(HR_SUCCEEDED(hr)) { hr = m_pMediaStream->QueryInterface(iid, ppvObject); } } return (hr); }
ULONG ImpICommChan::AddRef() { m_uRef++; DEBUGMSG(ZONE_REFCOUNT,("ImpICommChan::AddRef:(0x%08lX)->AddRef() m_uRef = 0x%08lX\r\n",this, m_uRef )); return m_uRef; }
ULONG ImpICommChan::Release() { m_uRef--; if(m_uRef == 0) { DEBUGMSG(ZONE_REFCOUNT,("ImpICommChan::Release:(0x%08lX)->Releasing\r\n", this)); delete this; return 0; } else { DEBUGMSG(ZONE_REFCOUNT,("ImpICommChan::Release:(0x%08lX)->Release() m_uRef = 0x%08lX\r\n",this, m_uRef )); return m_uRef; } }
HRESULT ImpICommChan::GetMediaType(LPGUID pGuid) { if(!pGuid) return CHAN_E_INVALID_PARAM; *pGuid = m_MediaID; return hrSuccess; }
HRESULT ImpICommChan::IsChannelOpen(BOOL *pbOpen) { if(!pbOpen) return CHAN_E_INVALID_PARAM; *pbOpen = (IsComchOpen()) ? TRUE:FALSE; return hrSuccess; }
STDMETHODIMP ImpICommChan::GetProperty(DWORD prop, PVOID pBuf, LPUINT pcbBuf) { #define CHECKSIZE(type) if(*pcbBuf != sizeof(type)) return CHAN_E_INVALID_PARAM;
#define OUTPROP(type) *(type *)pBuf
if(!pBuf || !pcbBuf) return CHAN_E_INVALID_PARAM; switch (prop) {
case PROP_TS_TRADEOFF: CHECKSIZE(DWORD); OUTPROP(DWORD) = m_TemporalSpatialTradeoff; break; case PROP_REMOTE_TS_CAPABLE: CHECKSIZE(BOOL); OUTPROP(BOOL) = m_bPublicizeTSTradeoff; break; case PROP_CHANNEL_ENABLED: CHECKSIZE(BOOL); OUTPROP(BOOL) = (m_dwFlags && COMCH_ENABLED )? TRUE:FALSE; break; case PROP_LOCAL_FORMAT_ID: CHECKSIZE(MEDIA_FORMAT_ID); OUTPROP(MEDIA_FORMAT_ID) = m_LocalFmt; break; case PROP_REMOTE_FORMAT_ID: CHECKSIZE(MEDIA_FORMAT_ID); OUTPROP(MEDIA_FORMAT_ID) = m_RemoteFmt; break; case PROP_REMOTE_PAUSED: CHECKSIZE(BOOL); OUTPROP(BOOL) = (IsStreamingRemote())? FALSE:TRUE; break; case PROP_LOCAL_PAUSE_RECV: case PROP_LOCAL_PAUSE_SEND: CHECKSIZE(BOOL); OUTPROP(BOOL) = IsPausedLocal(); break; case PROP_VIDEO_PREVIEW_ON: CHECKSIZE(BOOL); OUTPROP(BOOL) = IsStreamingStandby(); break; case PROP_VIDEO_PREVIEW_STANDBY: CHECKSIZE(BOOL); OUTPROP(BOOL) = IsConfigStandby(); break; default: if(m_pMediaStream) { // we don't recognize this property, pass to media control
return m_pMediaStream->GetProperty(prop, pBuf, (LPUINT)pcbBuf); } else return CHAN_E_INVALID_PARAM; break; } return hrSuccess; }
// Some properties are not writeable by client code. CtrlChanSetProperty allows setting of
// those properties. This method is *not* exposed in ICommChannel
STDMETHODIMP ImpICommChan::CtrlChanSetProperty(DWORD prop, PVOID pBuf, DWORD cbBuf) { FX_ENTRY("ImpICommChan::CtrlChanSetProperty"); BOOL bTemp; HRESULT hr = hrSuccess; if(!pBuf || !pBuf || !cbBuf) return CHAN_E_INVALID_PARAM;
#define CHECKSIZEIN(type) if(cbBuf != sizeof(type)) return CHAN_E_INVALID_PARAM;
#define INPROP(type) *(type *)pBuf
switch (prop) { case PROP_TS_TRADEOFF_IND: // remote sender changed T/S tradeoff of what it is
if(bIsSendDirection) // sending (valid for receive channels only)
return CHAN_E_INVALID_PARAM; m_TemporalSpatialTradeoff = INPROP(DWORD); if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing notification 0x%08lX\r\n",_fx_, CHANNEL_VIDEO_TS_TRADEOFF)); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), CHANNEL_VIDEO_TS_TRADEOFF); } break;
case PROP_REMOTE_FORMAT_ID: CHECKSIZEIN(DWORD); m_RemoteFmt = INPROP(DWORD); break; case PROP_REMOTE_TS_CAPABLE: // only valid for receive channels
if(bIsSendDirection) return CHAN_E_INVALID_PARAM; else { CHECKSIZEIN(BOOL); m_bPublicizeTSTradeoff = INPROP(BOOL); DEBUGMSG (ZONE_COMMCHAN,("%s:remote TS tradeoff cap %d\r\n", _fx_, m_bPublicizeTSTradeoff)); } break; default: return SetProperty(prop, pBuf, cbBuf); break; } return hr; }
STDMETHODIMP ImpICommChan::Preview(MEDIA_FORMAT_ID idLocalFormat, IMediaChannel * pMediaStream) { HRESULT hr = hrSuccess; FX_ENTRY("ImpICommChan::Preview"); SHOW_OBJ_ETIME("ImpICommChan::Preview"); LPVOID lpvFormatDetails; UINT uFormatSize; if(!bIsSendDirection) { hr = CHAN_E_INVALID_PARAM; goto EXIT; } if(NULL == pMediaStream) { // preview off
if(IsStreamingStandby()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)transition to preview OFF\r\n",_fx_, (bIsSendDirection)?"send":"recv"));
//turn preview off.
// if network side is paused or closed, stop all streaming
if(!IsComchOpen() || !IsStreamingNet()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)stopping local stream\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // Stop the stream, but DO NOT UNCONFIGURE becase we want to
// be able to start later
hr = m_pMediaStream->Stop(); if(!HR_SUCCEEDED(hr)) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)Stop() returned 0x%08lx\r\n",_fx_, (bIsSendDirection)?"send":"recv", hr)); } SHOW_OBJ_ETIME("ImpICommChan::Preview - stopped"); LocalStreamFlagOff(); } // else just need to turn off flag
StandbyFlagOff(); } else DEBUGMSG(ZONE_COMMCHAN,("%s:(%s) no change (%s)\r\n",_fx_, (bIsSendDirection)?"send":"recv", "OFF")); } else { // preview on
ASSERT(m_pCapObject); if(idLocalFormat == INVALID_MEDIA_FORMAT) { hr = CHAN_E_INVALID_PARAM; goto EXIT; }
ASSERT(!(m_pMediaStream && (m_pMediaStream != pMediaStream)));
if (m_pMediaStream == NULL) { m_pMediaStream = pMediaStream; m_pMediaStream->AddRef(); } if(!IsStreamingStandby()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)transition to preview ON\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // turn preview on.
if(!IsStreamingLocal()) { ASSERT(!IsStreamingNet()); if(IsComchOpen()) { // if the channel is open, local streaming should only be off
// if the network side of the channel is paused.
//ASSERT(!IsStreamingNet());
} else {
// ensure that the stream does not come up with network send enabled
// (!!!!! override default stream behavior !!!!!)
BOOL bPause = TRUE; hr = m_pMediaStream->SetProperty( (bIsSendDirection)? PROP_PAUSE_SEND:PROP_PAUSE_RECV, &bPause, sizeof(bPause)); // get format info for the specified format
m_pCapObject->GetEncodeFormatDetails(idLocalFormat, &lpvFormatDetails, &uFormatSize);
// fire up the local stream
// this is now a two step process
hr = m_pMediaStream->Configure((BYTE*)lpvFormatDetails, uFormatSize, NULL, 0, (IUnknown*)(ImpICommChan *)this); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: m_pMediaStream->Configure returned 0x%08lX\r\n", _fx_, hr)); goto EXIT; }
m_pMediaStream->SetNetworkInterface(NULL); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: m_pMediaStream->SetNetworkInterface returned 0x%08lX\r\n", _fx_, hr)); goto EXIT; }
SHOW_OBJ_ETIME("ImpICommChan::Preview - config'd for preview"); } // Start the stream
hr = m_pMediaStream->Start(); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: m_pMediaStream->Start returned 0x%08lX\r\n", _fx_, hr)); goto EXIT; } SHOW_OBJ_ETIME("ImpICommChan::Preview - started preview");
LocalStreamFlagOn(); } // else // just need to set flag to make preview sticky
StandbyFlagOn(); } else DEBUGMSG(ZONE_COMMCHAN,("%s:(%s) no change (%s)\r\n",_fx_, (bIsSendDirection)?"send":"recv", "ON")); }
EXIT: return hr; } STDMETHODIMP ImpICommChan::PauseNetworkStream(BOOL fPause) { if(fPause) LocalPauseFlagOn(); else LocalPauseFlagOff(); return PauseNet(fPause, FALSE);
}
BOOL ImpICommChan::IsNetworkStreamPaused(VOID) { return IsPausedLocal(); }
BOOL ImpICommChan::IsRemotePaused(VOID) { return (IsStreamingRemote())? FALSE:TRUE; }
STDMETHODIMP ImpICommChan::PauseNet(BOOL bPause, BOOL bRemoteInitiated) { HRESULT hr = hrSuccess;
FX_ENTRY("ImpICommChan::PauseNet");
// issue notification
if(bRemoteInitiated) { // keep track of remote state
if(bPause) RemoteStreamFlagOff(); else RemoteStreamFlagOn(); if(!IsNotificationSupressed()) { if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing %s notification \r\n",_fx_, (bPause)?"pause":"un-pause")); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), (bPause)? CHANNEL_REMOTE_PAUSE_ON: CHANNEL_REMOTE_PAUSE_OFF); } else DEBUGMSG(ZONE_COMMCHAN,("%s:not issuing %s notification: m_pH323ConfAdvise: 0x%08lX, m_pCtlChan:0x%08lX \r\n" ,_fx_, (bPause)?"pause":"un-pause", m_pH323ConfAdvise,m_pCtlChan)); } } if(bPause && IsStreamingNet()) { ASSERT(IsComchOpen()); // deactivate the channel
DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)transition to pause\r\n",_fx_, (bIsSendDirection)?"send":"recv" ));
if(!bRemoteInitiated) { // locally initiated, so signal remote
if(bIsSendDirection) { DEBUGMSG (ZONE_COMMCHAN,("%s:signaling pause of %s channel\r\n", _fx_, (bIsSendDirection)?"send":"recv" )); // signal remote
MiscellaneousIndication mi; mi.type.choice = logicalChannelInactive_chosen; hr = m_pCtlChan->MiscChannelIndication(this, &mi); if(!HR_SUCCEEDED(hr)) { DEBUGMSG (ZONE_COMMCHAN,("%s:(%s) CC_Mute returned 0x%08lx\r\n", _fx_, (bIsSendDirection)?"send":"recv", hr)); hr = hrSuccess; // don't care about signaling error, act normal
} } } //
hr = m_pMediaStream->SetProperty( (bIsSendDirection)? PROP_PAUSE_SEND:PROP_PAUSE_RECV, &bPause, sizeof(bPause)); NetworkStreamFlagOff(); // LOOKLOOK - can't stop receive streams because they can't be restarted
// check this with GeorgeJ
// if(!IsStreamingStandby()) // need local stream for anything?
if(!IsStreamingStandby() && bIsSendDirection) // need local stream for anything?
{ DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)stopping local stream\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // can shut off local streaming now
hr = m_pMediaStream->Stop(); LocalStreamFlagOff(); } } else if(!bPause && !IsStreamingNet()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)transition to unpause\r\n",_fx_, (bIsSendDirection)?"send":"recv"));
if(IsComchOpen()) { // activate the channel
if(!bRemoteInitiated) { // locally initiated, so signal remote
if(bIsSendDirection) { DEBUGMSG (ZONE_COMMCHAN,("%s:signaling UNpause of %s channel\r\n", _fx_, (bIsSendDirection)?"send":"recv" )); // signal remote
MiscellaneousIndication mi; mi.type.choice = logicalChannelActive_chosen; hr = m_pCtlChan->MiscChannelIndication(this, &mi); if(!HR_SUCCEEDED(hr)) { DEBUGMSG (ZONE_COMMCHAN,("%s:(%s) CC_UnMute returned 0x%08lx\r\n", _fx_, (bIsSendDirection)?"send":"recv", hr)); hr = hrSuccess; // don't care about signaling error, act normal
} } } else { // remotely initiated OR special case first time channel is unpaused
// after opening
AllowNotifications(); // stop supressing notifications
} if(!IsPausedLocal()) { // MUST ensure unpaused state before starting stream ????
hr = m_pMediaStream->SetProperty( (bIsSendDirection)? PROP_PAUSE_SEND:PROP_PAUSE_RECV, &bPause, sizeof(bPause)); // check local streaming state, start it if needed
if(!IsStreamingLocal()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)starting local stream\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // need to startup stream
hr = m_pMediaStream->Start(); LocalStreamFlagOn(); } else { if(bIsSendDirection) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)already streaming locally\r\n",_fx_, (bIsSendDirection)?"send":"recv" )); DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)RESTARTING local stream\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // This is temporary until it is possible to start the
// network side of a running stream
hr = m_pMediaStream->Stop(); hr = m_pMediaStream->Start(); } } NetworkStreamFlagOn();
//
// if this is a receive video channel, make the sender send an I-frame now
//
if(!bIsSendDirection && (GetTickCount() > (m_dwLastUpdateTick + MIN_IFRAME_REQ_TICKS))) { if((MEDIA_TYPE_H323VIDEO == m_MediaID)) { MiscellaneousCommand mc; // mc.logicalChannelNumber = ?; ** call control fills this in **
mc.type.choice = videoFastUpdatePicture_chosen; // do the control channel signaling for THIS channel
hr = m_pCtlChan->MiscChannelCommand(this, &mc); } m_dwLastUpdateTick = GetTickCount(); } } } else ERRORMESSAGE(("%s:(%s) Not open: bPause=%d, streaming=%d\r\n", _fx_, (bIsSendDirection)?"send":"recv", bPause, IsStreamingNet()));
} else { ERRORMESSAGE(("%s:(%s) bPause=%d, streaming=%d\r\n", _fx_, (bIsSendDirection)?"send":"recv", bPause, IsStreamingNet())); } return hr; }
STDMETHODIMP ImpICommChan::SetProperty(DWORD prop, PVOID pBuf, UINT cbBuf) { FX_ENTRY("ImpICommChan::SetProperty"); BOOL bTemp; HRESULT hr = hrSuccess; if(!pBuf || !pBuf || !cbBuf) return CHAN_E_INVALID_PARAM;
#define CHECKSIZEIN(type) if(cbBuf != sizeof(type)) return CHAN_E_INVALID_PARAM;
#define INPROP(type) *(type *)pBuf
#define SetMediaProperty() \
if(m_pMediaStream) \ {return m_pMediaStream->SetProperty(prop, pBuf, cbBuf); } \ else hr = CHAN_E_INVALID_PARAM; switch (prop) { // (read only) case PROP_REMOTE_FORMAT_ID:
// (read only) case PROP_LOCAL_FORMAT_ID:
// (read only) case PROP_REMOTE_TS_CAPABLE:
case PROP_TS_TRADEOFF: CHECKSIZEIN(DWORD); if(bIsSendDirection) // set local T/S tradeoff, then signal remote
{ // scale value - input is 0-31, (lower number = higher quality and lower frame rate)
m_TemporalSpatialTradeoff = INPROP(DWORD); DEBUGMSG (ZONE_COMMCHAN,("%s:TS tradeoff (tx) %d\r\n", _fx_, m_TemporalSpatialTradeoff));
// change our compression
if (m_pMediaStream) { HRESULT hr; hr = m_pMediaStream->SetProperty(PROP_VIDEO_IMAGE_QUALITY, &m_TemporalSpatialTradeoff, sizeof (m_TemporalSpatialTradeoff)); } if(m_bPublicizeTSTradeoff && m_pCtlChan) // check our own capability and if in a call
{ // we said we supported TS tradeoff, so we have to signal our
// new value
MiscellaneousIndication mi; // mi.logicalChannelNumber = ?; ** call control fills this in **
mi.type.choice = MIn_tp_vdTmprlSptlTrdOff_chosen; mi.type.u.MIn_tp_vdTmprlSptlTrdOff = LOWORD(m_TemporalSpatialTradeoff); // do the control channel signaling for THIS channel
hr = m_pCtlChan->MiscChannelIndication(this, &mi); } } else // signal remote to change its T/S tradoff of its send channel
{ m_TemporalSpatialTradeoff = INPROP(DWORD); DEBUGMSG (ZONE_COMMCHAN,("%s:TS tradeoff (rx) %d\r\n", _fx_, m_TemporalSpatialTradeoff));
if(m_bPublicizeTSTradeoff && m_pCtlChan)// check remote's TS capability
{ MiscellaneousCommand mc; // mc.logicalChannelNumber = ?; ** call control fills this in **
mc.type.choice = MCd_tp_vdTmprlSptlTrdOff_chosen; mc.type.u.MCd_tp_vdTmprlSptlTrdOff = LOWORD(m_TemporalSpatialTradeoff); hr = m_pCtlChan->MiscChannelCommand(this, &mc); } else // remote said it does not support TS tradeoff
return CHAN_E_INVALID_PARAM; } break; case PROP_CHANNEL_ENABLED: CHECKSIZEIN(BOOL); if(INPROP(BOOL)) { m_dwFlags |= COMCH_ENABLED; } else { m_dwFlags &= ~COMCH_ENABLED; } break; //
// Media streaming properties
//
case PROP_LOCAL_PAUSE_RECV: case PROP_LOCAL_PAUSE_SEND: CHECKSIZEIN(BOOL); bTemp = INPROP(BOOL); if(bTemp) LocalPauseFlagOn(); else LocalPauseFlagOff(); hr = PauseNet(bTemp, FALSE); break;
case PROP_PAUSE_RECV: case PROP_PAUSE_SEND: CHECKSIZEIN(BOOL); hr = PauseNet(INPROP(BOOL), FALSE); break; // case PROP_PAUSE_RECV:
// SetMediaProperty();
// break;
case PROP_VIDEO_PREVIEW_ON: ASSERT(0); break; case PROP_VIDEO_PREVIEW_STANDBY: CHECKSIZEIN(BOOL); bTemp = INPROP(BOOL); if(bTemp) StandbyConfigFlagOn(); else StandbyConfigFlagOff(); break; default: // we don't recognize this property, pass to media control
if(m_pMediaStream) { return m_pMediaStream->SetProperty(prop, pBuf, cbBuf); } else hr = CHAN_E_INVALID_PARAM; break; } return hr; }
HRESULT ImpICommChan::EnableOpen(BOOL bEnable) { if(bEnable) { m_dwFlags |= COMCH_ENABLED; } else { m_dwFlags &= ~COMCH_ENABLED; } return hrSuccess; }
HRESULT ImpICommChan::GetLocalParams(LPVOID lpvChannelParams, UINT uBufSize) { if(!lpvChannelParams || !pLocalParams || !uBufSize) return CHAN_E_INVALID_PARAM; if(uBufSize < uLocalParamSize) return CHAN_E_INVALID_PARAM; memcpy(lpvChannelParams, pLocalParams, uLocalParamSize); return hrSuccess; }
HRESULT ImpICommChan::ConfigureStream(MEDIA_FORMAT_ID idLocalFormat) { FX_ENTRY("ImpICommChan::ConfigureStream"); HRESULT hr; ASSERT(m_pRTPChan && m_pCapObject);
LPVOID lpvFormatGoo; UINT uFormatGooSize; IUnknown *pUnknown=NULL; // get format info for Configure()
if(bIsSendDirection) { m_pCapObject->GetEncodeFormatDetails(idLocalFormat, &lpvFormatGoo, &uFormatGooSize); } else { m_pCapObject->GetDecodeFormatDetails(idLocalFormat, &lpvFormatGoo, &uFormatGooSize); } hr = m_pMediaStream->Configure((BYTE*)lpvFormatGoo, uFormatGooSize, (BYTE*)pLocalParams, uLocalParamSize, (IUnknown*)(ImpICommChan *)this); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: Configure returned 0x%08lX\r\n", _fx_, hr)); }
// SetNetworkInterface expects an IUnknown pointer
// the IUnknown wil be QI'd for either an IRTPSend or an IRTPRecv
// interface. The IUnknown should be free'd by the caller.
if (m_pRTPChan) { m_pRTPChan->QueryInterface(IID_IUnknown, (void**)&pUnknown); ASSERT(pUnknown); }
hr = m_pMediaStream->SetNetworkInterface(pUnknown); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: SetNetworkInterface returned 0x%08lX\r\n", _fx_, hr)); } if (pUnknown) { pUnknown->Release(); }
return hr; } HRESULT ImpICommChan::ConfigureCapability(LPVOID lpvRemoteChannelParams, UINT uRemoteParamSize, LPVOID lpvLocalParams, UINT uGivenLocalParamSize) { HRESULT hr= hrSuccess; if(!lpvRemoteChannelParams) return CHAN_E_INVALID_PARAM; if(pRemoteParams) { MemFree(pRemoteParams); pRemoteParams = NULL; }
// if uParamSize ==0, it means that the memory that lpvRemoteChannelParams points to
// is being supplied
if(uRemoteParamSize) { pRemoteParams = MemAlloc(uRemoteParamSize); if(pRemoteParams) { memcpy(pRemoteParams, lpvRemoteChannelParams, uRemoteParamSize); } } else pRemoteParams = lpvRemoteChannelParams; if(lpvLocalParams) { // memory for local parameters is always supplied by the caller
if (!uGivenLocalParamSize) { hr = CHAN_E_INVALID_PARAM; goto EXIT; } if(pLocalParams) { MemFree(pLocalParams); // not needed pLocalParams= NULL;
} uLocalParamSize = uGivenLocalParamSize; pLocalParams = lpvLocalParams;
} EXIT: return hr; }
HRESULT ImpICommChan::OnChannelClose(DWORD dwStatus) { HRESULT hr = hrSuccess; FX_ENTRY("ImpICommChan::OnChannelClose"); BOOL fCloseAction = FALSE;
SHOW_OBJ_ETIME("ImpICommChan::OnChannelClose");
m_dwFlags &= ~COMCH_OPEN_PENDING; switch(dwStatus) { case CHANNEL_CLOSED: DEBUGMSG(ZONE_COMMCHAN,("%s:closing (%s)\r\n" ,_fx_, (bIsSendDirection)?"send":"recv")); if(IsComchOpen()) { fCloseAction = TRUE; m_dwFlags &= ~COMCH_OPEN; } else { ERRORMESSAGE(("%s: %d notification when not open (%s)\r\n", _fx_, dwStatus,(bIsSendDirection)?"send":"recv")); } break; //case CHANNEL_REJECTED:
//case CHANNEL_NO_CAPABILITY:
default: break; } // clear general purpose channel handle
dwhChannel = 0; // LOOKLOOK **** RIGHT HERE ***
// ** need to notify the UI of the channel event ON_CLOSING so that the last
// frame can be grabbed for rendering (a still picture is better than a black window)
// LOOKLOOK **** RIGHT HERE ***
// Now check preview state
if(IsStreamingStandby() && bIsSendDirection ) { if (m_pMediaStream != NULL) { DEBUGMSG(ZONE_COMMCHAN,("%s:transition back to preview\r\n" ,_fx_)); // need to stop sending and reconfigure for preview
// make sure send is paused
DWORD dwProp = TRUE; hr = m_pMediaStream->SetProperty (PROP_PAUSE_SEND,&dwProp, sizeof(dwProp)); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: m_pMediaStream->SetProperty returned 0x%08lX\r\n", _fx_, hr)); // do what now?
}
NetworkStreamFlagOff(); hr = m_pMediaStream->Stop(); LocalStreamFlagOff(); StandbyFlagOff(); ASSERT(hr == S_OK); } else { NetworkStreamFlagOff(); LocalStreamFlagOff(); } if(fCloseAction) { // Cleanup RTP session. This is a NOP if the opposite direction is still open.
if (m_pRTPChan) { m_pRTPChan->Release(); m_pRTPChan = NULL; } } } else // not previewing
{ //
// Stop the media stream
//
if (m_pMediaStream) { hr = m_pMediaStream->Stop(); // probably not necessary
ASSERT(hr == S_OK); // implement "capture device standby": don't unconfigure if
// the standby flag is set and it is a send stream.
if(!IsConfigStandby() || !bIsSendDirection) { if(!bIsSendDirection) // keep send stream reference until *this* object is released
{ m_pMediaStream->Release(); m_pMediaStream = NULL; } } } SHOW_OBJ_ETIME("ImpICommChan::OnChannelClose - stream stopped");
if(fCloseAction) { // Cleanup RTP session. This is a NOP if the opposite direction is still open.
if (m_pRTPChan) { m_pRTPChan->Release(); m_pRTPChan = NULL; } } StreamFlagsOff(); }// end if not previewing
if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing notification 0x%08lX\r\n",_fx_, dwStatus)); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), dwStatus); }
return hr; } HRESULT ImpICommChan::OnChannelOpening() { ASSERT((m_dwFlags & COMCH_OPEN_PENDING) ==0); m_dwFlags |= COMCH_OPEN_PENDING; return hrSuccess; }
HRESULT ImpICommChan::OnChannelOpen(DWORD dwStatus) { HRESULT hr; BOOL bConfigured = FALSE, bNewStream = FALSE; // these bools make error cleanup cleaner
FX_ENTRY("ImpICommChan::OnChannelOpen");
SHOW_OBJ_ETIME("ImpICommChan::OnChannelOpen"); // the open is no longer pending, regardless of success or failure
m_dwFlags &= ~COMCH_OPEN_PENDING; m_dwLastUpdateTick = 0; // reset tick count of last I-frame request so that one
// will be requested
if(IsComchOpen()) { ERRORMESSAGE(("%s: %d notification when open (%s)\r\n", _fx_, dwStatus, (bIsSendDirection)?"send":"recv")); } switch(dwStatus) { case CHANNEL_OPEN: m_dwFlags |= (COMCH_OPEN | COMCH_SUPPRESS_NOTIFICATION); break; default: dwStatus = CHANNEL_OPEN_ERROR; // fall through to notification
case CHANNEL_REJECTED: case CHANNEL_NO_CAPABILITY: goto NOTIFICATION; break; } // The channel is open as far as call control is concerned.
// if previewing, the stream already exists. We don't want another, nor do we
// want to tear it down at channel close time or in error cases
if(!m_pMediaStream) { ASSERT(!IsStreamingLocal() &&m_pH323ConfAdvise); // can't be streaming without a stream
bNewStream = TRUE; // Associate the media streaming endpoint with this channel
// see above
hr = m_pH323ConfAdvise->GetMediaChannel(&m_MediaID, bIsSendDirection, &m_pMediaStream); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: m_pH323ConfAdvise->GetMediaChannel returned 0x%08lX\r\n", _fx_, hr)); goto ERROR_NOTIFICATION; } } if(IsStreamingLocal()) { DEBUGMSG(ZONE_COMMCHAN,("%s:(%s)transition:preview -> send\r\n",_fx_, (bIsSendDirection)?"send":"recv")); // need to stop stream while configuring ( ***** check w/ RichP ******)
hr = m_pMediaStream->Stop(); LocalStreamFlagOff(); }
// notify upper layers of channel open now
if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing CHANNEL_OPEN notification\r\n",_fx_)); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), dwStatus); }
dwStatus = CHANNEL_ACTIVE; // new status! notification is posted below
ASSERT(m_pRTPChan);
// get format info for Configure()
hr = ConfigureStream(m_LocalFmt); if(!HR_SUCCEEDED(hr)) { ERRORMESSAGE(("%s: Configure returned 0x%08lX\r\n", _fx_, hr)); goto ERROR_NOTIFICATION; } SHOW_OBJ_ETIME("ImpICommChan::OnChannelOpen - configured stream"); bConfigured = TRUE; // turn on flow to the network
// SupressNotification() // pre-initialized above in both CHANNEL_OPEN_xxx cases
PauseNet(FALSE, TRUE); // unpause,
//dwStatus = CHANNEL_ACTIVE;
SHOW_OBJ_ETIME("ImpICommChan::OnChannelOpen - unpaused"); NOTIFICATION: if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing notification 0x%08lX\r\n",_fx_, dwStatus)); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), dwStatus); } else DEBUGMSG(ZONE_COMMCHAN,("%s: *** not issuing notification 0x%08lX m_pH323ConfAdvise: 0x%08lX, m_pCtlChan:0x%08lX \r\n" ,_fx_, dwStatus,m_pH323ConfAdvise,m_pCtlChan)); SHOW_OBJ_ETIME("ImpICommChan::OnChannelOpen - done ");
return hr; ERROR_NOTIFICATION: dwStatus = CHANNEL_OPEN_ERROR; if(m_pMediaStream) { if(bNewStream) // was the media stream just created?
{ m_pMediaStream->Release(); m_pMediaStream = NULL; } } if(m_pH323ConfAdvise && m_pCtlChan) { DEBUGMSG(ZONE_COMMCHAN,("%s:issuing notification 0x%08lX\r\n",_fx_, dwStatus)); m_pH323ConfAdvise->ChannelEvent(this, m_pCtlChan->GetIConnIF(), dwStatus); } else DEBUGMSG(ZONE_COMMCHAN,("%s: *** not issuing notification 0x%08lX m_pH323ConfAdvise: 0x%08lX, m_pCtlChan:0x%08lX \r\n" ,_fx_, dwStatus,m_pH323ConfAdvise,m_pCtlChan)); // close the channel.
if(m_pCtlChan) { // close channel, but hr already contains the relevant return code
m_pCtlChan->CloseChannel(this); } return hr; }
HRESULT ImpICommChan::Open(MEDIA_FORMAT_ID idLocalFormat, IH323Endpoint *pConnection) { HRESULT hr; MEDIA_FORMAT_ID idRemoteFormat; IConfAdvise * pConfAdvise = NULL; if((m_dwFlags & COMCH_OPEN_PENDING) || IsComchOpen() || (idLocalFormat == INVALID_MEDIA_FORMAT) || !pConnection) return CHAN_E_INVALID_PARAM;
if(!m_pCtlChan) // this channel is not part of a call
{ hr = pConnection->QueryInterface(IID_IConfAdvise, (void **)&pConfAdvise); if(!HR_SUCCEEDED(hr)) goto EXIT; hr = pConfAdvise->AddCommChannel(this); if(!HR_SUCCEEDED(hr)) goto EXIT; ASSERT(m_pCtlChan && m_pCapObject); } hr = m_pCapObject->ResolveToLocalFormat(idLocalFormat, &idRemoteFormat); if(!HR_SUCCEEDED(hr)) goto EXIT; // start the control channel stuff needed to open the channel
hr = m_pCtlChan->OpenChannel((ICtrlCommChan*)this, m_pCapObject, idLocalFormat, idRemoteFormat); EXIT: if(pConfAdvise) pConfAdvise->Release(); return hr; } HRESULT ImpICommChan::Close() { HRESULT hr = CHAN_E_INVALID_PARAM; if(!IsComchOpen() || !m_pCtlChan) goto EXIT; if(!bIsSendDirection) goto EXIT; hr = m_pCtlChan->CloseChannel(this);
EXIT: return hr; }
HRESULT ImpICommChan::BeginControlSession(IControlChannel *pCtlChan, LPIH323PubCap pCapObject) { // this channel is now "in a call".
// LOOKLOOK - it might help to notify (ICommChannel notifications to client)
// that the channel is part of a call now.
ASSERT((m_pCtlChan == NULL) && pCtlChan && pCapObject); if(m_pCapObject) { m_pCapObject->Release(); } m_pCtlChan = pCtlChan; m_pCapObject = pCapObject; m_pCapObject->AddRef(); return hrSuccess; } HRESULT ImpICommChan::EndControlSession() { // this channel is no longer "in a call".
m_pCtlChan = NULL; return hrSuccess; }
BOOL ImpICommChan::SelectPorts(LPIControlChannel pCtlChannel) { // create the RTP channel
HRESULT hr; PSOCKADDR_IN psin=NULL; pCtlChannel->GetLocalAddress(&psin);
PORT savedPort = psin->sin_port; if (!m_pRTPChan) { UINT sessFlags = bIsSendDirection ? SESSIONF_SEND : SESSIONF_RECV; UINT sessId; GUID mediaGuid; GetMediaType(&mediaGuid); if (mediaGuid == MEDIA_TYPE_H323VIDEO) { sessFlags |= SESSIONF_VIDEO; sessId = 2; } else { sessId = 1; sessFlags |= SESSIONF_AUDIO; } psin->sin_port = 0; // zero port forces RTP to choose a port
hr = g_pIRTP->OpenSession(sessId, sessFlags, (BYTE *)psin, sizeof(PSOCKADDR_IN), &m_pRTPChan); } else hr = m_pRTPChan->SetLocalAddress((BYTE *)psin,sizeof(SOCKADDR_IN)); psin->sin_port = savedPort;
return hr==S_OK; }
// get the address and port of the base port that was selected by SelectPorts().
// in this typical implementation, that is the address/port of the RTCP channel
PSOCKADDR_IN ImpICommChan::GetLocalAddress() { #ifdef OLDSTUFF
return m_pRTPChan ? m_pRTPChan->GetChannelDescription()->pLocalAddr : NULL; #else
const BYTE *pAddr; UINT cbAddr; HRESULT hr; hr = m_pRTPChan->GetLocalAddress(&pAddr, &cbAddr); return (SUCCEEDED(hr)) ? (PSOCKADDR_IN) pAddr : NULL; #endif
}
STDMETHODIMP ImpICommChan::GetRemoteAddress(PSOCKADDR_IN pAddrOutput) { HRESULT hr; if (!pAddrOutput) { return CHAN_E_INVALID_PARAM; } const BYTE *pAddr; UINT cbAddr; hr = m_pRTPChan->GetRemoteRTPAddress(&pAddr, &cbAddr); if(SUCCEEDED(hr)) { ASSERT(cbAddr == sizeof(SOCKADDR_IN)); *pAddrOutput = *((PSOCKADDR_IN) pAddr); } return hrSuccess; }
UINT ImpICommChan::Reset() { UINT uret; ASSERT(!IsComchOpen()); if (m_pRTPChan) { uret = m_pRTPChan->Release(); m_pRTPChan = NULL; } else uret = 0; return uret; } PORT ImpICommChan::GetLocalRTPPort() { #ifdef OLDSTUFF
return (m_pRTPChan ? ntohs(m_pRTPChan->GetChannelDescription()->pLocalAddr->sin_port) : 0); #else
const BYTE *pAddr; UINT cbAddr; HRESULT hr; hr = m_pRTPChan->GetLocalAddress(&pAddr, &cbAddr); return (SUCCEEDED(hr)) ? ntohs(((PSOCKADDR_IN) pAddr)->sin_port) : 0; #endif
}
PORT ImpICommChan::GetLocalRTCPPort() { #ifdef OLDSTUFF
return (m_pRTPChan ? ntohs(m_pRTPChan->GetChannelDescription()->pLocalRTCPAddr->sin_port) : 0); #else
const BYTE *pAddr; UINT cbAddr; HRESULT hr; hr = m_pRTPChan->GetLocalAddress(&pAddr, &cbAddr); return (SUCCEEDED(hr)) ? ntohs(((PSOCKADDR_IN) pAddr)->sin_port)+1 : 0; #endif
}
HRESULT ImpICommChan::AcceptRemoteRTCPAddress(PSOCKADDR_IN pSinC) { HRESULT hr; #ifdef OLDSTUFF
if (!m_pRTPChan) { RTPCHANNELDESC chanDesc = {0}; GetMediaType(&chanDesc.mediaId); chanDesc.pRemoteRTCPAddr = pSinC; hr = CreateRTPChannel(&chanDesc, &m_pRTPChan); } else hr = m_pRTPChan->SetRemoteAddresses(NULL,pSinC); #else
hr = m_pRTPChan->SetRemoteRTCPAddress((BYTE *)pSinC, sizeof(SOCKADDR_IN)); #endif
return hr; }
HRESULT ImpICommChan::AcceptRemoteAddress(PSOCKADDR_IN pSinD) { HRESULT hr; hr = m_pRTPChan->SetRemoteRTPAddress((BYTE *)pSinD, sizeof(SOCKADDR_IN)); return hr; }
HRESULT ImpICommChan::SetAdviseInterface(IH323ConfAdvise *pH323ConfAdvise) { if (!pH323ConfAdvise) { return CHAN_E_INVALID_PARAM; } m_pH323ConfAdvise = pH323ConfAdvise; return hrSuccess; } STDMETHODIMP ImpICommChan::PictureUpdateRequest() { FX_ENTRY ("ImpICommChan::PictureUpdateRequest"); HRESULT hr; if (!m_pCtlChan) { return CHAN_E_NOT_OPEN; } if(bIsSendDirection || (MEDIA_TYPE_H323VIDEO != m_MediaID)) { return CHAN_E_INVALID_PARAM; } // issue miscellaneous command for picture update
MiscellaneousCommand mc; // mc.logicalChannelNumber = ?; ** call control fills this in **
mc.type.choice = videoFastUpdatePicture_chosen; // do the control channel signaling for THIS channel
hr = m_pCtlChan->MiscChannelCommand(this, &mc); // record the tick count of this command
m_dwLastUpdateTick = GetTickCount(); return hr; }
STDMETHODIMP ImpICommChan::GetVersionInfo( PCC_VENDORINFO *ppLocalVendorInfo, PCC_VENDORINFO *ppRemoteVendorInfo) { FX_ENTRY ("ImpICommChan::GetVersionInfo"); if (!m_pCtlChan) { return CHAN_E_INVALID_PARAM; } return m_pCtlChan->GetVersionInfo(ppLocalVendorInfo, ppRemoteVendorInfo); }
ImpICommChan::ImpICommChan () :pRemoteParams(NULL), m_pMediaStream(NULL), pLocalParams(NULL), uLocalParamSize(0), m_pCtlChan(NULL), m_pH323ConfAdvise(NULL), m_pCapObject(NULL), m_dwFlags(0), dwhChannel(0), m_LocalFmt(INVALID_MEDIA_FORMAT), m_RemoteFmt(INVALID_MEDIA_FORMAT), m_TemporalSpatialTradeoff(0), // default to highest resolution
m_bPublicizeTSTradeoff(FALSE), m_uRef(1) { ZeroMemory(&m_MediaID, sizeof(m_MediaID)); }
ImpICommChan::~ImpICommChan () { if(pRemoteParams) MemFree(pRemoteParams); if(pLocalParams) MemFree(pLocalParams); if(m_pMediaStream) { m_pMediaStream->Stop(); // probably not necessary
m_pMediaStream->Release(); m_pMediaStream = NULL; } if(m_pCapObject) m_pCapObject->Release(); }
|