Source code of Windows XP (NT5)
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.

2942 lines
69 KiB

  1. /*++
  2. Copyright (c) 1997 Microsoft Corporation
  3. Module Name:
  4. confcall.cpp
  5. Abstract:
  6. This module contains implementation of CIPConfMSPCall.
  7. Author:
  8. Mu Han (muhan) 5-September-1998
  9. --*/
  10. #include "stdafx.h"
  11. #include <confpdu.h>
  12. CIPConfMSPCall::CIPConfMSPCall()
  13. : m_fLocalInfoRetrieved(FALSE),
  14. m_fShutDown(FALSE),
  15. m_dwIPInterface(INADDR_ANY),
  16. m_LoopbackMode(MM_NO_LOOPBACK),
  17. m_hAudioRTPSession(NULL),
  18. m_hVideoRTPSession(NULL),
  19. m_pIAudioDuplexController(NULL),
  20. m_fCallStarted(FALSE),
  21. m_pApplicationID(NULL),
  22. m_pApplicationGUID(NULL),
  23. m_pSubIDs(NULL),
  24. m_pCallQCRelay(NULL)
  25. {
  26. ZeroMemory(m_InfoItems, sizeof(m_InfoItems));
  27. }
  28. CIPConfMSPCall::~CIPConfMSPCall()
  29. {
  30. if (m_pApplicationID)
  31. {
  32. SysFreeString(m_pApplicationID);
  33. }
  34. if (m_pApplicationGUID)
  35. {
  36. SysFreeString(m_pApplicationGUID);
  37. }
  38. if (m_pSubIDs)
  39. {
  40. SysFreeString(m_pSubIDs);
  41. }
  42. if (m_pCallQCRelay)
  43. {
  44. delete m_pCallQCRelay;
  45. }
  46. }
  47. STDMETHODIMP CIPConfMSPCall::CreateStream(
  48. IN long lMediaType,
  49. IN TERMINAL_DIRECTION Direction,
  50. IN OUT ITStream ** ppStream
  51. )
  52. {
  53. // This MSP doesn't support creating new streams on the fly.
  54. return TAPI_E_NOTSUPPORTED;
  55. }
  56. STDMETHODIMP CIPConfMSPCall::RemoveStream(
  57. IN ITStream * pStream
  58. )
  59. {
  60. // This MSP doesn't support removing streams either.
  61. return TAPI_E_NOTSUPPORTED;
  62. }
  63. HRESULT CIPConfMSPCall::InitializeLocalParticipant()
  64. /*++
  65. Routine Description:
  66. This function uses the RTP filter to find out the local information that
  67. will be used in the call. The infomation is stored in a local participant
  68. object.
  69. Arguments:
  70. Return Value:
  71. HRESULT.
  72. --*/
  73. {
  74. m_fLocalInfoRetrieved = FALSE;
  75. // Create the RTP fitler.
  76. IRtpSession *pIRtpSession;
  77. HRESULT hr = CoCreateInstance(
  78. __uuidof(MSRTPSourceFilter),
  79. NULL,
  80. CLSCTX_INPROC_SERVER | CLSCTX_NO_CODE_DOWNLOAD,
  81. __uuidof(IRtpSession),
  82. (void **) &pIRtpSession
  83. );
  84. if (FAILED(hr))
  85. {
  86. LOG((MSP_ERROR, "can't create RTP filter for local info. %x", hr));
  87. return hr;
  88. }
  89. // Get the available local SDES info from the filter.
  90. WCHAR Buffer[MAX_PARTICIPANT_TYPED_INFO_LENGTH + 1];
  91. for (int i = 0; i < NUM_SDES_ITEMS; i ++)
  92. {
  93. DWORD dwLen = MAX_PARTICIPANT_TYPED_INFO_LENGTH;
  94. hr = pIRtpSession->GetSdesInfo(
  95. RTPSDES_CNAME + i,
  96. Buffer,
  97. &dwLen,
  98. 0 // local participant
  99. );
  100. if (SUCCEEDED(hr) && dwLen > 0)
  101. {
  102. _ASSERT(dwLen <= MAX_PARTICIPANT_TYPED_INFO_LENGTH);
  103. // allocate memory to store the string.
  104. m_InfoItems[i] = (WCHAR *)malloc((dwLen) * sizeof(WCHAR));
  105. if (m_InfoItems[i] == NULL)
  106. {
  107. LOG((MSP_ERROR, "out of mem for local info"));
  108. pIRtpSession->Release();
  109. return E_OUTOFMEMORY;
  110. }
  111. lstrcpynW(m_InfoItems[i], Buffer, dwLen);
  112. }
  113. }
  114. pIRtpSession->Release();
  115. m_fLocalInfoRetrieved = TRUE;
  116. return S_OK;
  117. }
  118. HRESULT CIPConfMSPCall::Init(
  119. IN CMSPAddress * pMSPAddress,
  120. IN MSP_HANDLE htCall,
  121. IN DWORD dwReserved,
  122. IN DWORD dwMediaType
  123. )
  124. /*++
  125. Routine Description:
  126. This method is called when the call is first created. It sets
  127. up the streams based on the mediatype specified.
  128. Arguments:
  129. pMSPAddress - The pointer to the address object.
  130. htCall - The handle to the Call in TAPI's space.
  131. Used in sending events.
  132. dwReserved - Reserved.
  133. dwMediaType - The media type of this call.
  134. Return Value:
  135. HRESULT.
  136. --*/
  137. {
  138. LOG((MSP_TRACE,
  139. "IPConfMSP call %x initialize entered,"
  140. " pMSPAddress:%x, htCall %x, dwMediaType %x",
  141. this, pMSPAddress, htCall, dwMediaType
  142. ));
  143. #ifdef DEBUG_REFCOUNT
  144. if (g_lStreamObjects != 0)
  145. {
  146. LOG((MSP_ERROR, "Number of Streams alive: %d", g_lStreamObjects));
  147. // DebugBreak();
  148. }
  149. #endif
  150. // initialize the participant array so that the array is not NULL.
  151. if (!m_Participants.Grow())
  152. {
  153. LOG((MSP_ERROR, "out of mem for participant list"));
  154. return E_OUTOFMEMORY;
  155. }
  156. // Call the base class's init.
  157. HRESULT hr= CMSPCallMultiGraph::Init(
  158. pMSPAddress,
  159. htCall,
  160. dwReserved,
  161. dwMediaType
  162. );
  163. if (FAILED(hr))
  164. {
  165. LOG((MSP_ERROR, "MSPCallMultiGraph init failed:%x", hr));
  166. return hr;
  167. }
  168. // create the quality control relay for this call.
  169. m_pCallQCRelay = new CCallQualityControlRelay ();
  170. if (NULL == m_pCallQCRelay)
  171. {
  172. LOG((MSP_ERROR, "call init: failed to create call quality control relay:%x", hr));
  173. return E_OUTOFMEMORY;
  174. }
  175. // initialize qc relay, a thread will be started
  176. if (FAILED (hr = m_pCallQCRelay->Initialize (this)))
  177. {
  178. LOG ((MSP_ERROR, "call init: failed to initialize qc relay. %x", hr));
  179. return hr;
  180. }
  181. // create streams based on the media types.
  182. if (dwMediaType & TAPIMEDIATYPE_AUDIO)
  183. {
  184. ITStream * pStream;
  185. // create a stream object.
  186. hr = InternalCreateStream(TAPIMEDIATYPE_AUDIO, TD_RENDER, &pStream);
  187. if (FAILED(hr))
  188. {
  189. LOG((MSP_ERROR, "create audio render stream failed:%x", hr));
  190. return hr;
  191. }
  192. // The stream is already in our array, we don't need this pointer.
  193. pStream->Release();
  194. // create a stream object.
  195. hr = InternalCreateStream(TAPIMEDIATYPE_AUDIO, TD_CAPTURE, &pStream);
  196. if (FAILED(hr))
  197. {
  198. LOG((MSP_ERROR, "create audio capture stream failed:%x", hr));
  199. return hr;
  200. }
  201. // The stream is already in our array, we don't need this pointer.
  202. pStream->Release();
  203. }
  204. if (dwMediaType & TAPIMEDIATYPE_VIDEO)
  205. {
  206. ITStream * pStream;
  207. // create a stream object.
  208. hr = InternalCreateStream(TAPIMEDIATYPE_VIDEO, TD_RENDER, &pStream);
  209. if (FAILED(hr))
  210. {
  211. LOG((MSP_ERROR, "create video render stream failed:%x", hr));
  212. return hr;
  213. }
  214. // The stream is already in our array, we don't need this pointer.
  215. pStream->Release();
  216. // create a stream object.
  217. hr = InternalCreateStream(TAPIMEDIATYPE_VIDEO, TD_CAPTURE, &pStream);
  218. if (FAILED(hr))
  219. {
  220. LOG((MSP_ERROR, "create video capture stream failed:%x", hr));
  221. return hr;
  222. }
  223. // The stream is already in our array, we don't need this pointer.
  224. pStream->Release();
  225. }
  226. DWORD dwLoopback = 0;
  227. if (TRUE == ::GetRegValue(gszMSPLoopback, &dwLoopback) && dwLoopback != 0)
  228. {
  229. m_LoopbackMode = MULTICAST_LOOPBACK_MODE(dwLoopback);
  230. }
  231. m_fShutDown = FALSE;
  232. return S_OK;
  233. }
  234. HRESULT CIPConfMSPCall::ShutDown()
  235. /*++
  236. Routine Description:
  237. Shutdown the call.
  238. Arguments:
  239. Return Value:
  240. HRESULT.
  241. --*/
  242. {
  243. InternalShutDown();
  244. // acquire the lock on call.
  245. m_lock.Lock();
  246. for (int i = 0; i < NUM_SDES_ITEMS; i ++)
  247. {
  248. if (m_InfoItems[i])
  249. {
  250. free(m_InfoItems[i]);
  251. m_InfoItems[i] = NULL;
  252. }
  253. }
  254. m_lock.Unlock();
  255. return S_OK;
  256. }
  257. HRESULT CIPConfMSPCall::InternalShutDown()
  258. /*++
  259. Routine Description:
  260. First call the base class's shutdown and then release all the participant
  261. objects.
  262. Arguments:
  263. Return Value:
  264. HRESULT.
  265. --*/
  266. {
  267. LOG((MSP_TRACE, "ConfMSPCall.InternalShutdown, entered"));
  268. // acquire the lock on the call.
  269. m_lock.Lock();
  270. if (m_fShutDown)
  271. {
  272. LOG((MSP_TRACE, "ConfMSPCall::InterShutdown, already shutdown"));
  273. m_lock.Unlock ();
  274. return S_OK;
  275. }
  276. m_fShutDown = TRUE;
  277. if (m_pCallQCRelay)
  278. {
  279. m_pCallQCRelay->Shutdown ();
  280. }
  281. int i;
  282. // Shutdown all the streams
  283. for (i = m_Streams.GetSize() - 1; i >= 0; i --)
  284. {
  285. UnregisterWaitEvent(i);
  286. ((CMSPStream*)m_Streams[i])->ShutDown();
  287. }
  288. m_ThreadPoolWaitBlocks.RemoveAll();
  289. // release all the streams
  290. for (i = m_Streams.GetSize() - 1; i >= 0; i --)
  291. {
  292. m_Streams[i]->Release();
  293. }
  294. m_Streams.RemoveAll();
  295. if (m_pIAudioDuplexController)
  296. {
  297. m_pIAudioDuplexController->Release();
  298. m_pIAudioDuplexController = NULL;
  299. }
  300. m_lock.Unlock();
  301. // release all the participants
  302. m_ParticipantLock.Lock();
  303. for (i = 0; i < m_Participants.GetSize(); i ++)
  304. {
  305. m_Participants[i]->Release();
  306. }
  307. m_Participants.RemoveAll();
  308. m_ParticipantLock.Unlock();
  309. return S_OK;
  310. }
  311. template <class T>
  312. HRESULT CreateStreamHelper(
  313. IN T * pT,
  314. IN HANDLE hAddress,
  315. IN CIPConfMSPCall* pMSPCall,
  316. IN IMediaEvent * pGraph,
  317. IN DWORD dwMediaType,
  318. IN TERMINAL_DIRECTION Direction,
  319. OUT ITStream ** ppITStream
  320. )
  321. /*++
  322. Routine Description:
  323. Create a stream object and initialize it. This method is called internally
  324. to create a stream object of different class.
  325. Arguments:
  326. hAddress - the handle to the address object.
  327. pCall - the call object.
  328. pGraph - the filter graph for this stream.
  329. dwMediaType - the media type of the stream.
  330. Direction - the direction of the steam.
  331. ppITStream - the interface on this stream object.
  332. Return Value:
  333. HRESULT.
  334. --*/
  335. {
  336. ENTER_FUNCTION ("CreateStreamHelper");
  337. CComObject<T> * pCOMMSPStream;
  338. HRESULT hr;
  339. hr = ::CreateCComObjectInstance(&pCOMMSPStream);
  340. if (NULL == pCOMMSPStream)
  341. {
  342. LOG((MSP_ERROR, "CreateMSPStream:could not create stream:%x", hr));
  343. return hr;
  344. }
  345. // get the interface pointer.
  346. hr = pCOMMSPStream->_InternalQueryInterface(
  347. __uuidof(ITStream),
  348. (void **)ppITStream
  349. );
  350. if (FAILED(hr))
  351. {
  352. LOG((MSP_ERROR, "CreateMSPStream:QueryInterface failed: %x", hr));
  353. delete pCOMMSPStream;
  354. return hr;
  355. }
  356. // Initialize the object.
  357. hr = pCOMMSPStream->Init(
  358. hAddress,
  359. pMSPCall,
  360. pGraph,
  361. dwMediaType,
  362. Direction
  363. );
  364. if (FAILED(hr))
  365. {
  366. LOG((MSP_ERROR, "CreateMSPStream:call init failed: %x", hr));
  367. (*ppITStream)->Release();
  368. return hr;
  369. }
  370. // retrieve inner call quality control
  371. IInnerCallQualityControl * pIInnerCallQC;
  372. if (FAILED (hr = pMSPCall->_InternalQueryInterface (
  373. __uuidof (IInnerCallQualityControl),
  374. (void **)&pIInnerCallQC
  375. )))
  376. {
  377. LOG ((MSP_ERROR, "%s failed to retrieve inner call qc relay: %x", __fxName, hr));
  378. (*ppITStream)->Release ();
  379. return hr;
  380. }
  381. // retrieve inner stream quality control
  382. IInnerStreamQualityControl *pIInnerStreamQC;
  383. if (FAILED (hr = (*ppITStream)->QueryInterface (
  384. __uuidof (IInnerStreamQualityControl),
  385. (void **)&pIInnerStreamQC
  386. )))
  387. {
  388. LOG ((MSP_ERROR, "%s failed to retrieve inner stream qc relay: %x", __fxName, hr));
  389. pIInnerCallQC->Release ();
  390. (*ppITStream)->Release ();
  391. return hr;
  392. }
  393. // store inner call qc
  394. if (FAILED (hr = pIInnerStreamQC->LinkInnerCallQC (pIInnerCallQC)))
  395. {
  396. LOG ((MSP_ERROR, "%s failed to setup inner call qc on stream, %x", __fxName, hr));
  397. pIInnerCallQC->Release ();
  398. pIInnerStreamQC->Release ();
  399. (*ppITStream)->Release ();
  400. return hr;
  401. }
  402. // register inner stream qc on the call
  403. hr = pIInnerCallQC->RegisterInnerStreamQC (pIInnerStreamQC);
  404. pIInnerStreamQC->Release ();
  405. pIInnerCallQC->Release ();
  406. if (FAILED (hr))
  407. {
  408. LOG ((MSP_ERROR, "%s failed to register inner stream qc relay: %x", __fxName, hr));
  409. (*ppITStream)->Release ();
  410. return hr;
  411. }
  412. return S_OK;
  413. }
  414. HRESULT CIPConfMSPCall::CreateStreamObject(
  415. IN DWORD dwMediaType,
  416. IN TERMINAL_DIRECTION Direction,
  417. IN IMediaEvent * pGraph,
  418. IN ITStream ** ppStream
  419. )
  420. /*++
  421. Routine Description:
  422. Create a media stream object based on the mediatype and direction.
  423. Arguments:
  424. pMediaType - TAPI3 media type.
  425. Direction - direction of this stream.
  426. IMediaEvent - The filter graph used in this stream.
  427. ppStream - the return pointer of the stream interface
  428. Return Value:
  429. HRESULT.
  430. --*/
  431. {
  432. LOG((MSP_TRACE, "CreateStreamObject, entered"));
  433. HRESULT hr = S_OK;
  434. ITStream * pIMSPStream = NULL;
  435. // Create a stream object based on the media type.
  436. if (dwMediaType == TAPIMEDIATYPE_AUDIO)
  437. {
  438. if (Direction == TD_RENDER)
  439. {
  440. CStreamAudioRecv *pAudioRecv = NULL;
  441. hr = ::CreateStreamHelper(
  442. pAudioRecv,
  443. m_pMSPAddress,
  444. this,
  445. pGraph,
  446. TAPIMEDIATYPE_AUDIO,
  447. TD_RENDER,
  448. &pIMSPStream
  449. );
  450. LOG((MSP_TRACE, "create audio receive:%x, hr:%x", pIMSPStream,hr));
  451. if (FAILED(hr))
  452. {
  453. LOG((MSP_ERROR, "create stream failed. %x", hr));
  454. return hr;
  455. }
  456. if (FAILED(hr = InitFullDuplexControler()))
  457. {
  458. LOG((MSP_ERROR, "Create full duplex controller failed. %x", hr));
  459. }
  460. else
  461. {
  462. ((CStreamAudioRecv *)pIMSPStream)->
  463. SetFullDuplexController(m_pIAudioDuplexController);
  464. }
  465. }
  466. else if (Direction == TD_CAPTURE)
  467. {
  468. CStreamAudioSend *pAudioSend = NULL;
  469. hr = ::CreateStreamHelper(
  470. pAudioSend,
  471. m_pMSPAddress,
  472. this,
  473. pGraph,
  474. TAPIMEDIATYPE_AUDIO,
  475. TD_CAPTURE,
  476. &pIMSPStream
  477. );
  478. LOG((MSP_TRACE, "create audio send:%x, hr:%x", pIMSPStream,hr));
  479. if (FAILED(hr))
  480. {
  481. LOG((MSP_ERROR, "create stream failed. %x", hr));
  482. return hr;
  483. }
  484. if (FAILED(hr = InitFullDuplexControler()))
  485. {
  486. LOG((MSP_ERROR, "Create full duplex controller failed. %x", hr));
  487. }
  488. else
  489. {
  490. ((CStreamAudioSend *)pIMSPStream)->
  491. SetFullDuplexController(m_pIAudioDuplexController);
  492. }
  493. }
  494. else
  495. {
  496. return TAPI_E_INVALIDDIRECTION;
  497. }
  498. }
  499. else if (dwMediaType == TAPIMEDIATYPE_VIDEO)
  500. {
  501. if (Direction == TD_RENDER)
  502. {
  503. CStreamVideoRecv *pVideoRecv = NULL;
  504. hr = ::CreateStreamHelper(
  505. pVideoRecv,
  506. m_pMSPAddress,
  507. this,
  508. pGraph,
  509. TAPIMEDIATYPE_VIDEO,
  510. TD_RENDER,
  511. &pIMSPStream
  512. );
  513. LOG((MSP_TRACE, "create video Recv:%x, hr:%x", pIMSPStream,hr));
  514. if (FAILED(hr))
  515. {
  516. LOG((MSP_ERROR, "create stream failed. %x", hr));
  517. return hr;
  518. }
  519. }
  520. else if (Direction == TD_CAPTURE)
  521. {
  522. CStreamVideoSend *pVideoSend = NULL;
  523. hr = ::CreateStreamHelper(
  524. pVideoSend,
  525. m_pMSPAddress,
  526. this,
  527. pGraph,
  528. TAPIMEDIATYPE_VIDEO,
  529. TD_CAPTURE,
  530. &pIMSPStream
  531. );
  532. LOG((MSP_TRACE, "create video send:%x, hr:%x", pIMSPStream,hr));
  533. if (FAILED(hr))
  534. {
  535. LOG((MSP_ERROR, "create stream failed. %x", hr));
  536. return hr;
  537. }
  538. }
  539. else
  540. {
  541. return TAPI_E_INVALIDDIRECTION;
  542. }
  543. }
  544. else
  545. {
  546. return TAPI_E_INVALIDMEDIATYPE;
  547. }
  548. *ppStream = pIMSPStream;
  549. return S_OK;
  550. }
  551. DWORD CIPConfMSPCall::FindInterfaceByName(IN WCHAR *pMachineName)
  552. /*++
  553. Routine Description:
  554. Given the machine name of the originator, find out which local interface
  555. can be used to reach that machine.
  556. Arguments:
  557. pMachineName - The machine name of the originator.
  558. Return Value:
  559. INADDR_NONE - nothing can be found.
  560. valid IP - succeeded.
  561. --*/
  562. {
  563. char buffer[MAXIPADDRLEN + 1];
  564. if (WideCharToMultiByte(
  565. GetACP(),
  566. 0,
  567. pMachineName,
  568. -1,
  569. buffer,
  570. MAXIPADDRLEN,
  571. NULL,
  572. NULL
  573. ) == 0)
  574. {
  575. LOG((MSP_ERROR, "can't convert originator's address:%ws", pMachineName));
  576. return INADDR_NONE;
  577. }
  578. DWORD dwAddr;
  579. if ((dwAddr = inet_addr(buffer)) != INADDR_NONE)
  580. {
  581. dwAddr = ntohl(dwAddr);
  582. LOG((MSP_INFO, "originator's IP:%x", dwAddr));
  583. return ((CIPConfMSP *)m_pMSPAddress)->FindLocalInterface(dwAddr);
  584. }
  585. struct hostent * pHost;
  586. // attempt to lookup hostname
  587. pHost = gethostbyname(buffer);
  588. // validate pointer
  589. if (pHost == NULL)
  590. {
  591. LOG((MSP_WARN, "can't resolve address:%s", buffer));
  592. return INADDR_NONE;
  593. }
  594. // for each of the addresses returned, find the local interface.
  595. for (DWORD i = 0; TRUE; i ++)
  596. {
  597. if (pHost->h_addr_list[i] == NULL)
  598. {
  599. break;
  600. }
  601. // retrieve host address from structure
  602. dwAddr = ntohl(*(unsigned long *)pHost->h_addr_list[i]);
  603. LOG((MSP_INFO, "originator's IP:%x", dwAddr));
  604. DWORD dwInterface =
  605. ((CIPConfMSP *)m_pMSPAddress)->FindLocalInterface(dwAddr);
  606. if (dwInterface != INADDR_NONE)
  607. {
  608. return dwInterface;
  609. }
  610. }
  611. return INADDR_NONE;
  612. }
  613. HRESULT CIPConfMSPCall::CheckOrigin(
  614. IN ITSdp * pITSdp,
  615. OUT BOOL * pFlag,
  616. OUT DWORD * pdwIP
  617. )
  618. /*++
  619. Routine Description:
  620. Check to see if the current user is the originator of the conference.
  621. If he is, he can send to a receive only conference.
  622. Arguments:
  623. pITSdp - a pointer to the ITSdp interface.
  624. pFlag - The result.
  625. pdwIP - The local IP interface that should be used to reach the originator.
  626. Return Value:
  627. HRESULT.
  628. --*/
  629. {
  630. const DWORD MAXUSERNAMELEN = 127;
  631. DWORD dwUserNameLen = MAXUSERNAMELEN;
  632. WCHAR szUserName[MAXUSERNAMELEN+1];
  633. // determine the name of the current user
  634. if (!GetUserNameW(szUserName, &dwUserNameLen))
  635. {
  636. LOG((MSP_ERROR, "cant' get user name. %x", GetLastError()));
  637. return E_UNEXPECTED;
  638. }
  639. LOG((MSP_INFO, "current user: %ws", szUserName));
  640. // find out if the current user is the originator of the conference.
  641. BSTR Originator = NULL;
  642. HRESULT hr = pITSdp->get_Originator(&Originator);
  643. if (FAILED(hr))
  644. {
  645. LOG((MSP_ERROR, "cant' get originator. %x", hr));
  646. return hr;
  647. }
  648. LOG((MSP_INFO, "originator: %ws", Originator));
  649. *pFlag = (_wcsnicmp(szUserName, Originator, lstrlenW(szUserName)) == 0);
  650. SysFreeString(Originator);
  651. // Get the machine IP address of the originator.
  652. BSTR MachineAddress = NULL;
  653. hr = pITSdp->get_MachineAddress(&MachineAddress);
  654. if (FAILED(hr))
  655. {
  656. LOG((MSP_ERROR, "cant' get MachineAddress. %x", hr));
  657. return hr;
  658. }
  659. LOG((MSP_INFO, "MachineAddress: %ws", MachineAddress));
  660. DWORD dwIP = FindInterfaceByName(MachineAddress);
  661. SysFreeString(MachineAddress);
  662. *pdwIP = dwIP;
  663. LOG((MSP_INFO, "Interface to use:%x", *pdwIP));
  664. return S_OK;
  665. }
  666. HRESULT GetAddress(
  667. IN IUnknown * pIUnknown,
  668. OUT DWORD * pdwAddress,
  669. OUT DWORD * pdwTTL,
  670. OUT BSTR * ppKey,
  671. OUT LONG * plBandwidth,
  672. OUT LONG * plConfBandwidth = NULL
  673. )
  674. /*++
  675. Routine Description:
  676. Get the IP address and TTL value from a connection. It is a "c=" line
  677. in the SDP blob.
  678. Arguments:
  679. pIUnknow - an object that might contain connection information.
  680. pdwAddress - the mem address to store the IP address.
  681. pdwTTL - the mem address to store the TTL value.
  682. plBandwidth - maximum bandwidth
  683. Return Value:
  684. HRESULT.
  685. --*/
  686. {
  687. // query for the ITConnection i/f
  688. CComPtr<ITConnection> pITConnection;
  689. HRESULT hr = pIUnknown->QueryInterface(__uuidof(ITConnection), (void **)&pITConnection);
  690. if (FAILED(hr))
  691. {
  692. LOG((MSP_ERROR, "get connection interface. %x", hr));
  693. return hr;
  694. }
  695. // clear
  696. if (plConfBandwidth != NULL) *plConfBandwidth = QCDEFAULT_QUALITY_UNSET;
  697. // get bandwidth
  698. const WCHAR * const AS = L"AS";
  699. const WCHAR * const CT = L"CT";
  700. BSTR pModifier = NULL;
  701. DOUBLE bandwidth;
  702. if (FAILED (hr = pITConnection->get_BandwidthModifier (&pModifier)))
  703. {
  704. *plBandwidth = QCDEFAULT_QUALITY_UNSET;
  705. // bandwidth modifier may not be presented
  706. // LOG ((MSP_TRACE, "get bandwidth modifiler. %x", hr));
  707. }
  708. else if (_wcsnicmp (AS, pModifier, lstrlenW (AS)) != 0)
  709. {
  710. // if not application specific
  711. *plBandwidth = QCDEFAULT_QUALITY_UNSET;
  712. // check conference-wide bandwidth limit
  713. if (_wcsnicmp (CT, pModifier, lstrlenW (CT)) == 0)
  714. {
  715. if (plConfBandwidth)
  716. {
  717. if (FAILED (hr = pITConnection->get_Bandwidth (&bandwidth)))
  718. {
  719. *plConfBandwidth = QCDEFAULT_QUALITY_UNSET;
  720. LOG ((MSP_ERROR, "get conf bandwidth. %x", hr));
  721. }
  722. else
  723. *plConfBandwidth = (LONG)(bandwidth * 1000);
  724. }
  725. }
  726. }
  727. else if (FAILED (hr = pITConnection->get_Bandwidth (&bandwidth)))
  728. {
  729. *plBandwidth = QCDEFAULT_QUALITY_UNSET;
  730. LOG ((MSP_ERROR, "get bandwidth. %x", hr));
  731. }
  732. else
  733. *plBandwidth = (LONG)(bandwidth * 1000);
  734. if (pModifier)
  735. {
  736. SysFreeString (pModifier);
  737. pModifier = NULL;
  738. }
  739. // get the start address,
  740. BSTR StartAddress = NULL;
  741. hr = pITConnection->get_StartAddress(&StartAddress);
  742. if (FAILED(hr))
  743. {
  744. LOG((MSP_WARN, "get start address. %x", hr));
  745. return hr;
  746. }
  747. // Get the IP address from the string.
  748. const DWORD MAXIPADDRLEN = 20;
  749. char Buffer[MAXIPADDRLEN+1];
  750. // first convert the string to ascii.
  751. Buffer[0] = '\0';
  752. if (!WideCharToMultiByte(
  753. CP_ACP,
  754. 0,
  755. StartAddress,
  756. -1,
  757. Buffer,
  758. MAXIPADDRLEN,
  759. NULL,
  760. NULL
  761. ))
  762. {
  763. LOG((MSP_ERROR, "converting address. %ws", StartAddress));
  764. SysFreeString(StartAddress);
  765. return E_UNEXPECTED;
  766. }
  767. SysFreeString(StartAddress);
  768. // convert the string to DWORD IP address.
  769. DWORD dwIP = ntohl(inet_addr(Buffer));
  770. if (dwIP == INADDR_NONE)
  771. {
  772. LOG((MSP_ERROR, "invalid IP address. %s", Buffer));
  773. return E_UNEXPECTED;
  774. }
  775. // get the TTL value.
  776. BYTE Ttl;
  777. hr = pITConnection->get_Ttl(&Ttl);
  778. if (FAILED(hr))
  779. {
  780. LOG((MSP_ERROR, "can't get TTL."));
  781. return hr;
  782. }
  783. // get the Encryption key.
  784. const WCHAR * const CLEAR = L"clear";
  785. VARIANT_BOOL fKeyValid;
  786. BSTR bstrKeyType = NULL;
  787. if (*ppKey)
  788. SysFreeString (*ppKey);
  789. *ppKey = NULL;
  790. if (FAILED (hr = pITConnection->GetEncryptionKey (&bstrKeyType, &fKeyValid, ppKey)))
  791. {
  792. LOG((MSP_WARN, "can't get EncryptionKey. %x", hr));
  793. }
  794. else if (_wcsnicmp (CLEAR, bstrKeyType, lstrlenW (CLEAR)) != 0)
  795. {
  796. if (*ppKey)
  797. {
  798. SysFreeString (*ppKey);
  799. *ppKey = NULL;
  800. }
  801. }
  802. if (bstrKeyType)
  803. SysFreeString (bstrKeyType);
  804. *pdwAddress = dwIP;
  805. *pdwTTL = Ttl;
  806. return S_OK;
  807. }
  808. HRESULT CheckAttributes(
  809. IN IUnknown * pIUnknown,
  810. OUT BOOL * pbSendOnly,
  811. OUT BOOL * pbRecvOnly,
  812. OUT DWORD * pdwMSPerPacket,
  813. OUT BOOL * pbCIF
  814. )
  815. /*++
  816. Routine Description:
  817. Check the direction of the media, find out if it is send only or
  818. receive only.
  819. Arguments:
  820. pIUnknow - an object that might have a attribute list.
  821. pbSendOnly - the mem address to store the returned BOOL.
  822. pbRecvOnly - the mem address to store the returned BOOL.
  823. pbCIF - if CIF is used for video.
  824. Return Value:
  825. HRESULT.
  826. --*/
  827. {
  828. // query for the ITAttributeList i/f
  829. CComPtr<ITAttributeList> pIAttList;
  830. HRESULT hr = pIUnknown->QueryInterface(__uuidof(ITAttributeList), (void **)&pIAttList);
  831. if (FAILED(hr))
  832. {
  833. LOG((MSP_ERROR, "get attribute interface. %x", hr));
  834. return hr;
  835. }
  836. // get the number of attributes
  837. long lCount;
  838. hr = pIAttList->get_Count(&lCount);
  839. if (FAILED(hr))
  840. {
  841. LOG((MSP_ERROR, "get attribute count. %x", hr));
  842. return hr;
  843. }
  844. *pbRecvOnly = FALSE;
  845. *pbSendOnly = FALSE;
  846. *pdwMSPerPacket = 0;
  847. *pbCIF = FALSE;
  848. const WCHAR * const SENDONLY = L"sendonly";
  849. const WCHAR * const RECVONLY = L"recvonly";
  850. const WCHAR * const FORMAT = L"fmtp";
  851. const WCHAR * const PTIME = L"ptime:";
  852. const WCHAR * const CIF = L" CIF=";
  853. for (long i = 1; i <= lCount; i ++)
  854. {
  855. // get the attributes and check if sendonly of recvonly is specified.
  856. BSTR Attribute = NULL;
  857. hr = pIAttList->get_Item(i, &Attribute);
  858. if (FAILED(hr))
  859. {
  860. LOG((MSP_ERROR, "get attribute item. %x", hr));
  861. return hr;
  862. }
  863. if (_wcsnicmp(SENDONLY, Attribute, lstrlen(SENDONLY)) == 0)
  864. {
  865. *pbSendOnly = TRUE;
  866. }
  867. else if (_wcsnicmp(RECVONLY, Attribute, lstrlen(RECVONLY)) == 0)
  868. {
  869. *pbRecvOnly = TRUE;
  870. }
  871. else if (_wcsnicmp(PTIME, Attribute, lstrlen(PTIME)) == 0)
  872. {
  873. // read the number of milliseconds per packet.
  874. *pdwMSPerPacket = (DWORD)_wtol(Attribute + lstrlen(PTIME));
  875. // RFC 1890 only requires an app to support 200ms packets.
  876. if (*pdwMSPerPacket > 200)
  877. {
  878. // invalid tag, we just use our default.
  879. *pdwMSPerPacket = 0;
  880. }
  881. }
  882. else if (_wcsnicmp(FORMAT, Attribute, lstrlen(FORMAT)) == 0)
  883. {
  884. if (wcsstr(Attribute, CIF))
  885. {
  886. *pbCIF = TRUE;
  887. }
  888. }
  889. SysFreeString(Attribute);
  890. }
  891. return S_OK;
  892. }
  893. HRESULT CIPConfMSPCall::ProcessMediaItem(
  894. IN ITMedia * pITMedia,
  895. IN DWORD dwMediaTypeMask,
  896. OUT DWORD * pdwMediaType,
  897. OUT WORD * pwPort,
  898. OUT DWORD * pdwPayloadTypes,
  899. IN OUT DWORD * pdwNumPayLoadType
  900. )
  901. /*++
  902. Routine Description:
  903. Process a "m=" line, find out the media type, port, and payload type.
  904. Arguments:
  905. dwMediaTypeMask - the media type of this call.
  906. pdwMediaType - return the media type of this media item.
  907. pwPort - return the port number used for this media.
  908. pdwPayloadType - an array to store the RTP payload types.
  909. pdwNumPayLoadType - The size of the above array. When return, it is the
  910. number of payload types read.
  911. Return Value:
  912. HRESULT.
  913. S_FALSE - everything is all right but the media type is not needed.
  914. --*/
  915. {
  916. // get the name of the media.
  917. BSTR MediaName = NULL;
  918. HRESULT hr = pITMedia->get_MediaName(&MediaName);
  919. if (FAILED(hr))
  920. {
  921. LOG((MSP_ERROR, "get media name. %x", hr));
  922. return hr;
  923. }
  924. LOG((MSP_INFO, "media name: %ws", MediaName));
  925. // check if the media is audio or video.
  926. const WCHAR * const AUDIO = L"audio";
  927. const WCHAR * const VIDEO = L"video";
  928. const DWORD NAMELEN = 5;
  929. DWORD dwMediaType = 0;
  930. if (_wcsnicmp(AUDIO, MediaName, NAMELEN) == 0)
  931. {
  932. dwMediaType = TAPIMEDIATYPE_AUDIO;
  933. }
  934. else if (_wcsnicmp(VIDEO, MediaName, NAMELEN) == 0)
  935. {
  936. dwMediaType = TAPIMEDIATYPE_VIDEO;
  937. }
  938. SysFreeString(MediaName);
  939. // check if the call wants this media type.
  940. if ((dwMediaType & dwMediaTypeMask) == 0)
  941. {
  942. // We don't need this media type in this call.
  943. LOG((MSP_INFO, "media skipped."));
  944. return S_FALSE;
  945. }
  946. // get start port
  947. long lStartPort;
  948. hr = pITMedia->get_StartPort(&lStartPort);
  949. if (FAILED(hr))
  950. {
  951. LOG((MSP_ERROR, "get start port. %x", hr));
  952. return hr;
  953. }
  954. // get the transport Protocol
  955. BSTR TransportProtocol = NULL;
  956. hr = pITMedia->get_TransportProtocol(&TransportProtocol);
  957. if (FAILED(hr))
  958. {
  959. LOG((MSP_ERROR, "get transport Protocol. %x", hr));
  960. return hr;
  961. }
  962. // varify that the protocol is RTP.
  963. const WCHAR * const RTP = L"RTP";
  964. const DWORD PROTOCOLLEN = 3;
  965. if (_wcsnicmp(RTP, TransportProtocol, PROTOCOLLEN) != 0)
  966. {
  967. LOG((MSP_ERROR, "wrong transport Protocol:%ws", TransportProtocol));
  968. SysFreeString(TransportProtocol);
  969. return S_FALSE;
  970. }
  971. SysFreeString(TransportProtocol);
  972. // get the format code list
  973. VARIANT Variant;
  974. VariantInit(&Variant);
  975. hr = pITMedia->get_FormatCodes(&Variant);
  976. if (FAILED(hr))
  977. {
  978. LOG((MSP_ERROR, "get format codes. %x", hr));
  979. return hr;
  980. }
  981. // Verify that the SafeArray is in proper shape.
  982. if(SafeArrayGetDim(V_ARRAY(&Variant)) != 1)
  983. {
  984. LOG((MSP_ERROR, "wrong dimension for the format code. %x", hr));
  985. VariantClear(&Variant);
  986. return E_UNEXPECTED;
  987. }
  988. long lLowerBound;
  989. long lUpperBound;
  990. if (FAILED(hr = SafeArrayGetLBound(V_ARRAY(&Variant), 1, &lLowerBound))
  991. || FAILED(hr = SafeArrayGetUBound(V_ARRAY(&Variant), 1, &lUpperBound)))
  992. {
  993. LOG((MSP_ERROR, "Can't get the array bounds. %x", hr));
  994. VariantClear(&Variant);
  995. return E_UNEXPECTED;
  996. }
  997. DWORD dwNumFormats = 0;
  998. for (long l = lLowerBound; l <= lUpperBound && dwNumFormats < *pdwNumPayLoadType; l ++)
  999. {
  1000. BSTR Format = NULL;
  1001. hr = SafeArrayGetElement(V_ARRAY(&Variant), &l, &Format);
  1002. if (FAILED(hr))
  1003. {
  1004. LOG((MSP_ERROR, "get format code. %x", hr));
  1005. continue;
  1006. }
  1007. LOG((MSP_INFO, "format code: %ws", Format));
  1008. pdwPayloadTypes[dwNumFormats ++] = (DWORD)_wtoi(Format);
  1009. SysFreeString(Format);
  1010. }
  1011. // clear the variant because we don't need it any more
  1012. VariantClear(&Variant);
  1013. *pdwMediaType = dwMediaType;
  1014. *pwPort = (WORD)lStartPort;
  1015. *pdwNumPayLoadType = dwNumFormats;
  1016. return S_OK;
  1017. }
  1018. HRESULT CIPConfMSPCall::ConfigStreamsBasedOnSDP(
  1019. IN ITSdp * pITSdp,
  1020. IN DWORD dwAudioQOSLevel,
  1021. IN DWORD dwVideoQOSLevel
  1022. )
  1023. /*++
  1024. Routine Description:
  1025. Configure the streams based on the information in the SDP blob.
  1026. Arguments:
  1027. pITSdp - the SDP object. It contains parsed information.
  1028. Return Value:
  1029. HRESULT.
  1030. --*/
  1031. {
  1032. // find out if the current user is the originator of the conference.
  1033. BOOL fIsOriginator;
  1034. DWORD dwLocalInterface = INADDR_NONE;
  1035. HRESULT hr = CheckOrigin(pITSdp, &fIsOriginator, &dwLocalInterface);
  1036. if (FAILED(hr))
  1037. {
  1038. LOG((MSP_ERROR, "check origin. %x", hr));
  1039. return hr;
  1040. }
  1041. LOG((MSP_INFO, "Local interface: %x", dwLocalInterface));
  1042. // get the start IP address and TTL value from the connection.
  1043. DWORD dwIPGlobal, dwTTLGlobal;
  1044. BSTR bstrKeyGlobal = NULL;
  1045. LONG lbandwidth, lConfBandwidth;
  1046. hr = GetAddress(pITSdp, &dwIPGlobal, &dwTTLGlobal, &bstrKeyGlobal, &lbandwidth, &lConfBandwidth);
  1047. if (FAILED(hr))
  1048. {
  1049. LOG((MSP_ERROR, "get global address. %x", hr));
  1050. return hr;
  1051. }
  1052. CLock lock(m_lock);
  1053. // store conference bandwidth
  1054. if (FAILED (m_pCallQCRelay->SetConfBitrate (lConfBandwidth)))
  1055. {
  1056. LOG ((MSP_ERROR, "bandwidth is out of range %d", lConfBandwidth));
  1057. }
  1058. // find out if this conference is sendonly or recvonly.
  1059. BOOL fSendOnlyGlobal = FALSE, fRecvOnlyGlobal = FALSE, fCIF = FALSE;
  1060. DWORD dwMSPerPacket;
  1061. hr = CheckAttributes(
  1062. pITSdp, &fSendOnlyGlobal, &fRecvOnlyGlobal, &dwMSPerPacket, &fCIF);
  1063. if (FAILED(hr))
  1064. {
  1065. LOG((MSP_ERROR, "check global attributes. %x", hr));
  1066. return hr;
  1067. }
  1068. // get the media information
  1069. CComPtr<ITMediaCollection> pICollection;
  1070. hr = pITSdp->get_MediaCollection(&pICollection);
  1071. if (FAILED(hr))
  1072. {
  1073. LOG((MSP_ERROR, "get the media collection. %x", hr));
  1074. return hr;
  1075. }
  1076. // find out how many media sessions are in the blobl.
  1077. long lCount;
  1078. hr = pICollection->get_Count(&lCount);
  1079. if (FAILED(hr))
  1080. {
  1081. LOG((MSP_ERROR, "get number of media items. %x", hr));
  1082. return hr;
  1083. }
  1084. if (lCount > 0)
  1085. {
  1086. // change the call into connected state since the SDP is OK.
  1087. // We are going to set up each every streams next.
  1088. SendTSPMessage(CALL_CONNECTED, 0);
  1089. }
  1090. DWORD dwNumSucceeded = 0;
  1091. // for each media session, get info configure a stream.
  1092. for(long i=1; i <= lCount; i++)
  1093. {
  1094. // get the media item first.
  1095. ITMedia *pITMedia;
  1096. hr = pICollection->get_Item(i, &pITMedia);
  1097. if (FAILED(hr))
  1098. {
  1099. LOG((MSP_ERROR, "get media item. %x", hr));
  1100. continue;
  1101. }
  1102. DWORD dwMediaType;
  1103. STREAMSETTINGS Setting;
  1104. ZeroMemory(&Setting, sizeof(STREAMSETTINGS));
  1105. // find out the information about the media. Here we pass in the media
  1106. // type of call so that we won't wasting time reading the attributes
  1107. // for a media type we don't need.
  1108. DWORD dwNumPayloadTypes = sizeof(Setting.PayloadTypes)
  1109. / sizeof(Setting.PayloadTypes[0]);
  1110. hr = ProcessMediaItem(
  1111. pITMedia,
  1112. m_dwMediaType,
  1113. &dwMediaType,
  1114. &Setting.wRTPPortRemote,
  1115. Setting.PayloadTypes,
  1116. &dwNumPayloadTypes
  1117. );
  1118. if (FAILED(hr))
  1119. {
  1120. LOG((MSP_ERROR, "process media. %x", hr));
  1121. continue;
  1122. }
  1123. Setting.dwNumPayloadTypes = dwNumPayloadTypes;
  1124. // if the return value is S_FALSE from the previous call, this media
  1125. // type is not needed for the call.
  1126. if (hr != S_OK)
  1127. {
  1128. // the media is not needed.
  1129. continue;
  1130. }
  1131. if (dwMediaType == TAPIMEDIATYPE_AUDIO)
  1132. {
  1133. Setting.dwQOSLevel = dwAudioQOSLevel;
  1134. Setting.phRTPSession = &m_hAudioRTPSession;
  1135. }
  1136. else
  1137. {
  1138. Setting.dwQOSLevel = dwVideoQOSLevel;
  1139. Setting.phRTPSession = &m_hVideoRTPSession;
  1140. }
  1141. // Get the local connect information.
  1142. DWORD dwIP, dwTTL;
  1143. BSTR bstrKey = NULL;
  1144. hr = GetAddress(pITMedia, &dwIP, &dwTTL, &bstrKey, &lbandwidth);
  1145. if (FAILED(hr))
  1146. {
  1147. LOG((MSP_WARN, "no local address, use global one", hr));
  1148. Setting.dwIPRemote = dwIPGlobal;
  1149. Setting.dwTTL = dwTTLGlobal;
  1150. Setting.lBandwidth = QCDEFAULT_QUALITY_UNSET;
  1151. }
  1152. else
  1153. {
  1154. Setting.dwIPRemote = dwIP;
  1155. Setting.dwTTL = dwTTL;
  1156. Setting.lBandwidth = lbandwidth;
  1157. }
  1158. // find out if this media is sendonly or recvonly.
  1159. BOOL fSendOnly = FALSE, fRecvOnly = FALSE, fCIF = FALSE;
  1160. hr = CheckAttributes(
  1161. pITMedia, &fSendOnly, &fRecvOnly, &dwMSPerPacket, &fCIF);
  1162. if (FAILED(hr))
  1163. {
  1164. LOG((MSP_ERROR, "check local attributes. %x", hr));
  1165. }
  1166. fSendOnly = fSendOnly || fSendOnlyGlobal;
  1167. fRecvOnly = (fRecvOnly || fRecvOnlyGlobal) && (!fIsOriginator);
  1168. Setting.dwMSPerPacket = dwMSPerPacket;
  1169. Setting.fCIF = fCIF;
  1170. // The media item is not needed after this point.
  1171. pITMedia->Release();
  1172. // Go through the existing streams and find out if any stream
  1173. // can be configured.
  1174. // Note: we are not creating any new streams now. We might want to
  1175. // do it in the future if we want to support two sessions of the
  1176. // same media type.
  1177. m_fCallStarted = TRUE;
  1178. for (long j = 0; j < m_Streams.GetSize(); j ++)
  1179. {
  1180. CIPConfMSPStream* pStream = (CIPConfMSPStream*)m_Streams[j];
  1181. if ((pStream->MediaType() != dwMediaType)
  1182. || pStream->IsConfigured()
  1183. || (fSendOnly && pStream->Direction() == TD_RENDER)
  1184. || (fRecvOnly && pStream->Direction() == TD_CAPTURE)
  1185. )
  1186. {
  1187. // this stream should not be configured.
  1188. continue;
  1189. }
  1190. // set the local interface that the call should bind to.
  1191. Setting.dwIPLocal = m_dwIPInterface;
  1192. if ((m_dwIPInterface == INADDR_ANY)
  1193. && (dwLocalInterface != INADDR_NONE))
  1194. {
  1195. Setting.dwIPLocal = dwLocalInterface;
  1196. }
  1197. // set the loopback mode of the stream.
  1198. Setting.LoopbackMode = m_LoopbackMode;
  1199. // set the qos application IDS.
  1200. Setting.pApplicationID = m_pApplicationID;
  1201. Setting.pApplicationGUID = m_pApplicationGUID;
  1202. Setting.pSubIDs = m_pSubIDs;
  1203. // configure the stream, it will not be started.
  1204. hr = pStream->Configure(Setting, (bstrKey) ? bstrKey : bstrKeyGlobal);
  1205. if (FAILED(hr))
  1206. {
  1207. LOG((MSP_ERROR, "configure stream failed. %x", hr));
  1208. }
  1209. }
  1210. SysFreeString(bstrKey);
  1211. }
  1212. SysFreeString(bstrKeyGlobal);
  1213. // after configuring the streams, start them.
  1214. for (int j = 0; j < m_Streams.GetSize(); j ++)
  1215. {
  1216. CIPConfMSPStream* pStream = (CIPConfMSPStream*)m_Streams[j];
  1217. // start the stream.
  1218. hr = pStream->FinishConfigure();
  1219. if (SUCCEEDED(hr))
  1220. {
  1221. dwNumSucceeded ++;
  1222. }
  1223. }
  1224. if (dwNumSucceeded == 0)
  1225. {
  1226. LOG((MSP_ERROR, "No media succeeded."));
  1227. return E_FAIL;
  1228. }
  1229. return S_OK;
  1230. }
  1231. HRESULT CIPConfMSPCall::ParseSDP(
  1232. IN WCHAR * pSDP,
  1233. IN DWORD dwAudioQOSLevel,
  1234. IN DWORD dwVideoQOSLevel
  1235. )
  1236. /*++
  1237. Routine Description:
  1238. Parse the SDP string. The function uses the SdpConferenceBlob object
  1239. to parse the string.
  1240. Arguments:
  1241. pSDP - the SDP string.
  1242. dwAudioQOSLevel - the QOS requirement for audio.
  1243. dwVideoQOSLevel - the QOS requirement for video.
  1244. Return Value:
  1245. HRESULT.
  1246. --*/
  1247. {
  1248. // co-create an sdp conference blob component
  1249. // query for the ITConferenceBlob interface
  1250. CComPtr<ITConferenceBlob> pIConfBlob;
  1251. HRESULT hr = ::CoCreateInstance(
  1252. CLSID_SdpConferenceBlob,
  1253. NULL,
  1254. CLSCTX_INPROC_SERVER | CLSCTX_NO_CODE_DOWNLOAD,
  1255. __uuidof(ITConferenceBlob),
  1256. (void **)&pIConfBlob
  1257. );
  1258. if (FAILED(hr))
  1259. {
  1260. LOG((MSP_ERROR, "creating a SDPBlob object. %x", hr));
  1261. return hr;
  1262. }
  1263. // conver the sdp into a BSTR to use the interface.
  1264. BSTR bstrSDP = SysAllocString(pSDP);
  1265. if (bstrSDP == NULL)
  1266. {
  1267. LOG((MSP_ERROR, "out of mem converting SDP to a BSTR."));
  1268. return E_OUTOFMEMORY;
  1269. }
  1270. // Parse the SDP string.
  1271. hr = pIConfBlob->Init(NULL, BCS_ASCII, bstrSDP);
  1272. // the string is not needed any more.
  1273. SysFreeString(bstrSDP);
  1274. if (FAILED(hr))
  1275. {
  1276. LOG((MSP_ERROR, "parse the SDPBlob object. %x", hr));
  1277. return hr;
  1278. }
  1279. // Get the ITSdp interface.
  1280. CComPtr<ITSdp> pITSdp;
  1281. hr = pIConfBlob->QueryInterface(__uuidof(ITSdp), (void **)&pITSdp);
  1282. if (FAILED(hr))
  1283. {
  1284. LOG((MSP_ERROR, "can't get the ITSdp interface. %x", hr));
  1285. return hr;
  1286. }
  1287. // check main sdp validity
  1288. VARIANT_BOOL IsValid;
  1289. hr = pITSdp->get_IsValid(&IsValid);
  1290. if (FAILED(hr))
  1291. {
  1292. LOG((MSP_ERROR, "can't get the valid flag on the SDP %x", hr));
  1293. return hr;
  1294. }
  1295. if (!IsValid)
  1296. {
  1297. LOG((MSP_ERROR, "the SDP is not valid %x", hr));
  1298. return E_FAIL;
  1299. }
  1300. return ConfigStreamsBasedOnSDP(
  1301. pITSdp,
  1302. dwAudioQOSLevel,
  1303. dwVideoQOSLevel
  1304. );
  1305. }
  1306. HRESULT CIPConfMSPCall::SendTSPMessage(
  1307. IN TSP_MSP_COMMAND command,
  1308. IN DWORD dwParam1,
  1309. IN DWORD dwParam2
  1310. ) const
  1311. /*++
  1312. Routine Description:
  1313. Send the TSP a message from the MSP.
  1314. Arguments:
  1315. command - the command to be sent.
  1316. dwParam1 - the first DWORD used in the command.
  1317. dwParam2 - the second DWORD used in the command.
  1318. Return Value:
  1319. HRESULT.
  1320. --*/
  1321. {
  1322. LOG((MSP_TRACE, "SendTSPMessage, command %d, dwParam1 %d, dwParam2",
  1323. command, dwParam1, dwParam2));
  1324. // first allocate the memory.
  1325. MSPEVENTITEM* pEventItem = AllocateEventItem(sizeof(MSG_TSPMSPDATA));
  1326. if (pEventItem == NULL)
  1327. {
  1328. LOG((MSP_ERROR, "No memory for the TSPMSP data"));
  1329. return E_OUTOFMEMORY;
  1330. }
  1331. // Fill in the necessary fields for the event structure.
  1332. pEventItem->MSPEventInfo.dwSize =
  1333. sizeof(MSP_EVENT_INFO) + sizeof(MSG_TSPMSPDATA);
  1334. pEventItem->MSPEventInfo.Event = ME_TSP_DATA;
  1335. pEventItem->MSPEventInfo.hCall = m_htCall;
  1336. // Fill in the data for the TSP.
  1337. pEventItem->MSPEventInfo.MSP_TSP_DATA.dwBufferSize = sizeof(MSG_TSPMSPDATA);
  1338. MSG_TSPMSPDATA *pData = (MSG_TSPMSPDATA *)
  1339. pEventItem->MSPEventInfo.MSP_TSP_DATA.pBuffer;
  1340. pData->command = command;
  1341. switch (command)
  1342. {
  1343. case CALL_DISCONNECTED:
  1344. pData->CallDisconnected.dwReason = dwParam1;
  1345. break;
  1346. case CALL_QOS_EVENT:
  1347. pData->QosEvent.dwEvent = dwParam1;
  1348. pData->QosEvent.dwMediaMode = dwParam2;
  1349. break;
  1350. case CALL_CONNECTED:
  1351. break;
  1352. default:
  1353. LOG((MSP_ERROR, "Wrong command type for TSP"));
  1354. FreeEventItem(pEventItem);
  1355. return E_UNEXPECTED;
  1356. }
  1357. HRESULT hr = m_pMSPAddress->PostEvent(pEventItem);
  1358. if (FAILED(hr))
  1359. {
  1360. LOG((MSP_ERROR, "Post event failed %x", hr));
  1361. FreeEventItem(pEventItem);
  1362. return hr;
  1363. }
  1364. return S_OK;
  1365. }
  1366. HRESULT CIPConfMSPCall::CheckUnusedStreams()
  1367. /*++
  1368. Routine Description:
  1369. Find out which streams are not used and send tapi events about them.
  1370. Arguments:
  1371. Return Value:
  1372. HRESULT.
  1373. --*/
  1374. {
  1375. LOG((MSP_TRACE, "CheckUnusedStreams"));
  1376. CLock lock(m_lock);
  1377. for (long j = 0; j < m_Streams.GetSize(); j ++)
  1378. {
  1379. CIPConfMSPStream* pStream = (CIPConfMSPStream*)m_Streams[j];
  1380. if (pStream->IsConfigured())
  1381. {
  1382. // find the next.
  1383. continue;
  1384. }
  1385. MSPEVENTITEM* pEventItem = AllocateEventItem();
  1386. if (pEventItem == NULL)
  1387. {
  1388. LOG((MSP_ERROR, "No memory for the TSPMSP data"));
  1389. return E_OUTOFMEMORY;
  1390. }
  1391. // Fill in the necessary fields for the event structure.
  1392. pEventItem->MSPEventInfo.dwSize = sizeof(MSP_EVENT_INFO);;
  1393. pEventItem->MSPEventInfo.Event = ME_CALL_EVENT;
  1394. pEventItem->MSPEventInfo.hCall = m_htCall;
  1395. pEventItem->MSPEventInfo.MSP_CALL_EVENT_INFO.Type = CALL_STREAM_NOT_USED;
  1396. pEventItem->MSPEventInfo.MSP_CALL_EVENT_INFO.Cause = CALL_CAUSE_REMOTE_REQUEST;
  1397. pEventItem->MSPEventInfo.MSP_CALL_EVENT_INFO.pStream = m_Streams[j];
  1398. // Addref to prevent it from going away.
  1399. m_Streams[j]->AddRef();
  1400. pEventItem->MSPEventInfo.MSP_CALL_EVENT_INFO.pTerminal = NULL;
  1401. pEventItem->MSPEventInfo.MSP_CALL_EVENT_INFO.hrError= 0;
  1402. // send the event to tapi.
  1403. HRESULT hr = m_pMSPAddress->PostEvent(pEventItem);
  1404. if (FAILED(hr))
  1405. {
  1406. LOG((MSP_ERROR, "Post event failed %x", hr));
  1407. FreeEventItem(pEventItem);
  1408. return hr;
  1409. }
  1410. }
  1411. return S_OK;
  1412. }
  1413. DWORD WINAPI CIPConfMSPCall::WorkerCallbackDispatcher(VOID *pContext)
  1414. /*++
  1415. Routine Description:
  1416. Because Parsing the SDP and configure the streams uses a lot of COM
  1417. stuff, we can't rely on the RPC thread the calls into the MSP to
  1418. receive the TSP data. So, we let our own working thread do the work.
  1419. This method is the callback function for the queued work item. It
  1420. just gets the call object from the context structure and calls a method
  1421. on the call object to handle the work item.
  1422. Arguments:
  1423. pContext - A pointer to a CALLWORKITEM structure.
  1424. Return Value:
  1425. HRESULT.
  1426. --*/
  1427. {
  1428. _ASSERTE(!IsBadReadPtr(pContext, sizeof CALLWORKITEM));
  1429. CALLWORKITEM *pItem = (CALLWORKITEM *)pContext;
  1430. pItem->pCall->ProcessWorkerCallBack(pItem->Buffer, pItem->dwLen);
  1431. pItem->pCall->MSPCallRelease();
  1432. free(pItem);
  1433. return NOERROR;
  1434. }
  1435. DWORD CIPConfMSPCall::ProcessWorkerCallBack(
  1436. IN PBYTE pBuffer,
  1437. IN DWORD dwSize
  1438. )
  1439. /*++
  1440. Routine Description:
  1441. This function handles the work item given by the TSP.
  1442. Arguments:
  1443. pBuffer - a buffer that contains a TSP_MSP command block.
  1444. dwSize - the size of the buffer.
  1445. Return Value:
  1446. NOERROR.
  1447. --*/
  1448. {
  1449. LOG((MSP_TRACE, "PreocessWorkerCallBAck"));
  1450. _ASSERTE(!IsBadReadPtr(pBuffer, dwSize));
  1451. MSG_TSPMSPDATA * pData = (MSG_TSPMSPDATA *)pBuffer;
  1452. HRESULT hr;
  1453. switch (pData->command)
  1454. {
  1455. case CALL_START:
  1456. // Parse the SDP contained in the command block.
  1457. hr = ParseSDP(pData->CallStart.szSDP,
  1458. pData->CallStart.dwAudioQOSLevel,
  1459. pData->CallStart.dwVideoQOSLevel
  1460. );
  1461. if (FAILED(hr))
  1462. {
  1463. // disconnect the call if someting terrible happend.
  1464. SendTSPMessage(CALL_DISCONNECTED, 0);
  1465. LOG((MSP_ERROR, "parsing theSDPBlob object. %x", hr));
  1466. return NOERROR;
  1467. }
  1468. // go through the streams and send events if they are not used.
  1469. hr = CheckUnusedStreams();
  1470. if (FAILED(hr))
  1471. {
  1472. LOG((MSP_ERROR, "start the streams failed. %x", hr));
  1473. }
  1474. break;
  1475. case CALL_STOP:
  1476. InternalShutDown();
  1477. break;
  1478. }
  1479. return NOERROR;
  1480. }
  1481. HRESULT CIPConfMSPCall::ReceiveTSPCallData(
  1482. IN PBYTE pBuffer,
  1483. IN DWORD dwSize
  1484. )
  1485. /*++
  1486. Routine Description:
  1487. This function handles the work item given by the TSP.
  1488. Arguments:
  1489. pBuffer - a buffer that contains a TSP_MSP command block.
  1490. dwSize - the size of the buffer.
  1491. Return Value:
  1492. NOERROR.
  1493. --*/
  1494. {
  1495. LOG((MSP_TRACE,
  1496. "ReceiveTSPCallData, pBuffer %x, dwSize %d", pBuffer, dwSize));
  1497. MSG_TSPMSPDATA * pData = (MSG_TSPMSPDATA *)pBuffer;
  1498. switch (pData->command)
  1499. {
  1500. case CALL_START:
  1501. // make sure the string is valid.
  1502. if ((IsBadReadPtr(pData->CallStart.szSDP,
  1503. (pData->CallStart.dwSDPLen + 1) * sizeof (WCHAR)))
  1504. || (pData->CallStart.szSDP[pData->CallStart.dwSDPLen] != 0))
  1505. {
  1506. LOG((MSP_ERROR, "the TSP data is invalid."));
  1507. return E_UNEXPECTED;
  1508. }
  1509. LOG((MSP_INFO, "SDP string\n%ws", pData->CallStart.szSDP));
  1510. break;
  1511. case CALL_STOP:
  1512. break;
  1513. default:
  1514. LOG((MSP_ERROR,
  1515. "wrong command received from the TSP:%x", pData->command));
  1516. return E_UNEXPECTED;
  1517. }
  1518. // allocate a work item structure for our worker thread.
  1519. CALLWORKITEM *pItem = (CALLWORKITEM *)malloc(sizeof(CALLWORKITEM) + dwSize);
  1520. if (pItem == NULL)
  1521. {
  1522. // Disconnect the call because of out of memory.
  1523. SendTSPMessage(CALL_DISCONNECTED, 0);
  1524. LOG((MSP_ERROR, "out of memory for work item."));
  1525. return E_OUTOFMEMORY;
  1526. }
  1527. this->MSPCallAddRef();
  1528. pItem->pCall = this;
  1529. pItem->dwLen = dwSize;
  1530. CopyMemory(pItem->Buffer, pBuffer, dwSize);
  1531. // post a work item to our worker thread.
  1532. HRESULT hr = g_Thread.QueueWorkItem(
  1533. WorkerCallbackDispatcher, // the callback
  1534. pItem, // the context.
  1535. FALSE // sync (FALSE means asyn)
  1536. );
  1537. if (FAILED(hr))
  1538. {
  1539. if (pData->command == CALL_START)
  1540. {
  1541. // Disconnect the call because we can't handle the work.
  1542. SendTSPMessage(CALL_DISCONNECTED, 0);
  1543. }
  1544. this->MSPCallRelease();
  1545. free(pItem);
  1546. LOG((MSP_ERROR, "queue work item failed."));
  1547. }
  1548. return hr;
  1549. }
  1550. STDMETHODIMP CIPConfMSPCall::EnumerateParticipants(
  1551. OUT IEnumParticipant ** ppEnumParticipant
  1552. )
  1553. /*++
  1554. Routine Description:
  1555. This method returns an enumerator to the participants.
  1556. Arguments:
  1557. ppEnumParticipant - the memory location to store the returned pointer.
  1558. Return Value:
  1559. S_OK
  1560. E_POINTER
  1561. E_OUTOFMEMORY
  1562. --*/
  1563. {
  1564. LOG((MSP_TRACE,
  1565. "EnumerateParticipants entered. ppEnumParticipant:%p", ppEnumParticipant));
  1566. //
  1567. // Check parameters.
  1568. //
  1569. if (IsBadWritePtr(ppEnumParticipant, sizeof(VOID *)))
  1570. {
  1571. LOG((MSP_ERROR, "CIPConfMSPCall::EnumerateParticipants - "
  1572. "bad pointer argument - exit E_POINTER"));
  1573. return E_POINTER;
  1574. }
  1575. //
  1576. // First see if this call has been shut down.
  1577. // acquire the lock before accessing the Participant object list.
  1578. //
  1579. CLock lock(m_ParticipantLock);
  1580. if (m_Participants.GetData() == NULL)
  1581. {
  1582. LOG((MSP_ERROR, "CIPConfMSPCall::EnumerateParticipants - "
  1583. "call appears to have been shut down - exit E_UNEXPECTED"));
  1584. // This call has been shut down.
  1585. return E_UNEXPECTED;
  1586. }
  1587. //
  1588. // Create an enumerator object.
  1589. //
  1590. HRESULT hr = CreateParticipantEnumerator(
  1591. m_Participants.GetData(), // the begin itor
  1592. m_Participants.GetData() + m_Participants.GetSize(), // the end itor,
  1593. ppEnumParticipant
  1594. );
  1595. if (FAILED(hr))
  1596. {
  1597. LOG((MSP_ERROR, "CIPConfMSPCall::EnumerateParticipants - "
  1598. "create enumerator object failed, %x", hr));
  1599. return hr;
  1600. }
  1601. LOG((MSP_TRACE, "CIPConfMSPCall::EnumerateParticipants - exit S_OK"));
  1602. return hr;
  1603. }
  1604. STDMETHODIMP CIPConfMSPCall::get_Participants(
  1605. OUT VARIANT * pVariant
  1606. )
  1607. {
  1608. LOG((MSP_TRACE, "CIPConfMSPCall::get_Participants - enter"));
  1609. //
  1610. // Check parameters.
  1611. //
  1612. if ( IsBadWritePtr(pVariant, sizeof(VARIANT) ) )
  1613. {
  1614. LOG((MSP_ERROR, "CIPConfMSPCall::get_Participants - "
  1615. "bad pointer argument - exit E_POINTER"));
  1616. return E_POINTER;
  1617. }
  1618. //
  1619. // See if this call has been shut down. Acquire the lock before accessing
  1620. // the Participant object list.
  1621. //
  1622. CLock lock(m_ParticipantLock);
  1623. if (m_Participants.GetData() == NULL)
  1624. {
  1625. LOG((MSP_ERROR, "CIPConfMSPCall::get_Participants - "
  1626. "call appears to have been shut down - exit E_UNEXPECTED"));
  1627. // This call has been shut down.
  1628. return E_UNEXPECTED;
  1629. }
  1630. //
  1631. // create the collection object - see mspcoll.h
  1632. //
  1633. HRESULT hr = CreateParticipantCollection(
  1634. m_Participants.GetData(), // the begin itor
  1635. m_Participants.GetData() + m_Participants.GetSize(), // the end itor,
  1636. m_Participants.GetSize(), // the size
  1637. pVariant
  1638. );
  1639. if (FAILED(hr))
  1640. {
  1641. LOG((MSP_ERROR, "CIPConfMSPCall::get_Participants - "
  1642. "create collection failed - exit 0x%08x", hr));
  1643. return hr;
  1644. }
  1645. LOG((MSP_TRACE, "CIPConfMSPCall::get_Participants - exit S_OK"));
  1646. return S_OK;
  1647. }
  1648. // IMulticastControl methods
  1649. STDMETHODIMP CIPConfMSPCall::get_LoopbackMode (
  1650. OUT MULTICAST_LOOPBACK_MODE * pMode
  1651. )
  1652. {
  1653. if (pMode == NULL)
  1654. {
  1655. return E_INVALIDARG;
  1656. }
  1657. *pMode = m_LoopbackMode;
  1658. return S_OK;
  1659. }
  1660. STDMETHODIMP CIPConfMSPCall::put_LoopbackMode (
  1661. IN MULTICAST_LOOPBACK_MODE mode
  1662. )
  1663. {
  1664. if (mode < MM_NO_LOOPBACK || mode > MM_SELECTIVE_LOOPBACK)
  1665. {
  1666. return E_INVALIDARG;
  1667. }
  1668. m_LoopbackMode = mode;
  1669. return S_OK;
  1670. }
  1671. // ITLocalParticipant methods, called by the app.
  1672. STDMETHODIMP CIPConfMSPCall::get_LocalParticipantTypedInfo(
  1673. IN PARTICIPANT_TYPED_INFO InfoType,
  1674. OUT BSTR * ppInfo
  1675. )
  1676. /*++
  1677. Routine Description:
  1678. Get a information item for the local participant. This information is
  1679. sent out to other participants in the conference.
  1680. Arguments:
  1681. InfoType - The type of the information asked.
  1682. ppInfo - the mem address to store a BSTR.
  1683. Return Value:
  1684. S_OK,
  1685. E_INVALIDARG,
  1686. E_POINTER,
  1687. E_OUTOFMEMORY,
  1688. TAPI_E_NOITEMS
  1689. */
  1690. {
  1691. LOG((MSP_TRACE, "CParticipant get info, type:%d", InfoType));
  1692. if (InfoType > PTI_PRIVATE || InfoType < PTI_CANONICALNAME)
  1693. {
  1694. LOG((MSP_ERROR, "CParticipant get info - invalid type:%d", InfoType));
  1695. return E_INVALIDARG;
  1696. }
  1697. if (IsBadWritePtr(ppInfo, sizeof(BSTR)))
  1698. {
  1699. LOG((MSP_ERROR, "CParticipant get info - exit E_POINTER"));
  1700. return E_POINTER;
  1701. }
  1702. // check if we have that info.
  1703. CLock lock(m_lock);
  1704. if (!m_fLocalInfoRetrieved)
  1705. {
  1706. HRESULT hr = InitializeLocalParticipant();
  1707. if (FAILED(hr))
  1708. {
  1709. return hr;
  1710. }
  1711. }
  1712. int index = (int)InfoType;
  1713. if (m_InfoItems[index] == NULL)
  1714. {
  1715. LOG((MSP_INFO, "no local participant info item for %d", InfoType));
  1716. return TAPI_E_NOITEMS;
  1717. }
  1718. // make a BSTR out of it.
  1719. BSTR pName = SysAllocString(m_InfoItems[index]);
  1720. if (pName == NULL)
  1721. {
  1722. LOG((MSP_ERROR, "CParticipant get info - exit out of mem"));
  1723. return E_POINTER;
  1724. }
  1725. // return the BSTR.
  1726. *ppInfo = pName;
  1727. return S_OK;
  1728. }
  1729. // ITLocalParticipant methods, called by the app.
  1730. STDMETHODIMP CIPConfMSPCall::put_LocalParticipantTypedInfo(
  1731. IN PARTICIPANT_TYPED_INFO InfoType,
  1732. IN BSTR pInfo
  1733. )
  1734. /*++
  1735. Routine Description:
  1736. Set a information item for the local participant. This information is
  1737. sent out to other participants in the conference.
  1738. Arguments:
  1739. InfoType - The type of the information item.
  1740. pInfo - the information item.
  1741. Return Value:
  1742. S_OK,
  1743. E_INVALIDARG,
  1744. E_POINTER,
  1745. E_OUTOFMEMORY,
  1746. TAPI_E_NOITEMS
  1747. */
  1748. {
  1749. LOG((MSP_TRACE, "set local info, type:%d", InfoType));
  1750. if (InfoType > PTI_PRIVATE || InfoType < PTI_CANONICALNAME)
  1751. {
  1752. LOG((MSP_ERROR, "set local info - invalid type:%d", InfoType));
  1753. return E_INVALIDARG;
  1754. }
  1755. if (IsBadStringPtr(pInfo, MAX_PARTICIPANT_TYPED_INFO_LENGTH))
  1756. {
  1757. LOG((MSP_ERROR, "set local info, bad ptr:%p", pInfo));
  1758. return E_POINTER;
  1759. }
  1760. DWORD dwStringLen = lstrlenW(pInfo);
  1761. if (dwStringLen > MAX_PARTICIPANT_TYPED_INFO_LENGTH)
  1762. {
  1763. LOG((MSP_ERROR, "local info too long"));
  1764. return E_INVALIDARG;
  1765. }
  1766. // check if we have that info.
  1767. CLock lock(m_lock);
  1768. if (m_fCallStarted)
  1769. {
  1770. return TAPI_E_INVALCALLSTATE;
  1771. }
  1772. if (!m_fLocalInfoRetrieved)
  1773. {
  1774. HRESULT hr = InitializeLocalParticipant();
  1775. if (FAILED(hr))
  1776. {
  1777. return hr;
  1778. }
  1779. }
  1780. int index = (int)InfoType;
  1781. if (m_InfoItems[index] != NULL)
  1782. {
  1783. if (lstrcmpW(m_InfoItems[index], pInfo) == 0)
  1784. {
  1785. // The info is the same as what we are using.
  1786. return S_OK;
  1787. }
  1788. // the infomation is different, release the old info.
  1789. free(m_InfoItems[index]);
  1790. m_InfoItems[index] = NULL;
  1791. }
  1792. // save the info.
  1793. m_InfoItems[index] = (WCHAR *)malloc((dwStringLen + 1)* sizeof(WCHAR));
  1794. if (m_InfoItems[index] == NULL)
  1795. {
  1796. LOG((MSP_ERROR, "out of mem for local info"));
  1797. return E_OUTOFMEMORY;
  1798. }
  1799. lstrcpynW(m_InfoItems[index], pInfo, dwStringLen + 1);
  1800. //
  1801. // The info is new, we need to set it on the streams.
  1802. //
  1803. for (int i = 0; i < m_Streams.GetSize(); i ++)
  1804. {
  1805. ((CIPConfMSPStream*)m_Streams[i])->SetLocalParticipantInfo(
  1806. InfoType,
  1807. m_InfoItems[index],
  1808. dwStringLen
  1809. );
  1810. }
  1811. return S_OK;
  1812. }
  1813. STDMETHODIMP CIPConfMSPCall::SetQOSApplicationID (
  1814. IN BSTR pApplicationID,
  1815. IN BSTR pApplicationGUID,
  1816. IN BSTR pSubIDs
  1817. )
  1818. /*++
  1819. Routine Description:
  1820. This method is called by the App to set the QOS specific application ID.
  1821. It can only be called before the call is connected.
  1822. Arguments:
  1823. pApplicationID - the Application ID.
  1824. pSubIDs - the SubIDs that will be appended to the end of policy locator.
  1825. Return Value:
  1826. S_OK
  1827. E_OUTOFMEMORY
  1828. --*/
  1829. {
  1830. CLock lock(m_lock);
  1831. if (m_fCallStarted)
  1832. {
  1833. return TAPI_E_INVALCALLSTATE;
  1834. }
  1835. if (pSubIDs!=NULL && lstrlenW(pSubIDs)>MAX_QOS_ID_LEN)
  1836. {
  1837. return E_INVALIDARG;
  1838. }
  1839. if (pApplicationID!=NULL && lstrlenW(pApplicationID)>MAX_QOS_ID_LEN)
  1840. {
  1841. return E_INVALIDARG;
  1842. }
  1843. try
  1844. {
  1845. if (m_pApplicationID) SysFreeString(m_pApplicationID);
  1846. m_pApplicationID = SysAllocString(pApplicationID);
  1847. }
  1848. catch(...)
  1849. {
  1850. return E_POINTER;
  1851. }
  1852. if (m_pApplicationID == NULL)
  1853. {
  1854. return E_OUTOFMEMORY;
  1855. }
  1856. try
  1857. {
  1858. if (m_pApplicationGUID)
  1859. {
  1860. SysFreeString(m_pApplicationGUID);
  1861. m_pApplicationGUID = NULL;
  1862. }
  1863. if (m_pSubIDs)
  1864. {
  1865. SysFreeString(m_pSubIDs);
  1866. m_pSubIDs = NULL;
  1867. }
  1868. if (pApplicationGUID)
  1869. {
  1870. m_pApplicationGUID = SysAllocString(pApplicationGUID);
  1871. }
  1872. if (pSubIDs)
  1873. {
  1874. m_pSubIDs = SysAllocString(pSubIDs);
  1875. }
  1876. }
  1877. catch(...)
  1878. {
  1879. SysFreeString(m_pApplicationID);
  1880. m_pApplicationID = NULL;
  1881. if (m_pApplicationGUID)
  1882. {
  1883. SysFreeString(m_pApplicationGUID);
  1884. }
  1885. m_pApplicationGUID = NULL;
  1886. if (m_pSubIDs)
  1887. {
  1888. SysFreeString(m_pSubIDs);
  1889. }
  1890. m_pSubIDs = NULL;
  1891. return E_POINTER;
  1892. }
  1893. if ((pApplicationGUID!=NULL && m_pApplicationGUID==NULL) ||
  1894. (pSubIDs!=NULL && m_pSubIDs==NULL))
  1895. {
  1896. return E_OUTOFMEMORY;
  1897. }
  1898. return S_OK;
  1899. }
  1900. HRESULT CIPConfMSPCall::NewParticipant(
  1901. IN ITStream * pITStream,
  1902. IN DWORD dwSSRC,
  1903. IN DWORD dwSendRecv,
  1904. IN DWORD dwMediaType,
  1905. IN WCHAR * szCName,
  1906. OUT ITParticipant ** ppITParticipant
  1907. )
  1908. /*++
  1909. Routine Description:
  1910. This method is called by a stream object when a new participant appears.
  1911. It looks throught the call's participant list, if the partcipant is
  1912. already in the list, it returns the pointer to the object. If it is not
  1913. found, a new object will be created and added into the list.
  1914. Arguments:
  1915. pITStream - the stream object.
  1916. dwSSRC - the SSRC of the participant in the stream.
  1917. dwSendRecv - a sender or a receiver.
  1918. dwMediaType - the media type of the stream.
  1919. szCName - the canonical name of the participant.
  1920. ppITParticipant - the address to store the returned pointer.
  1921. Return Value:
  1922. S_OK
  1923. E_OUTOFMEMORY
  1924. --*/
  1925. {
  1926. CLock lock(m_ParticipantLock);
  1927. HRESULT hr;
  1928. // First check to see if the participant is in our list. If he is already
  1929. // in the list, just return the object.
  1930. int index;
  1931. if (m_Participants.FindByCName(szCName, &index))
  1932. {
  1933. hr = ((CParticipant *)m_Participants[index])->
  1934. AddStream(pITStream, dwSSRC, dwSendRecv, dwMediaType);
  1935. if (FAILED(hr))
  1936. {
  1937. LOG((MSP_ERROR, "can not add a stream to a participant:%x", hr));
  1938. return hr;
  1939. }
  1940. *ppITParticipant = m_Participants[index];
  1941. (*ppITParticipant)->AddRef();
  1942. return S_OK;
  1943. }
  1944. // create a new participant object.
  1945. CComObject<CParticipant> * pCOMParticipant;
  1946. hr = ::CreateCComObjectInstance(&pCOMParticipant);
  1947. if (NULL == pCOMParticipant)
  1948. {
  1949. LOG((MSP_ERROR, "can not create a new participant:%x", hr));
  1950. return hr;
  1951. }
  1952. ITParticipant* pITParticipant;
  1953. // get the interface pointer.
  1954. hr = pCOMParticipant->_InternalQueryInterface(
  1955. __uuidof(ITParticipant),
  1956. (void **)&pITParticipant
  1957. );
  1958. if (FAILED(hr))
  1959. {
  1960. LOG((MSP_ERROR, "Participant QueryInterface failed: %x", hr));
  1961. delete pCOMParticipant;
  1962. return hr;
  1963. }
  1964. // Initialize the object.
  1965. hr = pCOMParticipant->Init(
  1966. szCName, pITStream, dwSSRC, dwSendRecv, dwMediaType
  1967. );
  1968. if (FAILED(hr))
  1969. {
  1970. LOG((MSP_ERROR, "Create participant:call init failed: %x", hr));
  1971. pITParticipant->Release();
  1972. return hr;
  1973. }
  1974. // Add the Participant into our list of Participants.
  1975. if (!m_Participants.InsertAt(index, pITParticipant))
  1976. {
  1977. pITParticipant->Release();
  1978. LOG((MSP_ERROR, "out of memory in adding a Participant."));
  1979. return E_OUTOFMEMORY;
  1980. }
  1981. // AddRef the interface pointer and return it.
  1982. pITParticipant->AddRef();
  1983. *ppITParticipant = pITParticipant;
  1984. SendParticipantEvent(PE_NEW_PARTICIPANT, pITParticipant);
  1985. return S_OK;
  1986. }
  1987. HRESULT CIPConfMSPCall::ParticipantLeft(
  1988. IN ITParticipant * pITParticipant
  1989. )
  1990. /*++
  1991. Routine Description:
  1992. This method is called by a stream object when a participant left the
  1993. conference.
  1994. Arguments:
  1995. pITParticipant - the participant that left.
  1996. Return Value:
  1997. S_OK
  1998. --*/
  1999. {
  2000. m_ParticipantLock.Lock();
  2001. BOOL fRemoved = m_Participants.Remove(pITParticipant);
  2002. m_ParticipantLock.Unlock();
  2003. if (fRemoved)
  2004. {
  2005. SendParticipantEvent(PE_PARTICIPANT_LEAVE, pITParticipant);
  2006. pITParticipant->Release();
  2007. }
  2008. else
  2009. {
  2010. LOG((MSP_ERROR, "can't remove Participant %p", pITParticipant));
  2011. }
  2012. return S_OK;
  2013. }
  2014. void CIPConfMSPCall::SendParticipantEvent(
  2015. IN PARTICIPANT_EVENT Event,
  2016. IN ITParticipant * pITParticipant,
  2017. IN ITSubStream * pITSubStream
  2018. ) const
  2019. /*++
  2020. Routine Description:
  2021. This method is called by a stream object to send a participant related
  2022. event to the app.
  2023. Arguments:
  2024. Event - the event code.
  2025. pITParticipant - the participant object.
  2026. pITSubStream - the substream object, if any.
  2027. Return Value:
  2028. nothing.
  2029. --*/
  2030. {
  2031. if (pITParticipant)
  2032. {
  2033. LOG((MSP_TRACE, "SendParticipantEvent, event %s, participant:%ws",
  2034. ParticipantEventString[Event],
  2035. ((CParticipant*)pITParticipant)->Name()
  2036. ));
  2037. }
  2038. else
  2039. {
  2040. LOG((MSP_TRACE, "SendParticipantEvent, event %s, participant: null (local)",
  2041. ParticipantEventString[Event]
  2042. ));
  2043. }
  2044. // Create a private event object.
  2045. CComPtr<IDispatch> pEvent;
  2046. HRESULT hr = CreateParticipantEvent(
  2047. Event,
  2048. pITParticipant,
  2049. pITSubStream,
  2050. &pEvent
  2051. );
  2052. if (FAILED(hr))
  2053. {
  2054. LOG((MSP_ERROR, "create event returned: %x", hr));
  2055. return;
  2056. }
  2057. MSPEVENTITEM* pEventItem = AllocateEventItem();
  2058. if (pEventItem == NULL)
  2059. {
  2060. LOG((MSP_ERROR, "No memory for the TSPMSP data"));
  2061. return;
  2062. }
  2063. // Fill in the necessary fields for the event structure.
  2064. pEventItem->MSPEventInfo.dwSize = sizeof(MSP_EVENT_INFO);;
  2065. pEventItem->MSPEventInfo.Event = ME_PRIVATE_EVENT;
  2066. pEventItem->MSPEventInfo.hCall = m_htCall;
  2067. pEventItem->MSPEventInfo.MSP_PRIVATE_EVENT_INFO.pEvent = pEvent;
  2068. pEventItem->MSPEventInfo.MSP_PRIVATE_EVENT_INFO.lEventCode = Event;
  2069. pEvent->AddRef();
  2070. // send the event to tapi.
  2071. hr = m_pMSPAddress->PostEvent(pEventItem);
  2072. if (FAILED(hr))
  2073. {
  2074. LOG((MSP_ERROR, "Post event failed %x", hr));
  2075. pEvent->Release();
  2076. FreeEventItem(pEventItem);
  2077. }
  2078. }
  2079. VOID CIPConfMSPCall::HandleGraphEvent(
  2080. IN MSPSTREAMCONTEXT * pContext
  2081. )
  2082. {
  2083. long lEventCode;
  2084. LONG_PTR lParam1, lParam2; // win64 fix
  2085. HRESULT hr = pContext->pIMediaEvent->GetEvent(&lEventCode, &lParam1, &lParam2, 0);
  2086. if (FAILED(hr))
  2087. {
  2088. LOG((MSP_ERROR, "Can not get the actual event. %x", hr));
  2089. return;
  2090. }
  2091. LOG((MSP_EVENT, "ProcessGraphEvent, code:%d param1:%x param2:%x",
  2092. lEventCode, lParam1, lParam2));
  2093. if (lEventCode == EC_PALETTE_CHANGED
  2094. || lEventCode == EC_VIDEO_SIZE_CHANGED)
  2095. {
  2096. LOG((MSP_EVENT, "event %d ignored", lEventCode));
  2097. return;
  2098. }
  2099. //
  2100. // Create an event data structure that we will pass to the worker thread.
  2101. //
  2102. MULTI_GRAPH_EVENT_DATA * pData;
  2103. pData = new MULTI_GRAPH_EVENT_DATA;
  2104. if (pData == NULL)
  2105. {
  2106. LOG((MSP_ERROR, "Out of memory for event data."));
  2107. return;
  2108. }
  2109. pData->pCall = this;
  2110. pData->pITStream = pContext->pITStream;
  2111. pData->lEventCode = lEventCode;
  2112. pData->lParam1 = lParam1;
  2113. pData->lParam2 = lParam2;
  2114. //
  2115. // also pass an addref'ed pointer to IMediaEvent, so that whoever processes
  2116. // the message has the opportunity to free event parameters
  2117. //
  2118. pData->pIMediaEvent = pContext->pIMediaEvent;
  2119. pData->pIMediaEvent->AddRef();
  2120. //
  2121. // Make sure the call and stream don't go away while we handle the event.
  2122. // but use our special inner object addref for the call
  2123. //
  2124. pData->pCall->MSPCallAddRef();
  2125. pData->pITStream->AddRef();
  2126. //
  2127. // Queue an async work item to call ProcessGraphEvent.
  2128. //
  2129. hr = g_Thread.QueueWorkItem(AsyncMultiGraphEvent,
  2130. (void *) pData,
  2131. FALSE); // asynchronous
  2132. if (FAILED(hr))
  2133. {
  2134. LOG((MSP_ERROR, "QueueWorkItem failed, return code:%x", hr));
  2135. pData->pCall->MSPCallRelease();
  2136. pData->pITStream->Release();
  2137. //
  2138. // no one is going to free event params and release the IMediaEvent
  2139. // pointer, so do it here
  2140. //
  2141. pContext->pIMediaEvent->FreeEventParams(lEventCode, lParam1, lParam2);
  2142. pData->pIMediaEvent->Release();
  2143. delete pData;
  2144. }
  2145. }
  2146. HRESULT CIPConfMSPCall::InitFullDuplexControler()
  2147. /*++
  2148. Routine Description:
  2149. This method creates the full-duplex controller object used to control
  2150. audio devices.
  2151. Arguments:
  2152. NONE
  2153. Return Value:
  2154. HRESULT.
  2155. --*/
  2156. {
  2157. ENTER_FUNCTION("CIPConfMSPCall::InitFullDuplexControler");
  2158. LOG((MSP_TRACE, "%s entered", __fxName));
  2159. CLock lock(m_lock);
  2160. if (m_pIAudioDuplexController != NULL)
  2161. {
  2162. return S_OK;
  2163. }
  2164. HRESULT hr;
  2165. IAudioDuplexController *pIAudioDuplexController;
  2166. if (FAILED(hr = CoCreateInstance(
  2167. __uuidof(TAPIAudioDuplexController),
  2168. NULL,
  2169. CLSCTX_INPROC_SERVER | CLSCTX_NO_CODE_DOWNLOAD,
  2170. __uuidof(IAudioDuplexController),
  2171. (void **) &pIAudioDuplexController
  2172. )))
  2173. {
  2174. LOG((MSP_ERROR, "%s, create AudioDuplexController failed. hr=%x",
  2175. __fxName, hr));
  2176. return hr;
  2177. }
  2178. m_pIAudioDuplexController = pIAudioDuplexController;
  2179. return S_OK;
  2180. }
  2181. /*++
  2182. Routine Description:
  2183. ITCallQualityControl method. Delegated to quality control relay
  2184. --*/
  2185. STDMETHODIMP
  2186. CIPConfMSPCall::GetRange (
  2187. IN CallQualityProperty Property,
  2188. OUT long *plMin,
  2189. OUT long *plMax,
  2190. OUT long *plSteppingDelta,
  2191. OUT long *plDefault,
  2192. OUT TAPIControlFlags *plFlags
  2193. )
  2194. {
  2195. return m_pCallQCRelay->GetRange (Property, plMin, plMax, plSteppingDelta, plDefault, plFlags);
  2196. }
  2197. /*++
  2198. Routine Description:
  2199. ITCallQualityControl method. Delegated to quality control relay
  2200. --*/
  2201. STDMETHODIMP
  2202. CIPConfMSPCall::Get (
  2203. IN CallQualityProperty Property,
  2204. OUT long *plValue,
  2205. OUT TAPIControlFlags *plFlags
  2206. )
  2207. {
  2208. return m_pCallQCRelay->Get (Property, plValue, plFlags);
  2209. }
  2210. /*++
  2211. Routine Description:
  2212. ITCallQualityControl method. Delegated to quality control relay
  2213. --*/
  2214. STDMETHODIMP
  2215. CIPConfMSPCall::Set (
  2216. IN CallQualityProperty Property,
  2217. IN long lValue,
  2218. IN TAPIControlFlags lFlags
  2219. )
  2220. {
  2221. return m_pCallQCRelay->Set (Property, lValue, lFlags);
  2222. }
  2223. /*++
  2224. Routine Description:
  2225. IInnerCallQualityControl method. Delegated to quality control relay
  2226. --*/
  2227. STDMETHODIMP_(ULONG)
  2228. CIPConfMSPCall::InnerCallAddRef (VOID)
  2229. {
  2230. return this->MSPCallAddRef ();
  2231. }
  2232. STDMETHODIMP_(ULONG)
  2233. CIPConfMSPCall::InnerCallRelease (VOID)
  2234. {
  2235. return this->MSPCallRelease ();
  2236. }
  2237. STDMETHODIMP
  2238. CIPConfMSPCall::RegisterInnerStreamQC (
  2239. IN IInnerStreamQualityControl *pIInnerStreamQC
  2240. )
  2241. {
  2242. return m_pCallQCRelay->RegisterInnerStreamQC (
  2243. pIInnerStreamQC
  2244. );
  2245. }
  2246. /*++
  2247. Routine Description:
  2248. IInnerCallQualityControl method. Delegated to quality control relay
  2249. --*/
  2250. STDMETHODIMP
  2251. CIPConfMSPCall::DeRegisterInnerStreamQC (
  2252. IN IInnerStreamQualityControl *pIInnerStreamQC
  2253. )
  2254. {
  2255. return m_pCallQCRelay->DeRegisterInnerStreamQC (
  2256. pIInnerStreamQC
  2257. );
  2258. }
  2259. /*++
  2260. Routine Description:
  2261. IInnerCallQualityControl method. Delegated to quality control relay
  2262. --*/
  2263. STDMETHODIMP
  2264. CIPConfMSPCall::ProcessQCEvent (
  2265. IN QCEvent event,
  2266. IN DWORD dwParam
  2267. )
  2268. {
  2269. return m_pCallQCRelay->ProcessQCEvent (event, dwParam);
  2270. }