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.

13658 lines
336 KiB

  1. /*++
  2. Copyright (c) 1997-1999 Microsoft Corporation
  3. Module Name:
  4. address.cpp
  5. Abstract:
  6. Implementation of the Address object for TAPI 3.0.
  7. Author:
  8. mquinton - 4/17/97
  9. Notes:
  10. optional-notes
  11. Revision History:
  12. --*/
  13. #include "stdafx.h"
  14. #include "uuids.h"
  15. #include "TermEvnt.h"
  16. #include "tapievt.h"
  17. const CLSID CLSID_WAVEMSP = {0x4DDB6D36,0x3BC1,0x11d2,{0x86,0xF2,0x00,0x60,0x08,0xB0,0xE5,0xD2}};
  18. extern HRESULT mapTAPIErrorCode(long lErrorCode);
  19. extern ULONG_PTR GenerateHandleAndAddToHashTable( ULONG_PTR Element);
  20. extern void RemoveHandleFromHashTable(ULONG_PTR dwHandle);
  21. extern CHashTable * gpHandleHashTable;
  22. extern CRetryQueue * gpRetryQueue;
  23. extern HANDLE ghAsyncRetryQueueEvent;
  24. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  25. // WAITORTIMERCALLBACK
  26. // CAddress::MSPEventCallback(
  27. // PVOID pContext,
  28. // BOOLEAN b
  29. // )
  30. //
  31. // this is the callback that gets called when an msp
  32. // sets it's message event
  33. //
  34. // queues a message to be handled in the callback thread
  35. //
  36. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  37. VOID
  38. NTAPI
  39. MSPEventCallback(
  40. VOID * pContext,
  41. BOOLEAN bFlag
  42. )
  43. {
  44. CAddress * pAddress;
  45. HRESULT hr;
  46. gpHandleHashTable->Lock();
  47. hr = gpHandleHashTable->Find( (ULONG_PTR)pContext, (ULONG_PTR *)&pAddress );
  48. if ( SUCCEEDED(hr) )
  49. {
  50. LOG((TL_INFO, "MSPEventCallback - Matched handle %p to Address object %p", pContext, pAddress ));
  51. ASYNCEVENTMSG msg;
  52. DWORD dwAddressHandle;
  53. pAddress->AddRef();
  54. gpHandleHashTable->Unlock();
  55. dwAddressHandle = CreateHandleTableEntry ((ULONG_PTR)pAddress);
  56. if (0 != dwAddressHandle)
  57. {
  58. ZeroMemory( &msg, sizeof(msg) );
  59. msg.TotalSize = sizeof(msg);
  60. msg.Msg = PRIVATE_MSPEVENT;
  61. msg.Param1 = dwAddressHandle;
  62. //
  63. // try to queue the event
  64. //
  65. BOOL bQueueEventSuccess = gpRetryQueue->QueueEvent(&msg);
  66. if (bQueueEventSuccess)
  67. {
  68. SetEvent(ghAsyncRetryQueueEvent);
  69. }
  70. else
  71. {
  72. //
  73. // RetryQueue is either no longer accepting entries or
  74. // failed to allocate resources it needed.
  75. //
  76. // in any case, the event will not be processed. cleanup.
  77. //
  78. DeleteHandleTableEntry (dwAddressHandle);
  79. pAddress->Release();
  80. pAddress = NULL;
  81. LOG((TL_ERROR, "MSPEventCallback - Couldn't enqueue event"));
  82. }
  83. }
  84. }
  85. else
  86. {
  87. LOG((TL_ERROR, "MSPEventCallback - Couldn't match handle %p to Address object ", pContext));
  88. gpHandleHashTable->Unlock();
  89. }
  90. }
  91. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  92. //
  93. // void HandlePrivateMSPEvent( PASYNCEVENTMSG pParams )
  94. //
  95. // actually handles the msp event. this is called in the
  96. // asynceventsthread
  97. //
  98. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  99. void HandlePrivateMSPEvent( PASYNCEVENTMSG pParams )
  100. {
  101. CAddress * pAddress;
  102. pAddress = (CAddress *) GetHandleTableEntry(pParams->Param1);
  103. DeleteHandleTableEntry (pParams->Param1);
  104. if (NULL != pAddress)
  105. {
  106. pAddress->MSPEvent();
  107. //
  108. // address is addref'd when event
  109. // is packaged.
  110. //
  111. pAddress->Release();
  112. }
  113. }
  114. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  115. //
  116. // Class : CEventMasks
  117. // Method : SetSubEventFlag
  118. //
  119. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  120. HRESULT CEventMasks::SetSubEventFlag(
  121. DWORD dwEvent, // The event
  122. DWORD dwFlag, // The flag that should be set
  123. BOOL bEnable
  124. )
  125. {
  126. //
  127. // Enter critical section
  128. //
  129. //m_cs.Lock();
  130. //
  131. // Prepare the subevent flag
  132. //
  133. DWORD dwSubEventFlag = 0;
  134. dwSubEventFlag = (dwFlag == EM_ALLSUBEVENTS ) ?
  135. EM_ALLSUBEVENTS :
  136. GET_SUBEVENT_FLAG( dwFlag );
  137. //
  138. // Set the mask for the events
  139. //
  140. switch( dwEvent )
  141. {
  142. case EM_ALLEVENTS:
  143. case TE_TAPIOBJECT:
  144. if( bEnable )
  145. {
  146. // Set the bit
  147. m_dwTapiObjectMask = m_dwTapiObjectMask | dwSubEventFlag;
  148. }
  149. else
  150. {
  151. // Reset the bit
  152. m_dwTapiObjectMask = m_dwTapiObjectMask & ( ~dwSubEventFlag );
  153. }
  154. if( dwEvent == TE_TAPIOBJECT)
  155. {
  156. break;
  157. }
  158. case TE_ADDRESS:
  159. if( bEnable )
  160. {
  161. // Set the bit
  162. m_dwAddressMask = m_dwAddressMask | dwSubEventFlag;
  163. }
  164. else
  165. {
  166. // Reset the bit
  167. m_dwAddressMask = m_dwAddressMask & ( ~dwSubEventFlag );
  168. }
  169. if( dwEvent == TE_ADDRESS)
  170. {
  171. break;
  172. }
  173. case TE_CALLNOTIFICATION:
  174. if( bEnable )
  175. {
  176. // Set the bit
  177. m_dwCallNotificationMask = m_dwCallNotificationMask | dwSubEventFlag;
  178. }
  179. else
  180. {
  181. // Reset the bit
  182. m_dwCallNotificationMask = m_dwCallNotificationMask & ( ~dwSubEventFlag );
  183. }
  184. if( dwEvent == TE_CALLNOTIFICATION)
  185. {
  186. break;
  187. }
  188. case TE_CALLSTATE:
  189. if( bEnable )
  190. {
  191. // Set the bit
  192. m_dwCallStateMask = m_dwCallStateMask | dwSubEventFlag;
  193. }
  194. else
  195. {
  196. // Reset the bit
  197. m_dwCallStateMask = m_dwCallStateMask & ( ~dwSubEventFlag );
  198. }
  199. if( dwEvent == TE_CALLSTATE)
  200. {
  201. break;
  202. }
  203. case TE_CALLMEDIA:
  204. if( bEnable )
  205. {
  206. // Set the bit
  207. m_dwCallMediaMask = m_dwCallMediaMask | dwSubEventFlag;
  208. }
  209. else
  210. {
  211. // Reset the bit
  212. m_dwCallMediaMask = m_dwCallMediaMask & ( ~dwSubEventFlag );
  213. }
  214. if( dwEvent == TE_CALLMEDIA)
  215. {
  216. break;
  217. }
  218. case TE_CALLHUB:
  219. if( bEnable )
  220. {
  221. // Set the bit
  222. m_dwCallHubMask = m_dwCallHubMask | dwSubEventFlag;
  223. }
  224. else
  225. {
  226. // Reset the bit
  227. m_dwCallHubMask = m_dwCallHubMask & ( ~dwSubEventFlag );
  228. }
  229. if( dwEvent == TE_CALLHUB)
  230. {
  231. break;
  232. }
  233. case TE_CALLINFOCHANGE:
  234. if( bEnable )
  235. {
  236. // Set the bit
  237. m_dwCallInfoChangeMask = m_dwCallInfoChangeMask | dwSubEventFlag;
  238. }
  239. else
  240. {
  241. // Reset the bit
  242. m_dwCallInfoChangeMask = m_dwCallInfoChangeMask & ( ~dwSubEventFlag );
  243. }
  244. if( dwEvent == TE_CALLINFOCHANGE)
  245. {
  246. break;
  247. }
  248. case TE_QOSEVENT:
  249. if( bEnable )
  250. {
  251. // Set the bit
  252. m_dwQOSEventMask = m_dwQOSEventMask | dwSubEventFlag;
  253. }
  254. else
  255. {
  256. // Reset the bit
  257. m_dwQOSEventMask = m_dwQOSEventMask & ( ~dwSubEventFlag );
  258. }
  259. if( dwEvent == TE_QOSEVENT)
  260. {
  261. break;
  262. }
  263. case TE_FILETERMINAL:
  264. if( bEnable )
  265. {
  266. // Set the bit
  267. m_dwFileTerminalMask = m_dwFileTerminalMask | dwSubEventFlag;
  268. }
  269. else
  270. {
  271. // Reset the bit
  272. m_dwFileTerminalMask = m_dwFileTerminalMask & ( ~dwSubEventFlag );
  273. }
  274. if( dwEvent == TE_FILETERMINAL)
  275. {
  276. break;
  277. }
  278. case TE_PRIVATE:
  279. if( bEnable )
  280. {
  281. // Set the entire mask
  282. m_dwPrivateMask = EM_ALLSUBEVENTS;
  283. }
  284. else
  285. {
  286. // Reset the entire mask
  287. m_dwPrivateMask = 0;
  288. }
  289. if( dwEvent == TE_PRIVATE)
  290. {
  291. break;
  292. }
  293. case TE_ADDRESSDEVSPECIFIC:
  294. if( bEnable )
  295. {
  296. // Set the bit
  297. m_dwAddressDevSpecificMask = m_dwAddressDevSpecificMask | dwSubEventFlag;
  298. }
  299. else
  300. {
  301. // Reset the bit
  302. m_dwAddressDevSpecificMask = m_dwAddressDevSpecificMask & ( ~dwSubEventFlag );
  303. }
  304. if( dwEvent == TE_ADDRESSDEVSPECIFIC)
  305. {
  306. break;
  307. }
  308. case TE_PHONEDEVSPECIFIC:
  309. if( bEnable )
  310. {
  311. // Set the bit
  312. m_dwPhoneDevSpecificMask = m_dwPhoneDevSpecificMask | dwSubEventFlag;
  313. }
  314. else
  315. {
  316. // Reset the bit
  317. m_dwPhoneDevSpecificMask = m_dwPhoneDevSpecificMask & ( ~dwSubEventFlag );
  318. }
  319. if( dwEvent == TE_PHONEDEVSPECIFIC)
  320. {
  321. break;
  322. }
  323. }
  324. //
  325. // Leave critical section
  326. //
  327. //m_cs.Unlock();
  328. return S_OK;
  329. }
  330. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  331. //
  332. // Class : CEventMasks
  333. // Method : GetSubEventFlag
  334. //
  335. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  336. HRESULT CEventMasks::GetSubEventFlag(
  337. DWORD dwEvent, // The event
  338. DWORD dwFlag, // The flag that should be setted
  339. BOOL* pEnable
  340. )
  341. {
  342. //
  343. // Enter critical section
  344. //
  345. //m_cs.Lock();
  346. //
  347. // Prepare the subevent flag
  348. //
  349. DWORD dwSubEventFlag = 0;
  350. dwSubEventFlag = GET_SUBEVENT_FLAG( dwFlag );
  351. //
  352. // Reset thre flag
  353. //
  354. *pEnable = FALSE;
  355. //
  356. // Get the flag for the events
  357. //
  358. switch( dwEvent )
  359. {
  360. case TE_TAPIOBJECT:
  361. *pEnable = ((m_dwTapiObjectMask & dwSubEventFlag) != 0);
  362. break;
  363. case TE_ADDRESS:
  364. *pEnable = ((m_dwAddressMask & dwSubEventFlag) != 0);
  365. break;
  366. case TE_CALLNOTIFICATION:
  367. *pEnable = ((m_dwCallNotificationMask & dwSubEventFlag) != 0);
  368. break;
  369. case TE_CALLSTATE:
  370. *pEnable = ((m_dwCallStateMask & dwSubEventFlag) != 0);
  371. break;
  372. case TE_CALLMEDIA:
  373. *pEnable = ((m_dwCallMediaMask & dwSubEventFlag) != 0);
  374. break;
  375. case TE_CALLHUB:
  376. *pEnable = ((m_dwCallHubMask & dwSubEventFlag) != 0);
  377. break;
  378. case TE_CALLINFOCHANGE:
  379. *pEnable = ((m_dwCallInfoChangeMask & dwSubEventFlag) != 0);
  380. break;
  381. case TE_QOSEVENT:
  382. *pEnable = ((m_dwQOSEventMask & dwSubEventFlag) != 0);
  383. break;
  384. case TE_FILETERMINAL:
  385. *pEnable = ((m_dwFileTerminalMask & dwSubEventFlag) != 0);
  386. break;
  387. case TE_PRIVATE:
  388. // We don't have subevents for private events
  389. *pEnable = ( m_dwPrivateMask != 0);
  390. break;
  391. case TE_ADDRESSDEVSPECIFIC:
  392. // We don't have subevents for TE_ADDRESSDEVSPECIFIC events
  393. *pEnable = ( m_dwAddressDevSpecificMask != 0);
  394. break;
  395. case TE_PHONEDEVSPECIFIC:
  396. // We don't have subevents for TE_PHONEDEVSPECIFIC events
  397. *pEnable = ( m_dwPhoneDevSpecificMask != 0);
  398. break;
  399. }
  400. //
  401. // Leave critical section
  402. //
  403. //m_cs.Unlock();
  404. return S_OK;
  405. }
  406. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  407. //
  408. // Class : CEventMasks
  409. // Method : GetSubEventMask
  410. //
  411. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  412. DWORD CEventMasks::GetSubEventMask(
  413. TAPI_EVENT TapiEvent
  414. )
  415. {
  416. //
  417. // Enter critical section
  418. //
  419. //m_cs.Lock();
  420. DWORD dwSubEventMask = EM_NOSUBEVENTS;
  421. switch( TapiEvent )
  422. {
  423. case TE_TAPIOBJECT:
  424. dwSubEventMask = m_dwTapiObjectMask;
  425. break;
  426. case TE_ADDRESS:
  427. dwSubEventMask = m_dwAddressMask;
  428. break;
  429. case TE_CALLNOTIFICATION:
  430. dwSubEventMask = m_dwCallNotificationMask;
  431. break;
  432. case TE_CALLSTATE:
  433. dwSubEventMask = m_dwCallStateMask;
  434. break;
  435. case TE_CALLMEDIA:
  436. dwSubEventMask = m_dwCallMediaMask;
  437. break;
  438. case TE_CALLHUB:
  439. dwSubEventMask = m_dwCallHubMask;
  440. break;
  441. case TE_CALLINFOCHANGE:
  442. dwSubEventMask = m_dwCallInfoChangeMask;
  443. break;
  444. case TE_QOSEVENT:
  445. dwSubEventMask = m_dwQOSEventMask;
  446. break;
  447. case TE_FILETERMINAL:
  448. dwSubEventMask = m_dwFileTerminalMask;
  449. break;
  450. case TE_PRIVATE:
  451. dwSubEventMask = m_dwPrivateMask;
  452. break;
  453. case TE_ADDRESSDEVSPECIFIC:
  454. dwSubEventMask = m_dwAddressDevSpecificMask;
  455. break;
  456. case TE_PHONEDEVSPECIFIC:
  457. dwSubEventMask = m_dwPhoneDevSpecificMask;
  458. break;
  459. default:
  460. dwSubEventMask = EM_NOSUBEVENTS;
  461. break;
  462. }
  463. //
  464. // Leave critical section
  465. //
  466. //m_cs.Unlock();
  467. return dwSubEventMask;
  468. }
  469. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  470. //
  471. // Class : CEventMasks
  472. // Method : IsSubEventValid
  473. //
  474. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  475. BOOL CEventMasks::IsSubEventValid(
  476. TAPI_EVENT TapiEvent,
  477. DWORD dwSubEvent,
  478. BOOL bAcceptAllSubEvents,
  479. BOOL bCallLevel
  480. )
  481. {
  482. BOOL bValid = TRUE;
  483. switch( TapiEvent )
  484. {
  485. case TE_TAPIOBJECT:
  486. //
  487. // Just at the address level
  488. //
  489. if( bCallLevel)
  490. {
  491. bValid = FALSE;
  492. break;
  493. }
  494. //
  495. // ALLSUBEVENTS
  496. //
  497. if( bAcceptAllSubEvents )
  498. {
  499. //
  500. // We could accept 'ALLSUBEVENTS' flag
  501. // when we try to set for all subevents
  502. //
  503. if(dwSubEvent == EM_ALLSUBEVENTS)
  504. {
  505. break;
  506. }
  507. }
  508. //
  509. // In CTapiObjectEvent::get_Address we accept just
  510. // these three subevents
  511. //
  512. if( (dwSubEvent != TE_ADDRESSCREATE) &&
  513. (dwSubEvent != TE_ADDRESSREMOVE) &&
  514. (dwSubEvent != TE_ADDRESSCLOSE) )
  515. {
  516. bValid = FALSE;
  517. }
  518. break;
  519. case TE_ADDRESS:
  520. //
  521. // Just at the address level
  522. //
  523. if( bCallLevel)
  524. {
  525. bValid = FALSE;
  526. break;
  527. }
  528. //
  529. // ALLSUBEVENTS
  530. //
  531. if( bAcceptAllSubEvents )
  532. {
  533. //
  534. // We could accept 'ALLSUBEVENTS' flag
  535. // when we try to set for all subevents
  536. //
  537. if(dwSubEvent == EM_ALLSUBEVENTS)
  538. {
  539. break;
  540. }
  541. }
  542. if( AE_LASTITEM < dwSubEvent )
  543. {
  544. bValid = FALSE;
  545. }
  546. break;
  547. case TE_CALLNOTIFICATION:
  548. //
  549. // Just at the address level
  550. //
  551. if( bCallLevel )
  552. {
  553. bValid = FALSE;
  554. break;
  555. }
  556. //
  557. // Accept at address and call level
  558. // ALLSUBEVENTS
  559. //
  560. if( bAcceptAllSubEvents )
  561. {
  562. //
  563. // We could accept 'ALLSUBEVENTS' flag
  564. // when we try to set for all subevents
  565. //
  566. if(dwSubEvent == EM_ALLSUBEVENTS)
  567. {
  568. break;
  569. }
  570. }
  571. if( CNE_LASTITEM < dwSubEvent )
  572. {
  573. bValid = FALSE;
  574. }
  575. break;
  576. case TE_CALLSTATE:
  577. //
  578. // Accept at address and call level
  579. // ALLSUBEVENTS
  580. //
  581. if( bAcceptAllSubEvents )
  582. {
  583. //
  584. // We could accept 'ALLSUBEVENTS' flag
  585. // when we try to set for all subevents
  586. //
  587. if(dwSubEvent == EM_ALLSUBEVENTS)
  588. {
  589. break;
  590. }
  591. }
  592. if( CS_LASTITEM < dwSubEvent )
  593. {
  594. bValid = FALSE;
  595. }
  596. break;
  597. case TE_CALLMEDIA:
  598. //
  599. // Accept at address and call level
  600. // ALLSUBEVENTS
  601. //
  602. if( bAcceptAllSubEvents )
  603. {
  604. //
  605. // We could accept 'ALLSUBEVENTS' flag
  606. // when we try to set for all subevents
  607. //
  608. if(dwSubEvent == EM_ALLSUBEVENTS)
  609. {
  610. break;
  611. }
  612. }
  613. if( CME_LASTITEM < dwSubEvent )
  614. {
  615. bValid = FALSE;
  616. }
  617. break;
  618. case TE_CALLHUB:
  619. //
  620. // Accept at address and call level
  621. // ALLSUBEVENTS
  622. //
  623. if( bAcceptAllSubEvents )
  624. {
  625. //
  626. // We could accept 'ALLSUBEVENTS' flag
  627. // when we try to set for all subevents
  628. //
  629. if(dwSubEvent == EM_ALLSUBEVENTS)
  630. {
  631. break;
  632. }
  633. }
  634. if( CHE_LASTITEM < dwSubEvent )
  635. {
  636. bValid = FALSE;
  637. }
  638. break;
  639. case TE_CALLINFOCHANGE:
  640. //
  641. // Accept at address and call level
  642. // ALLSUBEVENTS
  643. //
  644. if( bAcceptAllSubEvents )
  645. {
  646. //
  647. // We could accept 'ALLSUBEVENTS' flag
  648. // when we try to set for all subevents
  649. //
  650. if(dwSubEvent == EM_ALLSUBEVENTS)
  651. {
  652. break;
  653. }
  654. }
  655. if( CIC_LASTITEM < dwSubEvent )
  656. {
  657. bValid = FALSE;
  658. }
  659. break;
  660. case TE_QOSEVENT:
  661. //
  662. // Accept at address and call level
  663. // ALLSUBEVENTS
  664. //
  665. if( bAcceptAllSubEvents )
  666. {
  667. //
  668. // We could accept 'ALLSUBEVENTS' flag
  669. // when we try to set for all subevents
  670. //
  671. if(dwSubEvent == EM_ALLSUBEVENTS)
  672. {
  673. break;
  674. }
  675. }
  676. if( QE_LASTITEM < dwSubEvent )
  677. {
  678. bValid = FALSE;
  679. }
  680. break;
  681. case TE_FILETERMINAL:
  682. //
  683. // Accept at address and call level
  684. // ALLSUBEVENTS
  685. //
  686. if( bAcceptAllSubEvents )
  687. {
  688. //
  689. // We could accept 'ALLSUBEVENTS' flag
  690. // when we try to set for all subevents
  691. //
  692. if(dwSubEvent == EM_ALLSUBEVENTS)
  693. {
  694. break;
  695. }
  696. }
  697. if( TMS_LASTITEM < dwSubEvent )
  698. {
  699. bValid = FALSE;
  700. }
  701. break;
  702. case TE_PRIVATE:
  703. case TE_ADDRESSDEVSPECIFIC:
  704. case TE_PHONEDEVSPECIFIC:
  705. //
  706. // Accept at address and call level
  707. // ALLSUBEVENTS
  708. //
  709. if( bAcceptAllSubEvents )
  710. {
  711. //
  712. // We could accept 'ALLSUBEVENTS' flag
  713. // when we try to set for all subevents
  714. //
  715. if(dwSubEvent == EM_ALLSUBEVENTS)
  716. {
  717. break;
  718. }
  719. }
  720. else
  721. {
  722. // we accept just all subevents because
  723. // we don't have subevents
  724. bValid = TRUE;
  725. break;
  726. }
  727. break;
  728. default:
  729. //
  730. // Invalid event type
  731. //
  732. bValid = FALSE;
  733. break;
  734. }
  735. LOG((TL_TRACE, "IsSubEventValid - exit %d", bValid));
  736. return bValid;
  737. }
  738. HRESULT CEventMasks::CopyEventMasks(
  739. CEventMasks* pEventMasks
  740. )
  741. {
  742. //
  743. // Enter critical section
  744. //
  745. //m_cs.Lock();
  746. pEventMasks->m_dwTapiObjectMask = m_dwTapiObjectMask;
  747. pEventMasks->m_dwAddressMask = m_dwAddressMask;
  748. pEventMasks->m_dwCallNotificationMask = m_dwCallNotificationMask;
  749. pEventMasks->m_dwCallStateMask = m_dwCallStateMask;
  750. pEventMasks->m_dwCallMediaMask = m_dwCallMediaMask;
  751. pEventMasks->m_dwCallHubMask = m_dwCallHubMask;
  752. pEventMasks->m_dwCallInfoChangeMask = m_dwCallInfoChangeMask;
  753. pEventMasks->m_dwQOSEventMask = m_dwQOSEventMask;
  754. pEventMasks->m_dwFileTerminalMask = m_dwFileTerminalMask;
  755. pEventMasks->m_dwPrivateMask = m_dwPrivateMask;
  756. pEventMasks->m_dwAddressDevSpecificMask = m_dwAddressDevSpecificMask;
  757. pEventMasks->m_dwPhoneDevSpecificMask = m_dwPhoneDevSpecificMask;
  758. //
  759. // Leave critical section
  760. //
  761. //m_cs.Unlock();
  762. return S_OK;
  763. }
  764. ULONG64 CEventMasks::GetTapiSrvEventMask(
  765. IN BOOL bCallLevel
  766. )
  767. {
  768. //
  769. // Convert flags to server side 64 bit masks
  770. //
  771. ULONG64 ulEventMask = EM_LINE_CALLSTATE | // TE_CALLSTATE
  772. EM_LINE_APPNEWCALL | // TE_CALLNOTIFICATION
  773. EM_PHONE_CLOSE | // TE_PHONEEVENT
  774. EM_PHONE_STATE | // TE_PHONEEVENT
  775. EM_PHONE_BUTTONMODE | // TE_PHONEEVENT
  776. EM_PHONE_BUTTONSTATE | // TE_PHONEVENT
  777. EM_LINE_APPNEWCALLHUB | // TE_CALLHUB
  778. EM_LINE_CALLHUBCLOSE | // TE_CALLHUB
  779. EM_LINE_CALLINFO; // TE_CALLINFOCHAnGE
  780. BOOL bEnable = FALSE;
  781. if( !bCallLevel)
  782. {
  783. //
  784. // These flags will be not read if we
  785. // ask for call level
  786. //
  787. // TE_TAPIOBJECT - TE_ADDRESSCREATE
  788. // At address level
  789. GetSubEventFlag(TE_TAPIOBJECT, TE_ADDRESSCREATE, &bEnable);
  790. if( bEnable )
  791. {
  792. ulEventMask |= EM_LINE_CREATE;
  793. }
  794. // TE_TAPIOBJECT - TE_ADDRESSREMOVE
  795. // At address level
  796. GetSubEventFlag(TE_TAPIOBJECT, TE_ADDRESSREMOVE, &bEnable);
  797. if( bEnable )
  798. {
  799. ulEventMask |= EM_LINE_REMOVE;
  800. }
  801. // TE_TAPIOBJECT - TE_ADDRESSCLOSE
  802. // At address level
  803. GetSubEventFlag(TE_TAPIOBJECT, TE_ADDRESSCLOSE, &bEnable);
  804. if( bEnable )
  805. {
  806. ulEventMask |= EM_LINE_CLOSE;
  807. }
  808. // AE_NEWTERMINAL : ignore private MSP events
  809. // AE_REMOVETERMINAL : ignore private MSP events
  810. //
  811. // We read also the flags for call level,
  812. // so go ahead.
  813. }
  814. // TE_CALLNOTIFICATION - CNE_OWNER
  815. // At call level
  816. // We pass EM_LINE_APPNEWCALL for TE_CALLNOTIFICATION everytime because
  817. // we need this events for internal TAPI3 status
  818. // See ulEventMask declaration
  819. // TE_CALLSTATE - All the subevents
  820. // At call level
  821. // We always pass EM_LINE_CALLSTATE for TE_CALLSTATE because
  822. // we need this event for internal TAPI3 status
  823. // See ulEventMask declaration
  824. // TE_CALLHUB - CHE_CALLHUBNEW
  825. // At call level
  826. // We always pass EM_LINE_APPNEWCALLHUB, EM_LINE_CALLHUBCLOSE
  827. // for TE_CALLHUB because
  828. // we need this event for internal TAPI3 status
  829. // See ulEventMask declaration
  830. // TE_CALLINFOCHANGE - All the subevents
  831. // At call level
  832. // We always pass EM_LINE_CALLINFO for TE_CALLINFOChANGE because
  833. // we need this event for internal TAPI3 status
  834. // See ulEventMask declaration
  835. // TE_QOSEVENT - All the subevents
  836. // At call level
  837. if ( m_dwQOSEventMask )
  838. {
  839. ulEventMask |= EM_LINE_QOSINFO;
  840. }
  841. if ( m_dwAddressDevSpecificMask )
  842. {
  843. ulEventMask |= EM_LINE_DEVSPECIFIC | EM_LINE_DEVSPECIFICEX;
  844. }
  845. if ( m_dwPhoneDevSpecificMask )
  846. {
  847. ulEventMask |= EM_PHONE_DEVSPECIFIC;
  848. }
  849. // TE_FILTETERMIAL it is an MSP stuff
  850. // TE_PRIVATE it is an SMP stuff
  851. return ulEventMask;
  852. }
  853. DWORD CEventMasks::GetTapiSrvLineStateMask()
  854. {
  855. //
  856. // These flags shuld be read just
  857. // at address level
  858. //
  859. DWORD dwLineDevStateSubMasks = 0;
  860. BOOL bEnable = FALSE;
  861. // TE_ADDRESS - AE_STATE
  862. // At address level
  863. GetSubEventFlag(TE_ADDRESS, AE_STATE, &bEnable);
  864. if( bEnable )
  865. {
  866. dwLineDevStateSubMasks |= LINEDEVSTATE_CONNECTED |
  867. LINEDEVSTATE_INSERVICE |
  868. LINEDEVSTATE_OUTOFSERVICE |
  869. LINEDEVSTATE_MAINTENANCE |
  870. LINEDEVSTATE_REMOVED |
  871. LINEDEVSTATE_DISCONNECTED |
  872. LINEDEVSTATE_LOCK |
  873. LINEDEVSTATE_MSGWAITON |
  874. LINEDEVSTATE_MSGWAITOFF ;
  875. }
  876. // TE_ADDRESS - AE_CAPSCHANGE
  877. // At address level
  878. GetSubEventFlag(TE_ADDRESS, AE_CAPSCHANGE, &bEnable);
  879. if( bEnable )
  880. {
  881. dwLineDevStateSubMasks |= LINEDEVSTATE_CAPSCHANGE;
  882. }
  883. // TE_ADDRESS - AE_RINGING
  884. // At address level
  885. GetSubEventFlag(TE_ADDRESS, AE_RINGING, &bEnable);
  886. if( bEnable )
  887. {
  888. dwLineDevStateSubMasks |= LINEDEVSTATE_RINGING;
  889. }
  890. // TE_ADDRESS - AE_CONFIGCHANGE
  891. // At address level
  892. GetSubEventFlag(TE_ADDRESS, AE_CONFIGCHANGE, &bEnable);
  893. if( bEnable )
  894. {
  895. dwLineDevStateSubMasks |= LINEDEVSTATE_CONFIGCHANGE;
  896. }
  897. return dwLineDevStateSubMasks;
  898. }
  899. DWORD CEventMasks::GetTapiSrvAddrStateMask()
  900. {
  901. //
  902. // These flags should be read just
  903. // at address level
  904. //
  905. DWORD dwAddrStateSubMasks = 0;
  906. BOOL bEnable = FALSE;
  907. // TE_ADDRESS - AE_CAPSCHANGE
  908. // Address level
  909. GetSubEventFlag(TE_ADDRESS, AE_CAPSCHANGE, &bEnable);
  910. if( bEnable )
  911. {
  912. dwAddrStateSubMasks |= LINEADDRESSSTATE_CAPSCHANGE |
  913. LINEDEVSTATE_CAPSCHANGE;
  914. }
  915. // TE_ADDRESS - AE_FORWARD
  916. // Address level
  917. GetSubEventFlag(TE_ADDRESS, AE_FORWARD, &bEnable);
  918. if( bEnable )
  919. {
  920. dwAddrStateSubMasks |= LINEADDRESSSTATE_FORWARD;
  921. }
  922. return dwAddrStateSubMasks;
  923. }
  924. /*++
  925. SetTapiSrvAddressEventMask
  926. Set the event mask to the TapiSrv level for an address
  927. It's called immediatly after we open a line
  928. --*/
  929. HRESULT CEventMasks::SetTapiSrvAddressEventMask(
  930. IN HLINE hLine
  931. )
  932. {
  933. LOG((TL_TRACE, "SetTapiSrvAddressEventMask - Enter"));
  934. HRESULT hr = E_FAIL;
  935. //
  936. // Get the TapiSrvEvent masks
  937. //
  938. ULONG64 ulEventMasks = 0;
  939. DWORD dwLineDevStateSubMasks = 0;
  940. DWORD dwAddrStateSubMasks = 0;
  941. ulEventMasks = GetTapiSrvEventMask(
  942. FALSE // Address level
  943. );
  944. LOG((TL_INFO, "GetTapiSrvEventMask returns %x",
  945. ulEventMasks));
  946. dwLineDevStateSubMasks = GetTapiSrvLineStateMask();
  947. LOG((TL_INFO, "GetTapiSrvLineStateMask returns %x",
  948. dwLineDevStateSubMasks));
  949. dwAddrStateSubMasks = GetTapiSrvAddrStateMask();
  950. LOG((TL_INFO, "GetTapiSrvAddrStateMask returns %x",
  951. dwAddrStateSubMasks));
  952. hr = tapiSetEventFilterMasks (
  953. TAPIOBJ_HLINE,
  954. hLine,
  955. ulEventMasks
  956. );
  957. if (hr == 0)
  958. {
  959. hr = tapiSetEventFilterSubMasks (
  960. TAPIOBJ_HLINE,
  961. hLine,
  962. EM_LINE_LINEDEVSTATE,
  963. dwLineDevStateSubMasks
  964. );
  965. }
  966. if (hr == 0)
  967. {
  968. hr = tapiSetEventFilterSubMasks (
  969. TAPIOBJ_HLINE,
  970. hLine,
  971. EM_LINE_ADDRESSSTATE,
  972. dwAddrStateSubMasks
  973. );
  974. }
  975. if (hr != 0)
  976. {
  977. hr = mapTAPIErrorCode(hr);
  978. LOG((TL_ERROR,
  979. "CEventMasks::SetTapiSrvAddressEventMask - failed TapiSrv 0x%08x", hr));
  980. // Leave critical section
  981. return hr;
  982. }
  983. LOG((TL_TRACE, "SetTapiSrvAddressEventMask - Exit 0x%08x", hr));
  984. return hr;
  985. }
  986. HRESULT CEventMasks::SetTapiSrvCallEventMask(
  987. IN HCALL hCall
  988. )
  989. {
  990. LOG((TL_TRACE, "SetTapiSrvCallEventMask - Enter. call handle[%lx]", hCall));
  991. HRESULT hr = E_FAIL;
  992. //
  993. // Get the TapiSrvEvent masks
  994. //
  995. ULONG64 ulEventMasks = 0;
  996. ulEventMasks = GetTapiSrvEventMask(
  997. TRUE // Call level
  998. );
  999. LOG((TL_INFO, "GetTapiSrvEventMask returns %x",
  1000. ulEventMasks));
  1001. hr = tapiSetEventFilterMasks (
  1002. TAPIOBJ_HCALL,
  1003. hCall,
  1004. ulEventMasks
  1005. );
  1006. if (hr != 0)
  1007. {
  1008. LOG((TL_ERROR,
  1009. "CEventMasks::SetTapiSrvCallEventMaskr - TapiSrv failed 0x%08x", hr));
  1010. return hr;
  1011. }
  1012. LOG((TL_TRACE, "SetTapiSrvCallEventMask - Exit 0x%08x", hr));
  1013. return hr;
  1014. }
  1015. //
  1016. // ITAddress methods
  1017. //
  1018. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  1019. //
  1020. // get_State
  1021. //
  1022. // retries the current ADDRESS_STATE of the address
  1023. //
  1024. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  1025. HRESULT
  1026. STDMETHODCALLTYPE
  1027. CAddress::get_State(
  1028. ADDRESS_STATE * pAddressState
  1029. )
  1030. {
  1031. HRESULT hr = S_OK;
  1032. LOG((TL_TRACE, "get_State enter" ));
  1033. if (TAPIIsBadWritePtr( pAddressState, sizeof( ADDRESS_STATE ) ) )
  1034. {
  1035. LOG((TL_ERROR, "get_State - bad pointer"));
  1036. return E_POINTER;
  1037. }
  1038. Lock();
  1039. *pAddressState = m_AddressState;
  1040. Unlock();
  1041. LOG((TL_TRACE, "get_State exit - return %lx", hr));
  1042. return hr;
  1043. }
  1044. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1045. //
  1046. // GetAddressName
  1047. // Copy address name to ppName
  1048. // The application must free the name returned through sysfreestirng.
  1049. //
  1050. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1051. HRESULT
  1052. STDMETHODCALLTYPE
  1053. CAddress::get_AddressName(
  1054. BSTR * ppName
  1055. )
  1056. {
  1057. HRESULT hr = S_OK;
  1058. LOG((TL_TRACE, "get_AddressName enter" ));
  1059. LOG((TL_TRACE, " ppName --------->%p", ppName ));
  1060. if ( TAPIIsBadWritePtr( ppName, sizeof( BSTR ) ) )
  1061. {
  1062. LOG((TL_ERROR, "get_AddressName - bad pointer"));
  1063. return E_POINTER;
  1064. }
  1065. Lock();
  1066. if ( m_dwAddressFlags & ADDRESSFLAG_DEVCAPSCHANGE )
  1067. {
  1068. hr = UpdateLineDevCaps();
  1069. if ( SUCCEEDED(hr) )
  1070. {
  1071. hr = SaveAddressName( m_pDevCaps );
  1072. }
  1073. m_dwAddressFlags &= ~ADDRESSFLAG_DEVCAPSCHANGE;
  1074. }
  1075. *ppName = SysAllocString(
  1076. m_szAddressName
  1077. );
  1078. Unlock();
  1079. if (NULL == *ppName)
  1080. {
  1081. LOG((TL_ERROR, "get_AddressName exit - return E_OUTOFMEMORY"));
  1082. return E_OUTOFMEMORY;
  1083. }
  1084. LOG((TL_TRACE, "get_AddressName exit - return %lx", hr ));
  1085. return hr;
  1086. }
  1087. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1088. //
  1089. // get_ServiceProviderName
  1090. // Get the name of the service provider that owns this line/
  1091. // address.
  1092. // Note that we don't get the name on startup - we only get
  1093. // it if it is requested. This shouldn't be used too often.
  1094. //
  1095. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1096. HRESULT
  1097. STDMETHODCALLTYPE
  1098. CAddress::get_ServiceProviderName(
  1099. BSTR * ppName
  1100. )
  1101. {
  1102. HRESULT hr = S_OK;
  1103. LPVARSTRING pVarString = NULL;
  1104. DWORD dwProviderID;
  1105. LPLINEPROVIDERLIST pProviderList = NULL;
  1106. LOG((TL_TRACE, "get_ServiceProviderName enter" ));
  1107. if (TAPIIsBadWritePtr( ppName, sizeof( BSTR ) ) )
  1108. {
  1109. LOG((TL_ERROR, "get_ServiceProviderName - bad pointer"));
  1110. return E_POINTER;
  1111. }
  1112. Lock();
  1113. //
  1114. // if we have already determined the sp name,
  1115. // just return it.
  1116. //
  1117. if (NULL != m_szProviderName)
  1118. {
  1119. *ppName = SysAllocString( m_szProviderName );
  1120. Unlock();
  1121. if (NULL == *ppName)
  1122. {
  1123. LOG((TL_ERROR, "get_ServiceProviderName - alloc ppName failed" ));
  1124. return E_OUTOFMEMORY;
  1125. }
  1126. LOG((TL_TRACE, "get_ServiceProviderName - exit - return SUCCESS" ));
  1127. return S_OK;
  1128. }
  1129. Unlock();
  1130. //
  1131. // get the provider list
  1132. //
  1133. hr = LineGetProviderList(
  1134. &pProviderList
  1135. );
  1136. if (S_OK != hr)
  1137. {
  1138. if (NULL != pProviderList )
  1139. {
  1140. ClientFree( pProviderList );
  1141. }
  1142. LOG((TL_ERROR, "get_ServiceProvideName - LineGetProviderList returned %lx", hr ));
  1143. return hr;
  1144. }
  1145. LPLINEPROVIDERENTRY pProvEntry;
  1146. PWSTR pszProviderName;
  1147. DWORD dwCount;
  1148. hr = S_OK;
  1149. pProvEntry = (LPLINEPROVIDERENTRY)( ( (LPBYTE) pProviderList ) + pProviderList->dwProviderListOffset );
  1150. Lock();
  1151. //
  1152. // make sure the name was not set by another thread while we were waiting
  1153. // for the lock. if it was, do nothing.
  1154. //
  1155. if (NULL == m_szProviderName)
  1156. {
  1157. //
  1158. // search through the list for the provider id, until done with all
  1159. // providers or encounter an error
  1160. //
  1161. for ( dwCount = 0; dwCount < pProviderList->dwNumProviders; dwCount++ )
  1162. {
  1163. if (pProvEntry->dwPermanentProviderID == m_dwProviderID)
  1164. {
  1165. //
  1166. // it's a match! copy the name, if we can.
  1167. //
  1168. m_szProviderName = (PWSTR)ClientAlloc( pProvEntry->dwProviderFilenameSize + sizeof(WCHAR) );
  1169. if (NULL == m_szProviderName)
  1170. {
  1171. LOG((TL_ERROR, "get_ServiceProviderName - alloc m_szProviderName failed" ));
  1172. hr = E_OUTOFMEMORY;
  1173. }
  1174. else
  1175. {
  1176. //
  1177. // save it in the object
  1178. //
  1179. memcpy(
  1180. m_szProviderName,
  1181. ((LPBYTE)pProviderList) + pProvEntry->dwProviderFilenameOffset,
  1182. pProvEntry->dwProviderFilenameSize
  1183. );
  1184. }
  1185. //
  1186. // found a match. m_szProviderName contains the string, or hr
  1187. // contains the error if failed.
  1188. //
  1189. break;
  1190. }
  1191. pProvEntry++;
  1192. }
  1193. //
  1194. // did we find it?
  1195. //
  1196. if (dwCount == pProviderList->dwNumProviders)
  1197. {
  1198. LOG((TL_ERROR, "get_ServiceProviderName - could not find provider in list" ));
  1199. //
  1200. // if we looped through the whole list but did not find a match, hr
  1201. // should still be S_OK
  1202. //
  1203. // assert to protect from future code changes erroneously
  1204. // overwritingthe error code.
  1205. //
  1206. _ASSERTE(SUCCEEDED(hr));
  1207. //
  1208. // couldn't find the provider.
  1209. //
  1210. hr = TAPI_E_NODRIVER;
  1211. }
  1212. } // if (NULL == m_szProviderName)
  1213. //
  1214. // if hr does not contain an error, the m_szProviderName must contain a
  1215. // valid string
  1216. //
  1217. if (SUCCEEDED(hr))
  1218. {
  1219. _ASSERTE( NULL != m_szProviderName );
  1220. *ppName = SysAllocString( m_szProviderName );
  1221. if (NULL == *ppName)
  1222. {
  1223. //
  1224. // could not allocated string
  1225. //
  1226. LOG((TL_ERROR,
  1227. "get_ServiceProviderName - failed to allocate memory for provider name string" ));
  1228. hr = E_OUTOFMEMORY;
  1229. }
  1230. }
  1231. Unlock();
  1232. ClientFree( pProviderList );
  1233. LOG((TL_TRACE, "get_ServiceProviderName exit - return %lx", hr));
  1234. return hr;
  1235. }
  1236. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1237. //
  1238. // get_TAPIObject
  1239. //
  1240. // return the owning tapi object
  1241. //
  1242. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1243. HRESULT
  1244. STDMETHODCALLTYPE
  1245. CAddress::get_TAPIObject(
  1246. ITTAPI ** ppTapiObject
  1247. )
  1248. {
  1249. HRESULT hr = S_OK;
  1250. LOG((TL_TRACE, "get_TAPIObject enter" ));
  1251. if (TAPIIsBadWritePtr( ppTapiObject, sizeof( ITTAPI * ) ) )
  1252. {
  1253. LOG((TL_ERROR, "get_TAPIObject - bad pointer"));
  1254. return E_POINTER;
  1255. }
  1256. *ppTapiObject = m_pTAPI;
  1257. m_pTAPI->AddRef();
  1258. LOG((TL_TRACE, "get_TAPIObject exit - return %lx", hr ));
  1259. return hr;
  1260. }
  1261. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1262. //
  1263. // GetTapi
  1264. //
  1265. // private method to get the tapi object
  1266. //
  1267. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1268. CTAPI *
  1269. CAddress::GetTapi(BOOL bAddRefTapi)
  1270. {
  1271. CTAPI * p = NULL;
  1272. Lock();
  1273. //
  1274. // get a pointer to tapi object
  1275. //
  1276. if( m_pTAPI != NULL )
  1277. {
  1278. p = dynamic_cast<CTAPI *>(m_pTAPI);
  1279. }
  1280. //
  1281. // addref tapi if needed and if possible
  1282. //
  1283. if ( (NULL != p) && bAddRefTapi)
  1284. {
  1285. //
  1286. // we have an object and we need to addref it before returing
  1287. //
  1288. p->AddRef();
  1289. }
  1290. Unlock();
  1291. LOG((TL_TRACE, "GetTapi - returning [%p]", p));
  1292. return p;
  1293. }
  1294. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1295. //
  1296. // InternalCreateCall
  1297. //
  1298. // called by a couple of places to create a call.
  1299. //
  1300. // pszDestAddress
  1301. // destaddress of the call
  1302. //
  1303. // cp
  1304. // current call privilege
  1305. //
  1306. // hCall
  1307. // tapi2 call handle
  1308. //
  1309. // bExpose
  1310. // is this a hidden call?
  1311. //
  1312. // ppCall
  1313. // return call object here
  1314. //
  1315. //
  1316. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1317. HRESULT
  1318. CAddress::InternalCreateCall(
  1319. PWSTR pszDestAddress,
  1320. long lAddressType,
  1321. long lMediaType,
  1322. CALL_PRIVILEGE cp,
  1323. BOOL bNeedToNotify,
  1324. HCALL hCall,
  1325. BOOL bExpose,
  1326. CCall ** ppCall
  1327. )
  1328. {
  1329. HRESULT hr;
  1330. LOG((TL_TRACE, "InternalCreateCall enter" ));
  1331. //
  1332. // create & initialize a new call object
  1333. //
  1334. CComObject<CCall> * pCall;
  1335. hr = CComObject<CCall>::CreateInstance( &pCall );
  1336. if (NULL == pCall)
  1337. {
  1338. LOG((TL_ERROR, "InternalCreateCall - could not create call instance" ));
  1339. return E_OUTOFMEMORY;
  1340. }
  1341. //
  1342. // initialize the call object.
  1343. //
  1344. // note: getting address lock while initializing a
  1345. // call object can lead to deadlocks
  1346. //
  1347. hr = pCall->Initialize(
  1348. this,
  1349. pszDestAddress,
  1350. lAddressType,
  1351. lMediaType,
  1352. cp,
  1353. bNeedToNotify,
  1354. bExpose,
  1355. hCall,
  1356. &m_EventMasks
  1357. );
  1358. if (S_OK != hr)
  1359. {
  1360. delete pCall;
  1361. LOG((TL_ERROR, "InternalCreateCall failed - Call object failed init - %lx", hr ));
  1362. return hr;
  1363. }
  1364. //
  1365. // serialize call creation with call cleanup (which happens when tapi tells
  1366. // us it is being shutdown).
  1367. //
  1368. Lock();
  1369. //
  1370. // tapi object already signaled shutdown?
  1371. //
  1372. if (m_bTapiSignaledShutdown)
  1373. {
  1374. Unlock();
  1375. //
  1376. // tapi object already notified us of its shutdown and we already did
  1377. // cleanup. no need to create any new calls
  1378. //
  1379. LOG((TL_ERROR,
  1380. "InternalCreateCall - tapi object shut down. cannot create call" ));
  1381. //
  1382. // we have no use for the call object
  1383. //
  1384. pCall->Release();
  1385. if (bNeedToNotify)
  1386. {
  1387. //
  1388. // compensate for the extra refcount added if bNeedToNotify is on
  1389. //
  1390. pCall->Release();
  1391. }
  1392. pCall = NULL;
  1393. return TAPI_E_WRONG_STATE;
  1394. }
  1395. if (bExpose)
  1396. {
  1397. ITCallInfo * pITCallInfo;
  1398. pITCallInfo = dynamic_cast<ITCallInfo *>(pCall);
  1399. //
  1400. // save the call in this address's list
  1401. //
  1402. hr = AddCall( pITCallInfo );
  1403. }
  1404. //
  1405. // the call was created and initialized, it is now safe for call cleanup
  1406. // logic to kick in if it was waiting for the lock
  1407. //
  1408. Unlock();
  1409. //
  1410. // return it
  1411. //
  1412. *ppCall = pCall;
  1413. LOG((TL_TRACE, "InternalCreateCall exit - return S_OK" ));
  1414. return S_OK;
  1415. }
  1416. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1417. //
  1418. // CreateCall
  1419. //
  1420. // An application calls this method to create an outgoing call object
  1421. //
  1422. // lpszDestAddress
  1423. // DestAddress of the call
  1424. //
  1425. // ppCall
  1426. // return call object here
  1427. //
  1428. //
  1429. // returns
  1430. //
  1431. //
  1432. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1433. HRESULT
  1434. STDMETHODCALLTYPE
  1435. CAddress::CreateCall(
  1436. BSTR lpszDestAddress,
  1437. long lAddressType,
  1438. long lMediaType,
  1439. ITBasicCallControl ** ppCall
  1440. )
  1441. {
  1442. HRESULT hr;
  1443. CCall * pCall;
  1444. LOG((TL_TRACE, "CreateCall enter" ));
  1445. if (TAPIIsBadWritePtr( ppCall, sizeof(ITBasicCallControl *) ) )
  1446. {
  1447. LOG((TL_ERROR, "CreateCall - bad pointer"));
  1448. return E_POINTER;
  1449. }
  1450. if ((lpszDestAddress != NULL) && IsBadStringPtrW( lpszDestAddress, -1 )) // NULL is ok
  1451. {
  1452. LOG((TL_ERROR, "CreateCall - bad string"));
  1453. return E_POINTER;
  1454. }
  1455. //
  1456. // use internal function
  1457. //
  1458. hr = InternalCreateCall(
  1459. lpszDestAddress,
  1460. lAddressType,
  1461. lMediaType,
  1462. CP_OWNER,
  1463. FALSE,
  1464. NULL,
  1465. TRUE,
  1466. &pCall
  1467. );
  1468. if (S_OK != hr)
  1469. {
  1470. LOG((TL_ERROR, "CreateCall - InternalCreateCall failed hr = %lx", hr));
  1471. return hr;
  1472. }
  1473. //
  1474. // put result in out pointer - also
  1475. // increments the ref count
  1476. //
  1477. hr = pCall->QueryInterface(
  1478. IID_ITBasicCallControl,
  1479. (void **) ppCall
  1480. );
  1481. //
  1482. // internalcreatecall has a reference - don't keep it
  1483. //
  1484. pCall->Release();
  1485. if (S_OK != hr)
  1486. {
  1487. LOG((TL_ERROR, "CreateCall - saving call failed" ));
  1488. return hr;
  1489. }
  1490. LOG((TL_TRACE, "CreateCall exit - return SUCCESS" ));
  1491. return S_OK;
  1492. }
  1493. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1494. //
  1495. // get_Calls
  1496. // return a collection of calls that this address currently
  1497. // owns
  1498. //
  1499. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1500. HRESULT
  1501. STDMETHODCALLTYPE
  1502. CAddress::get_Calls(
  1503. VARIANT * pVariant
  1504. )
  1505. {
  1506. HRESULT hr;
  1507. IDispatch * pDisp;
  1508. LOG((TL_TRACE, "get_Calls enter" ));
  1509. if (TAPIIsBadWritePtr( pVariant, sizeof(VARIANT) ) )
  1510. {
  1511. LOG((TL_ERROR, "get_Calls - bad pointer"));
  1512. return E_POINTER;
  1513. }
  1514. CComObject< CTapiCollection< ITCallInfo > > * p;
  1515. CComObject< CTapiCollection< ITCallInfo > >::CreateInstance( &p );
  1516. if (NULL == p)
  1517. {
  1518. LOG((TL_ERROR, "get_Calls - could not create collection" ));
  1519. return E_OUTOFMEMORY;
  1520. }
  1521. Lock();
  1522. // initialize
  1523. hr = p->Initialize( m_CallArray );
  1524. Unlock();
  1525. if (S_OK != hr)
  1526. {
  1527. LOG((TL_ERROR, "get_Calls - could not initialize collection" ));
  1528. delete p;
  1529. return hr;
  1530. }
  1531. // get the IDispatch interface
  1532. hr = p->_InternalQueryInterface( IID_IDispatch, (void **) &pDisp );
  1533. if (S_OK != hr)
  1534. {
  1535. LOG((TL_ERROR, "get_Calls - could not get IDispatch interface" ));
  1536. delete p;
  1537. return hr;
  1538. }
  1539. // put it in the variant
  1540. VariantInit(pVariant);
  1541. pVariant->vt = VT_DISPATCH;
  1542. pVariant->pdispVal = pDisp;
  1543. LOG((TL_TRACE, "get_Calls - exit - return %lx", hr ));
  1544. return hr;
  1545. }
  1546. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1547. //
  1548. // EnumerateCalls
  1549. // returns an enumeration of calls the this address currently owns
  1550. //
  1551. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1552. HRESULT
  1553. STDMETHODCALLTYPE
  1554. CAddress::EnumerateCalls(
  1555. IEnumCall ** ppCallEnum
  1556. )
  1557. {
  1558. HRESULT hr = S_OK;
  1559. LOG((TL_TRACE, "EnumerateCalls enter" ));
  1560. LOG((TL_TRACE, " ppCallEnum----->%p", ppCallEnum ));
  1561. if (TAPIIsBadWritePtr(ppCallEnum, sizeof( ppCallEnum ) ) )
  1562. {
  1563. LOG((TL_ERROR, "EnumerateCalls - bad pointer"));
  1564. return E_POINTER;
  1565. }
  1566. //
  1567. // create the enumerator
  1568. //
  1569. CComObject< CTapiEnum< IEnumCall, ITCallInfo, &IID_IEnumCall > > * p;
  1570. hr = CComObject< CTapiEnum< IEnumCall, ITCallInfo, &IID_IEnumCall > >
  1571. ::CreateInstance( &p );
  1572. if (S_OK != hr)
  1573. {
  1574. LOG((TL_ERROR, "EnumerateCalls - could not create enum" ));
  1575. return hr;
  1576. }
  1577. Lock();
  1578. //
  1579. // initialize it with our call list
  1580. //
  1581. p->Initialize( m_CallArray );
  1582. Unlock();
  1583. //
  1584. // return it
  1585. //
  1586. *ppCallEnum = p;
  1587. LOG((TL_TRACE, "EnumerateCalls exit - return %lx", hr ));
  1588. return hr;
  1589. }
  1590. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1591. //
  1592. // FinalRelease
  1593. // Clean up anything in the address object.
  1594. //
  1595. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1596. void CAddress::FinalRelease()
  1597. {
  1598. CTAPI * pCTapi;
  1599. LOG((TL_TRACE, "FinalRelease[%p] - enter", this));
  1600. Lock();
  1601. // remove from the handle hash table, so any more messages
  1602. // from msp callback events are ignored
  1603. //
  1604. RemoveHandleFromHashTable(m_MSPContext);
  1605. m_MSPContext = NULL;
  1606. pCTapi = GetTapi();
  1607. if( NULL == pCTapi )
  1608. {
  1609. LOG((TL_ERROR, "dynamic cast operation failed"));
  1610. }
  1611. else
  1612. {
  1613. pCTapi->InvalidateBuffer( BUFFERTYPE_LINEDEVCAP, (UINT_PTR)this );
  1614. pCTapi->InvalidateBuffer( BUFFERTYPE_ADDRCAP, (UINT_PTR)this );
  1615. }
  1616. if ( NULL != m_szAddress )
  1617. {
  1618. ClientFree( m_szAddress);
  1619. m_szAddress = NULL;
  1620. }
  1621. if ( NULL != m_szAddressName)
  1622. {
  1623. LOG((TL_INFO, "FinalRelease - m_szAddressName is [%ls]", m_szAddressName ));
  1624. ClientFree( m_szAddressName );
  1625. m_szAddressName = NULL;
  1626. }
  1627. if ( NULL != m_szProviderName )
  1628. {
  1629. ClientFree( m_szProviderName );
  1630. m_szProviderName = NULL;
  1631. }
  1632. //
  1633. // release calls
  1634. //
  1635. m_CallArray.Shutdown();
  1636. //
  1637. // release terminals
  1638. //
  1639. m_TerminalArray.Shutdown();
  1640. //
  1641. // release addresslines
  1642. //
  1643. PtrList::iterator l;
  1644. for (l = m_AddressLinesPtrList.begin(); l != m_AddressLinesPtrList.end(); l++)
  1645. {
  1646. HRESULT hr;
  1647. AddressLineStruct * pLine;
  1648. pLine = (AddressLineStruct *)(*l);
  1649. if (pLine == m_pCallHubTrackingLine)
  1650. {
  1651. // this line is used for call hub tracking
  1652. // make sure we don't try and delete this again below
  1653. m_pCallHubTrackingLine = NULL;
  1654. }
  1655. LineClose(
  1656. &(pLine->t3Line)
  1657. );
  1658. ClientFree (pLine);
  1659. }
  1660. if (m_pCallHubTrackingLine != NULL)
  1661. {
  1662. LineClose( &(m_pCallHubTrackingLine->t3Line) );
  1663. }
  1664. //
  1665. // free the msp
  1666. //
  1667. if ( NULL != m_pMSPAggAddress )
  1668. {
  1669. ITMSPAddress * pMSPAddress = GetMSPAddress();
  1670. pMSPAddress->Shutdown();
  1671. pMSPAddress->Release();
  1672. m_pMSPAggAddress->Release();
  1673. m_pMSPAggAddress = NULL;
  1674. }
  1675. //
  1676. // unregister the msp wait
  1677. // event in the thread pool
  1678. //
  1679. if ( NULL != m_hWaitEvent )
  1680. {
  1681. LOG((TL_TRACE, "FinalRelease - unregistering callback"));
  1682. UnregisterWaitEx( m_hWaitEvent, INVALID_HANDLE_VALUE);
  1683. m_hWaitEvent = NULL;
  1684. }
  1685. //
  1686. // close the msp event
  1687. //
  1688. if ( NULL != m_hMSPEvent )
  1689. {
  1690. CloseHandle(m_hMSPEvent );
  1691. }
  1692. //
  1693. // release the private object
  1694. //
  1695. if ( NULL != m_pPrivate )
  1696. {
  1697. m_pPrivate->Release();
  1698. }
  1699. m_pTAPI->Release();
  1700. m_pTAPI = NULL;
  1701. Unlock();
  1702. LOG((TL_TRACE, "FinalRelease - exit"));
  1703. }
  1704. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1705. //
  1706. // CreateWaveMSPObject
  1707. //
  1708. // special case - creates the wavemsp object
  1709. //
  1710. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1711. HRESULT
  1712. CreateWaveMSPObject(
  1713. IUnknown * pUnk,
  1714. IUnknown ** ppMSPAggAddress
  1715. )
  1716. {
  1717. HRESULT hr = S_OK;
  1718. LOG((TL_TRACE, "CreateWaveMSPObject - enter"));
  1719. hr = CoCreateInstance(
  1720. CLSID_WAVEMSP,
  1721. pUnk,
  1722. CLSCTX_INPROC_SERVER,
  1723. IID_IUnknown,
  1724. (void **) ppMSPAggAddress
  1725. );
  1726. if (!SUCCEEDED(hr))
  1727. {
  1728. LOG((TL_ERROR, "CreateWaveMSPObject failed to CoCreate - %lx", hr));
  1729. return hr;
  1730. }
  1731. LOG((TL_TRACE, "CreateWaveMSPObject - exit"));
  1732. return hr;
  1733. }
  1734. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  1735. //
  1736. // CreateWaveInfo
  1737. //
  1738. // for a wave device address, get the wave device IDs to give
  1739. // to the wavemsp
  1740. //
  1741. // First DWORD = Command Second DWORD Third DWORD
  1742. // 0 Set wave IDs for call WaveIn ID WaveOut ID
  1743. // 1 Start streaming <ignored> <ignored>
  1744. // 2 Stop streaming <ignored> <ignored>
  1745. // 3 Set wave IDs for address WaveIn ID WaveOut ID
  1746. // 4 Wave IDS for address not available
  1747. // 5 stop streaming because a blocking tapi functions was called
  1748. // 6
  1749. // 7
  1750. // 8 full duplex support
  1751. //
  1752. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  1753. HRESULT
  1754. CreateWaveInfo(
  1755. HLINE hLine,
  1756. DWORD dwAddressID,
  1757. HCALL hCall,
  1758. DWORD dwCallSelect,
  1759. BOOL bFullDuplex,
  1760. LPDWORD pdwIDs
  1761. )
  1762. {
  1763. LPVARSTRING pVarString;
  1764. HRESULT hr;
  1765. if ( bFullDuplex )
  1766. {
  1767. hr = LineGetID(
  1768. hLine,
  1769. dwAddressID,
  1770. hCall,
  1771. dwCallSelect,
  1772. &pVarString,
  1773. L"wave/in/out"
  1774. );
  1775. if (!SUCCEEDED(hr))
  1776. {
  1777. if (NULL != pVarString)
  1778. {
  1779. ClientFree( pVarString );
  1780. }
  1781. LOG((TL_ERROR, "LineGetID failed for waveinout device - %lx", hr ));
  1782. return hr;
  1783. }
  1784. pdwIDs[1] = ((LPDWORD)(pVarString+1))[0];
  1785. pdwIDs[2] = ((LPDWORD)(pVarString+1))[1];
  1786. ClientFree( pVarString );
  1787. return S_OK;
  1788. }
  1789. //
  1790. // get the wave/in id
  1791. //
  1792. hr = LineGetID(
  1793. hLine,
  1794. dwAddressID,
  1795. hCall,
  1796. dwCallSelect,
  1797. &pVarString,
  1798. L"wave/in"
  1799. );
  1800. if (!SUCCEEDED(hr))
  1801. {
  1802. if (NULL != pVarString)
  1803. {
  1804. ClientFree( pVarString );
  1805. }
  1806. LOG((TL_ERROR, "LineGetID failed for wavein device - %lx", hr ));
  1807. return hr;
  1808. }
  1809. //
  1810. // saveit
  1811. //
  1812. pdwIDs[1] = *((LPDWORD)(pVarString+1));
  1813. ClientFree( pVarString );
  1814. //
  1815. // get the waveout id
  1816. //
  1817. hr = LineGetID(
  1818. hLine,
  1819. dwAddressID,
  1820. hCall,
  1821. dwCallSelect,
  1822. &pVarString,
  1823. L"wave/out"
  1824. );
  1825. if (!SUCCEEDED(hr))
  1826. {
  1827. if (NULL != pVarString)
  1828. {
  1829. ClientFree( pVarString );
  1830. }
  1831. LOG((TL_ERROR, "LineGetID failed for waveout device - %lx", hr ));
  1832. return hr;
  1833. }
  1834. //
  1835. // save it
  1836. //
  1837. pdwIDs[2] = *((LPDWORD)(pVarString+1));
  1838. ClientFree( pVarString );
  1839. return S_OK;
  1840. }
  1841. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1842. //
  1843. //
  1844. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1845. HRESULT
  1846. CAddress::InitializeWaveDeviceIDs( HLINE hLine )
  1847. {
  1848. DWORD adwIDs[3];
  1849. HRESULT hr;
  1850. hr = CreateWaveInfo(
  1851. hLine,
  1852. GetAddressID(),
  1853. NULL,
  1854. LINECALLSELECT_ADDRESS,
  1855. HasFullDuplexWaveDevice(),
  1856. adwIDs
  1857. );
  1858. if ( SUCCEEDED(hr) )
  1859. {
  1860. adwIDs[0] = 3;
  1861. }
  1862. else
  1863. {
  1864. adwIDs[0] = 4;
  1865. }
  1866. hr = ReceiveTSPData(
  1867. NULL,
  1868. (LPBYTE)adwIDs,
  1869. sizeof(adwIDs)
  1870. );
  1871. if ( TAPI_VERSION3_0 <= m_dwAPIVersion )
  1872. {
  1873. adwIDs[0] = 8;
  1874. if ( m_dwAddressFlags & ADDRESSFLAG_WAVEFULLDUPLEX )
  1875. {
  1876. adwIDs[1] = 1;
  1877. }
  1878. else
  1879. {
  1880. adwIDs[1] = 0;
  1881. }
  1882. hr = ReceiveTSPData(
  1883. NULL,
  1884. (LPBYTE)adwIDs,
  1885. sizeof(adwIDs)
  1886. );
  1887. }
  1888. return hr;
  1889. }
  1890. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1891. //
  1892. // SaveAddressName
  1893. //
  1894. // saves the address name based on the linename in devcaps
  1895. //
  1896. // called in lock
  1897. //
  1898. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  1899. HRESULT
  1900. CAddress::SaveAddressName( LPLINEDEVCAPS pDevCaps )
  1901. {
  1902. PWSTR pAddressName = NULL;
  1903. PWSTR pAddressString = NULL;
  1904. DWORD dwAddressStringSize = 0;
  1905. //
  1906. // save the line name
  1907. //
  1908. if ( pDevCaps->dwNumAddresses > 1 )
  1909. {
  1910. pAddressString = MyLoadString( IDS_ADDRESS );
  1911. if ( NULL == pAddressString )
  1912. {
  1913. return E_UNEXPECTED;
  1914. }
  1915. dwAddressStringSize = (lstrlenW( pAddressString ) + 1) * sizeof(WCHAR);
  1916. //
  1917. // need to add some room for whitespace
  1918. //
  1919. dwAddressStringSize += 10 * sizeof(WCHAR);
  1920. }
  1921. if (0 != pDevCaps->dwLineNameSize)
  1922. {
  1923. DWORD dwSize;
  1924. dwSize = pDevCaps->dwLineNameSize + sizeof(WCHAR);
  1925. dwSize += dwAddressStringSize;
  1926. pAddressName = (PWSTR) ClientAlloc(dwSize);
  1927. if (NULL == pAddressName)
  1928. {
  1929. LOG((TL_ERROR, "Initialize - alloc pAddressName failed" ));
  1930. if ( NULL != pAddressString )
  1931. {
  1932. ClientFree( pAddressString );
  1933. }
  1934. return E_OUTOFMEMORY;
  1935. }
  1936. memcpy(
  1937. pAddressName,
  1938. ((LPBYTE)(pDevCaps)) + pDevCaps->dwLineNameOffset,
  1939. pDevCaps->dwLineNameSize
  1940. );
  1941. }
  1942. else
  1943. {
  1944. PWSTR pTempBuffer;
  1945. DWORD dwSize = 0;
  1946. pTempBuffer = MyLoadString( IDS_LINE );
  1947. if ( NULL == pTempBuffer )
  1948. {
  1949. LOG((TL_ERROR, "Initialize - couldn't load LINE resource"));
  1950. if ( NULL != pAddressString )
  1951. {
  1952. ClientFree( pAddressString );
  1953. }
  1954. return E_UNEXPECTED;
  1955. }
  1956. dwSize = (lstrlenW( pTempBuffer ) + 1) * sizeof(WCHAR);
  1957. dwSize += dwAddressStringSize;
  1958. //
  1959. // need some whitespace
  1960. //
  1961. dwSize += 5 * sizeof(WCHAR);
  1962. pAddressName = (PWSTR) ClientAlloc( dwSize );
  1963. if ( NULL == pAddressName )
  1964. {
  1965. ClientFree( pTempBuffer );
  1966. if ( NULL != pAddressString )
  1967. {
  1968. ClientFree( pAddressString );
  1969. }
  1970. return E_OUTOFMEMORY;
  1971. }
  1972. wsprintfW(
  1973. pAddressName,
  1974. L"%s %d",
  1975. pTempBuffer,
  1976. m_dwDeviceID
  1977. );
  1978. ClientFree( pTempBuffer );
  1979. }
  1980. //
  1981. // append the address # if there is more than one address on the line
  1982. //
  1983. if (pDevCaps->dwNumAddresses > 1)
  1984. {
  1985. wsprintfW(
  1986. pAddressName,
  1987. L"%s - %s %d",
  1988. pAddressName,
  1989. pAddressString,
  1990. m_dwAddressID
  1991. );
  1992. }
  1993. if ( NULL != pAddressString )
  1994. {
  1995. ClientFree( pAddressString );
  1996. }
  1997. if ( NULL != m_szAddressName )
  1998. {
  1999. ClientFree( m_szAddressName);
  2000. }
  2001. m_szAddressName = pAddressName;
  2002. return S_OK;
  2003. }
  2004. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2005. //
  2006. // Initialize
  2007. // Init the CAddress object
  2008. //
  2009. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2010. HRESULT
  2011. CAddress::Initialize(
  2012. ITTAPI * pTapi,
  2013. HLINEAPP hLineApp,
  2014. #ifdef USE_PHONEMSP
  2015. HPHONEAPP hPhoneApp,
  2016. #endif USE_PHONEMSP
  2017. DWORD dwAPIVersion,
  2018. DWORD dwDeviceID,
  2019. DWORD dwAddressID,
  2020. DWORD dwProviderID,
  2021. LPLINEDEVCAPS pDevCaps,
  2022. DWORD dwEventFilterMask
  2023. )
  2024. {
  2025. LONG lResult;
  2026. HRESULT hr;
  2027. T3LINE t3Line;
  2028. #ifdef USE_PHONEMSP
  2029. BOOL bPreferPhoneMSPForProvider = FALSE;
  2030. #endif USE_PHONEMSP
  2031. t3Line.hLine = NULL;
  2032. t3Line.dwAddressLineStructHandle = 0;
  2033. LOG((TL_TRACE, "Initialize[%p] enter", this ));
  2034. LOG((TL_INFO, " hLineApp ---------->%lx", hLineApp ));
  2035. #ifdef USE_PHONEMSP
  2036. LOG((TL_INFO, " hPhoneApp --------->%lx", hPhoneApp ));
  2037. #endif USE_PHONEMSP
  2038. LOG((TL_INFO, " dwAPIVersion ------>%lx", dwAPIVersion ));
  2039. LOG((TL_INFO, " dwDeviceID -------->%lx", dwDeviceID ));
  2040. LOG((TL_INFO, " dwAddressID ------->%lx", dwAddressID ));
  2041. LOG((TL_INFO, " pDevCaps ---------->%p", pDevCaps ));
  2042. Lock();
  2043. //
  2044. // save relevant info
  2045. //
  2046. m_pTAPI = pTapi;
  2047. m_dwDeviceID = dwDeviceID;
  2048. m_dwAddressID = dwAddressID;
  2049. m_dwMediaModesSupported = pDevCaps->dwMediaModes;
  2050. m_hLineApp = hLineApp;
  2051. #ifdef USE_PHONEMSP
  2052. m_hPhoneApp = hPhoneApp;
  2053. #endif USE_PHONEMSP
  2054. m_dwAPIVersion = dwAPIVersion;
  2055. m_AddressState = AS_INSERVICE;
  2056. m_pCallHubTrackingLine = NULL;
  2057. m_dwProviderID = dwProviderID;
  2058. m_pPrivate = NULL;
  2059. m_szAddressName = NULL;
  2060. m_pAddressCaps = NULL;
  2061. m_pDevCaps = NULL;
  2062. //
  2063. // Read the event filter mask from TAPIobject
  2064. //
  2065. SetEventFilterMask( dwEventFilterMask );
  2066. AddRef();
  2067. m_pTAPI->AddRef();
  2068. //
  2069. // default address type
  2070. //
  2071. if (m_dwAPIVersion >= TAPI_VERSION3_0)
  2072. {
  2073. //
  2074. // save msp and address types support
  2075. //
  2076. if ( (pDevCaps->dwDevCapFlags) & LINEDEVCAPFLAGS_MSP )
  2077. {
  2078. m_dwAddressFlags |= ADDRESSFLAG_MSP;
  2079. LOG((TL_INFO, "Initialize - has an msp" ));
  2080. }
  2081. if ( (pDevCaps->dwDevCapFlags) & LINEDEVCAPFLAGS_PRIVATEOBJECTS )
  2082. {
  2083. m_dwAddressFlags |= ADDRESSFLAG_PRIVATEOBJECTS;
  2084. LOG((TL_INFO, "Initialize - has private object" ));
  2085. }
  2086. if ( (pDevCaps->dwDevCapFlags) & LINEDEVCAPFLAGS_CALLHUB )
  2087. {
  2088. m_dwAddressFlags |= ADDRESSFLAG_CALLHUB;
  2089. LOG((TL_INFO, "Initialize - supports callhubs" ));
  2090. }
  2091. if ( (pDevCaps->dwDevCapFlags) & LINEDEVCAPFLAGS_CALLHUBTRACKING )
  2092. {
  2093. m_dwAddressFlags |= ADDRESSFLAG_CALLHUBTRACKING;
  2094. LOG((TL_INFO, "Initialize - supports callhub tracking" ));
  2095. }
  2096. }
  2097. //
  2098. // check for wave device if it doesn't
  2099. // have it's own MSP
  2100. //
  2101. // the sp lets tapi know of wave support through the
  2102. // device classes field, as well
  2103. //
  2104. // so, go through the array
  2105. // and look for the appropriate string
  2106. //
  2107. if ( !(m_dwAddressFlags & ADDRESSFLAG_MSP) && (m_dwAPIVersion >= TAPI_VERSION2_0 ) )
  2108. {
  2109. if (0 != pDevCaps->dwDeviceClassesOffset)
  2110. {
  2111. PWSTR pszDevices;
  2112. //
  2113. // look for full duplex
  2114. // if not full duplex then
  2115. // look for wave out
  2116. // look for wave in
  2117. // if not wave then
  2118. // look for wave
  2119. //
  2120. pszDevices = (PWSTR)( ( (PBYTE)pDevCaps ) + pDevCaps->dwDeviceClassesOffset );
  2121. while (NULL != *pszDevices)
  2122. {
  2123. if (0 == lstrcmpiW(pszDevices, L"wave/in/out"))
  2124. {
  2125. m_dwAddressFlags |= ADDRESSFLAG_WAVEFULLDUPLEX;
  2126. LOG((TL_INFO, "Initialize - supports full duplex wave"));
  2127. break;
  2128. }
  2129. pszDevices += (lstrlenW(pszDevices) + 1 );
  2130. }
  2131. if (!HasWaveDevice())
  2132. {
  2133. pszDevices = (PWSTR)( ( (PBYTE)pDevCaps ) + pDevCaps->dwDeviceClassesOffset );
  2134. //
  2135. // look for wave out
  2136. //
  2137. while (NULL != *pszDevices)
  2138. {
  2139. if (0 == lstrcmpiW(pszDevices, L"wave/out"))
  2140. {
  2141. m_dwAddressFlags |= ADDRESSFLAG_WAVEOUTDEVICE;
  2142. LOG((TL_INFO, "Initialize - supports wave/out device" ));
  2143. break;
  2144. }
  2145. pszDevices += (lstrlenW(pszDevices) +1);
  2146. }
  2147. pszDevices = (PWSTR)( ( (PBYTE)pDevCaps ) + pDevCaps->dwDeviceClassesOffset );
  2148. //
  2149. // look for wave in
  2150. //
  2151. while (NULL != *pszDevices)
  2152. {
  2153. if (0 == lstrcmpiW(pszDevices, L"wave/in"))
  2154. {
  2155. m_dwAddressFlags |= ADDRESSFLAG_WAVEINDEVICE;
  2156. LOG((TL_INFO, "Initialize - supports wave/in device" ));
  2157. break;
  2158. }
  2159. pszDevices += (lstrlenW(pszDevices) +1);
  2160. }
  2161. if (!HasWaveDevice())
  2162. {
  2163. pszDevices = (PWSTR)( ( (PBYTE)pDevCaps ) + pDevCaps->dwDeviceClassesOffset );
  2164. //
  2165. // look for just wave
  2166. // some sps don't differentiate between wave out and wave in
  2167. //
  2168. while (NULL != *pszDevices)
  2169. {
  2170. if (0 == lstrcmpiW(pszDevices, L"wave"))
  2171. {
  2172. m_dwAddressFlags |= (ADDRESSFLAG_WAVEINDEVICE|ADDRESSFLAG_WAVEOUTDEVICE);
  2173. LOG((TL_INFO, "Initialize - supports wave device" ));
  2174. break;
  2175. }
  2176. pszDevices += (lstrlenW(pszDevices) + 1);
  2177. }
  2178. }
  2179. }
  2180. }
  2181. }
  2182. IUnknown * pUnk;
  2183. _InternalQueryInterface(IID_IUnknown, (void**)&pUnk);
  2184. #ifdef USE_PHONEMSP
  2185. if ( HasWaveDevice() || ( NULL != GetHPhoneApp() ) )
  2186. #else
  2187. if ( HasWaveDevice() )
  2188. #endif USE_PHONEMSP
  2189. {
  2190. t3Line.hLine = NULL;
  2191. t3Line.pAddress = this;
  2192. t3Line.dwAddressLineStructHandle = 0;
  2193. hr = LineOpen(
  2194. GetHLineApp(),
  2195. GetDeviceID(),
  2196. GetAddressID(),
  2197. &t3Line,
  2198. GetAPIVersion(),
  2199. LINECALLPRIVILEGE_NONE,
  2200. LINEMEDIAMODE_UNKNOWN,
  2201. 0,
  2202. NULL,
  2203. this,
  2204. GetTapi(),
  2205. FALSE // no need to add to line hash table
  2206. );
  2207. if (S_OK != hr)
  2208. {
  2209. LOG((TL_ERROR, "Initialize failed to open the line"));
  2210. t3Line.hLine = NULL;
  2211. t3Line.dwAddressLineStructHandle = 0;
  2212. }
  2213. //
  2214. // Try to set the event filter mask to the TapiSrv level
  2215. //
  2216. hr = m_EventMasks.SetTapiSrvAddressEventMask(
  2217. t3Line.hLine
  2218. );
  2219. if( FAILED(hr) )
  2220. {
  2221. LOG((TL_ERROR, "SetTapiSrvAddressEventMask failed "));
  2222. }
  2223. m_hLine = t3Line.hLine;
  2224. }
  2225. #ifdef USE_PHONEMSP
  2226. {
  2227. HKEY hKey;
  2228. TCHAR szProviderKeyName[256];
  2229. DWORD dwDataType;
  2230. DWORD dwDataSize = sizeof(DWORD);
  2231. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE,
  2232. "Software\\Microsoft\\Windows\\CurrentVersion\\Telephony\\TAPI3",
  2233. 0,
  2234. KEY_READ,
  2235. &hKey
  2236. ) == ERROR_SUCCESS)
  2237. {
  2238. // form reg name
  2239. wsprintf(szProviderKeyName, ("PreferPhoneMSPForProvider%d"), m_dwProviderID);
  2240. RegQueryValueEx(hKey,
  2241. szProviderKeyName,
  2242. 0,
  2243. &dwDataType,
  2244. (LPBYTE) &bPreferPhoneMSPForProvider,
  2245. &dwDataSize
  2246. );
  2247. RegCloseKey(hKey);
  2248. }
  2249. }
  2250. #endif USE_PHONEMSP
  2251. if ( (m_dwAddressFlags & ADDRESSFLAG_MSP) )
  2252. {
  2253. //
  2254. // CreateMSPObject
  2255. //
  2256. hr = CreateMSPObject(
  2257. m_dwDeviceID,
  2258. pUnk,
  2259. &m_pMSPAggAddress
  2260. );
  2261. if (S_OK != hr)
  2262. {
  2263. LOG((TL_ERROR, "Initialize - CreateMSPObject return %lx", hr));
  2264. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_MSP);
  2265. }
  2266. }
  2267. else
  2268. {
  2269. #ifdef USE_PHONEMSP
  2270. if(bPreferPhoneMSPForProvider == TRUE)
  2271. {
  2272. // Create the Phone MSP as preference
  2273. //
  2274. if ( NULL != GetHPhoneApp() )
  2275. {
  2276. hr = CreatePhoneDeviceMSP(
  2277. pUnk,
  2278. t3Line.hLine,
  2279. &m_pMSPAggAddress
  2280. );
  2281. if (!SUCCEEDED(hr))
  2282. {
  2283. LOG((TL_ERROR, "Initialize - failed to create phone msp object - %lx", hr));
  2284. }
  2285. }
  2286. //
  2287. // if it's a wave device, create wavemsp object
  2288. //
  2289. else if ( HasWaveDevice() )
  2290. {
  2291. hr = CreateWaveMSPObject(
  2292. pUnk,
  2293. &m_pMSPAggAddress
  2294. );
  2295. if (!SUCCEEDED(hr))
  2296. {
  2297. LOG((TL_ERROR, "Initialize - failed to create wave msp object - %lx", hr));
  2298. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_WAVEINDEVICE | ADDRESSFLAG_WAVEOUTDEVICE | ADDRESSFLAG_WAVEFULLDUPLEX);
  2299. }
  2300. }
  2301. }
  2302. else
  2303. #endif USE_PHONEMSP
  2304. {
  2305. //
  2306. // if it's a wave device, create wavemsp object as preference
  2307. //
  2308. if ( HasWaveDevice() )
  2309. {
  2310. hr = CreateWaveMSPObject(
  2311. pUnk,
  2312. &m_pMSPAggAddress
  2313. );
  2314. if (!SUCCEEDED(hr))
  2315. {
  2316. LOG((TL_ERROR, "Initialize - failed to create wave msp object - %lx", hr));
  2317. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_WAVEINDEVICE | ADDRESSFLAG_WAVEOUTDEVICE | ADDRESSFLAG_WAVEFULLDUPLEX);
  2318. }
  2319. }
  2320. #ifdef USE_PHONEMSP
  2321. // Create the Phone MSP
  2322. //
  2323. else if ( NULL != GetHPhoneApp() )
  2324. {
  2325. hr = CreatePhoneDeviceMSP(
  2326. pUnk,
  2327. t3Line.hLine,
  2328. &m_pMSPAggAddress
  2329. );
  2330. if (!SUCCEEDED(hr))
  2331. {
  2332. LOG((TL_ERROR, "Initialize - failed to create phone msp object - %lx", hr));
  2333. }
  2334. }
  2335. #endif USE_PHONEMSP
  2336. }
  2337. }
  2338. pUnk->Release();
  2339. if (NULL != m_pMSPAggAddress)
  2340. {
  2341. m_hMSPEvent = CreateEvent(
  2342. NULL,
  2343. FALSE,
  2344. FALSE,
  2345. NULL
  2346. );
  2347. if ( NULL == m_hMSPEvent )
  2348. {
  2349. LOG((TL_ERROR, "Initialize - can't create MSP event"));
  2350. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_AMREL);
  2351. //
  2352. // release msp object, so we are not tempted to do Shutdown on it
  2353. // later without having (successfully) initialized it first
  2354. //
  2355. m_pMSPAggAddress->Release();
  2356. m_pMSPAggAddress = NULL;
  2357. }
  2358. else
  2359. {
  2360. // Create a context handle to give the Callback & associate it with this object
  2361. // in the global handle hash table
  2362. m_MSPContext = GenerateHandleAndAddToHashTable((ULONG_PTR)this);
  2363. LOG((TL_INFO, "Initialize - Map MSP handle %p to Address object %p", m_MSPContext, this ));
  2364. BOOL fSuccess = RegisterWaitForSingleObject(
  2365. & m_hWaitEvent,
  2366. m_hMSPEvent,
  2367. MSPEventCallback,
  2368. (PVOID)m_MSPContext,
  2369. INFINITE,
  2370. WT_EXECUTEDEFAULT
  2371. );
  2372. if ( ( ! fSuccess ) || ( NULL == m_hWaitEvent ) )
  2373. {
  2374. LOG((TL_ERROR, "Initialize - RegisterWaitForSingleObject failed"));
  2375. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_AMREL);
  2376. m_pMSPAggAddress->Release();
  2377. m_pMSPAggAddress = NULL;
  2378. }
  2379. else
  2380. {
  2381. ITMSPAddress * pMSPAddress = GetMSPAddress();
  2382. hr = pMSPAddress->Initialize(
  2383. (MSP_HANDLE)m_hMSPEvent
  2384. );
  2385. if ( SUCCEEDED(hr) && HasWaveDevice() )
  2386. {
  2387. InitializeWaveDeviceIDs( t3Line.hLine );
  2388. }
  2389. pMSPAddress->Release();
  2390. if (!SUCCEEDED(hr))
  2391. {
  2392. LOG((TL_ERROR, "Initialize - failed to initialize msp object - %lx", hr));
  2393. UnregisterWait( m_hWaitEvent );
  2394. m_hWaitEvent = NULL;
  2395. m_dwAddressFlags = m_dwAddressFlags & ~(ADDRESSFLAG_AMREL);
  2396. m_pMSPAggAddress->Release();
  2397. m_pMSPAggAddress = NULL;
  2398. }
  2399. }
  2400. }
  2401. }
  2402. if ( NULL != t3Line.hLine )
  2403. {
  2404. LineClose( &t3Line );
  2405. }
  2406. //
  2407. // get the address caps structure
  2408. //
  2409. hr = UpdateAddressCaps();
  2410. if (S_OK != hr)
  2411. {
  2412. LOG((TL_ERROR, "Initialize - LineGetAddressCaps failed - return %lx", hr ));
  2413. Unlock();
  2414. LOG((TL_ERROR, hr, "Initialize - exit LineGetAddressCaps failed"));
  2415. return hr;
  2416. }
  2417. //
  2418. // save the address name
  2419. //
  2420. if (0 != m_pAddressCaps->dwAddressSize)
  2421. {
  2422. m_szAddress = (PWSTR) ClientAlloc (m_pAddressCaps->dwAddressSize + sizeof(WCHAR));
  2423. if (NULL == m_szAddress)
  2424. {
  2425. LOG((TL_ERROR, "Initialize - alloc m_szAddress failed" ));
  2426. Unlock();
  2427. LOG((TL_ERROR, E_OUTOFMEMORY, "Initialize - exit alloc m_szAddress failed"));
  2428. return E_OUTOFMEMORY;
  2429. }
  2430. memcpy(
  2431. m_szAddress,
  2432. ((LPBYTE)(m_pAddressCaps)) + m_pAddressCaps->dwAddressOffset,
  2433. m_pAddressCaps->dwAddressSize
  2434. );
  2435. LOG((TL_INFO, "CAddress - Address is '%ls'", m_szAddress ));
  2436. }
  2437. hr = SaveAddressName( pDevCaps );
  2438. if ( !SUCCEEDED(hr) )
  2439. {
  2440. LOG((TL_ERROR, "Initialize - SaveAddressName failed %lx", hr));
  2441. ClientFree(m_szAddress);
  2442. /*This is to take care of AV*/
  2443. m_szAddress = NULL;
  2444. Unlock();
  2445. LOG((TL_ERROR, hr, "Initialize - exit SaveAddressName failed"));
  2446. return hr;
  2447. }
  2448. LOG((TL_INFO, "CAddress - m_szAddressName is '%ls'", m_szAddressName ));
  2449. LOG((TL_TRACE, "Initialize - exit S_OK" ));
  2450. Unlock();
  2451. return S_OK;
  2452. }
  2453. #ifdef USE_PHONEMSP
  2454. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2455. //
  2456. // CreatePhoneDeviceMSP
  2457. // This address has a tapi phone devices, so create
  2458. // the relevant terminals
  2459. //
  2460. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2461. HRESULT
  2462. CAddress::CreatePhoneDeviceMSP(
  2463. IUnknown * pUnk,
  2464. HLINE hLine,
  2465. IUnknown ** ppMSPAggAddress
  2466. )
  2467. {
  2468. HRESULT hr = S_OK;
  2469. BOOL bSucceeded = FALSE;
  2470. LPPHONECAPS pPhoneCaps;
  2471. DWORD dwPhoneDevice;
  2472. LPVARSTRING pVarString = NULL;
  2473. CComAggObject<CPhoneMSP> * pPhoneMSP;
  2474. CPhoneMSP * pCPhoneMSP;
  2475. ITMSPAddress * pMSPAddress;
  2476. LOG((TL_TRACE, "CreatePhoneTerminal enter" ));
  2477. //
  2478. // get the related phone device id
  2479. //
  2480. hr = LineGetID(
  2481. hLine,
  2482. GetAddressID(),
  2483. NULL,
  2484. LINECALLSELECT_ADDRESS,
  2485. &pVarString,
  2486. L"tapi/phone"
  2487. );
  2488. //
  2489. // there is no phone device
  2490. //
  2491. if (S_OK != hr)
  2492. {
  2493. if ( NULL != pVarString )
  2494. {
  2495. ClientFree( pVarString );
  2496. /*This is to take care of AV*/
  2497. pVarString = NULL;
  2498. }
  2499. return hr;
  2500. }
  2501. //
  2502. // get the phone device id at the endo
  2503. // of the var string
  2504. //
  2505. if (pVarString->dwStringSize < sizeof(DWORD))
  2506. {
  2507. LOG((TL_ERROR, "CreatePhoneDeviceMSP - dwStringSize < 4" ));
  2508. LOG((TL_ERROR, "CreatePhoneDeviceMSP exit - return LINEERR_OPERATIONFAILED" ));
  2509. ClientFree( pVarString );
  2510. /*This is to take care of AV*/
  2511. pVarString = NULL;
  2512. return mapTAPIErrorCode( LINEERR_OPERATIONFAILED );
  2513. }
  2514. //
  2515. // get the phone device
  2516. //
  2517. dwPhoneDevice = (DWORD) ( * ( ( (PBYTE)pVarString ) + pVarString->dwStringOffset ) );
  2518. ClientFree( pVarString );
  2519. /*This is to take care of AV*/
  2520. pVarString = NULL;
  2521. //
  2522. // create the msp object
  2523. //
  2524. pPhoneMSP = new CComAggObject<CPhoneMSP>(pUnk);
  2525. /*NikhilB: This is to take care of an AV*/
  2526. if ( NULL == pPhoneMSP )
  2527. {
  2528. LOG((TL_ERROR, "Could not allocate for phone MSP object" ));
  2529. return E_OUTOFMEMORY;
  2530. }
  2531. //
  2532. // save the aggregated interface in
  2533. // the msppointer
  2534. //
  2535. pPhoneMSP->QueryInterface(
  2536. IID_IUnknown,
  2537. (void **)ppMSPAggAddress
  2538. );
  2539. //
  2540. // get to the real object
  2541. //
  2542. pMSPAddress = GetMSPAddress();
  2543. pCPhoneMSP = dynamic_cast<CPhoneMSP *>(pMSPAddress);
  2544. //
  2545. // initialize it
  2546. //
  2547. hr = pCPhoneMSP->InitializeTerminals(
  2548. GetHPhoneApp(),
  2549. m_dwAPIVersion,
  2550. dwPhoneDevice,
  2551. this
  2552. );
  2553. pCPhoneMSP->Release();
  2554. if ( !SUCCEEDED(hr) )
  2555. {
  2556. }
  2557. LOG((TL_TRACE, "CreatePhoneDeviceMSP exit - return %lx", hr ));
  2558. return hr;
  2559. }
  2560. #endif USE_PHONEMSP
  2561. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2562. //
  2563. // AddCall
  2564. // Keep track of calls
  2565. //
  2566. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2567. HRESULT
  2568. CAddress::AddCall(
  2569. ITCallInfo * pCallInfo
  2570. )
  2571. {
  2572. Lock();
  2573. m_CallArray.Add( pCallInfo );
  2574. Unlock();
  2575. return S_OK;
  2576. }
  2577. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2578. //
  2579. // RemoveCall
  2580. //
  2581. // remove call from address's list
  2582. //
  2583. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2584. HRESULT
  2585. CAddress::RemoveCall(
  2586. ITCallInfo * pCallInfo
  2587. )
  2588. {
  2589. Lock();
  2590. m_CallArray.Remove( pCallInfo );
  2591. Unlock();
  2592. return S_OK;
  2593. }
  2594. // ITMediaSupport methods
  2595. HRESULT
  2596. STDMETHODCALLTYPE
  2597. CAddress::get_MediaTypes(
  2598. long * plMediaTypes
  2599. )
  2600. {
  2601. LOG((TL_TRACE, "get_MediaTypes enter" ));
  2602. LOG((TL_TRACE, " plMediaType ------->%p", plMediaTypes ));
  2603. if (TAPIIsBadWritePtr( plMediaTypes, sizeof( long ) ) )
  2604. {
  2605. LOG((TL_ERROR, "get_MediaTypes - bad pointer"));
  2606. return E_POINTER;
  2607. }
  2608. *plMediaTypes = (long)m_dwMediaModesSupported;
  2609. if (*plMediaTypes & LINEMEDIAMODE_INTERACTIVEVOICE)
  2610. {
  2611. *plMediaTypes |= LINEMEDIAMODE_AUTOMATEDVOICE;
  2612. }
  2613. *plMediaTypes &= ALLMEDIAMODES;
  2614. return S_OK;
  2615. }
  2616. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2617. //
  2618. // QueryMediaType
  2619. // find out of mediatype is supported
  2620. //
  2621. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  2622. HRESULT
  2623. STDMETHODCALLTYPE
  2624. CAddress::QueryMediaType(
  2625. long lMediaType,
  2626. VARIANT_BOOL * pbSupport
  2627. )
  2628. {
  2629. HRESULT hr = S_OK;
  2630. DWORD dwMediaMode;
  2631. LOG((TL_TRACE, "QueryMediaType enter"));
  2632. LOG((TL_TRACE, " lMediaType--------->%lx", lMediaType));
  2633. LOG((TL_TRACE, " pbSupport---------->%p", pbSupport));
  2634. if ( TAPIIsBadWritePtr( pbSupport, sizeof(VARIANT_BOOL) ) )
  2635. {
  2636. LOG((TL_ERROR, "QueryMediaType - inval pointer"));
  2637. return E_POINTER;
  2638. }
  2639. //
  2640. // get the tapi mediamode that
  2641. // the application is asking about
  2642. //
  2643. if (GetMediaMode(
  2644. lMediaType,
  2645. &dwMediaMode
  2646. ) )
  2647. {
  2648. *pbSupport = VARIANT_TRUE;
  2649. }
  2650. else
  2651. {
  2652. *pbSupport = VARIANT_FALSE;
  2653. }
  2654. LOG((TL_TRACE, "QueryMediaType exit - return success"));
  2655. return S_OK;
  2656. }
  2657. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2658. //
  2659. // AddCallNotification
  2660. //
  2661. // Adds a callback to notify on call events
  2662. //
  2663. // dwPrivs
  2664. // IN tapi 2 style Privileges
  2665. //
  2666. // dwMediaModes
  2667. // IN tapi 2 style mediamodes
  2668. //
  2669. // pNotification
  2670. // IN the callback to notify
  2671. //
  2672. // pulRegiser
  2673. // OUT the unique ID of the callback (so they can unregister)
  2674. //
  2675. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2676. HRESULT
  2677. CAddress::AddCallNotification(
  2678. DWORD dwPrivs,
  2679. DWORD dwMediaModes,
  2680. long lInstance,
  2681. PVOID * ppRegister
  2682. )
  2683. {
  2684. HRESULT hr;
  2685. AddressLineStruct * pAddressLine;
  2686. //
  2687. // create addressline
  2688. //
  2689. pAddressLine = (AddressLineStruct *)ClientAlloc( sizeof( AddressLineStruct ) );
  2690. if (NULL == pAddressLine)
  2691. {
  2692. LOG((TL_ERROR, "AddCallNotification - pAddressLine is NULL"));
  2693. return E_OUTOFMEMORY;
  2694. }
  2695. //
  2696. // initialize
  2697. //
  2698. pAddressLine->dwPrivs = dwPrivs;
  2699. pAddressLine->t3Line.pAddress = this;
  2700. pAddressLine->t3Line.dwAddressLineStructHandle = 0;
  2701. pAddressLine->InitializeRefcount(1);
  2702. pAddressLine->lCallbackInstance = lInstance;
  2703. if (ALLMEDIATYPES == dwMediaModes)
  2704. {
  2705. pAddressLine->dwMediaModes = GetMediaModes();
  2706. }
  2707. else
  2708. {
  2709. pAddressLine->dwMediaModes = dwMediaModes;
  2710. //
  2711. // if we are listening on any audio mode
  2712. //
  2713. if (pAddressLine->dwMediaModes & AUDIOMEDIAMODES)
  2714. {
  2715. //
  2716. // add in all the audio modes that we support
  2717. //
  2718. pAddressLine->dwMediaModes |= (GetMediaModes() & AUDIOMEDIAMODES);
  2719. }
  2720. }
  2721. //
  2722. // get a line
  2723. //
  2724. hr = LineOpen(
  2725. GetHLineApp(),
  2726. GetDeviceID(),
  2727. GetAddressID(),
  2728. &(pAddressLine->t3Line),
  2729. GetAPIVersion(),
  2730. pAddressLine->dwPrivs,
  2731. pAddressLine->dwMediaModes,
  2732. pAddressLine,
  2733. NULL,
  2734. (CAddress *)this,
  2735. GetTapi()
  2736. );
  2737. if (!SUCCEEDED(hr))
  2738. {
  2739. LOG((TL_ERROR, "AddCallNotification - LineOpen failed %lx", hr));
  2740. ClientFree( pAddressLine );
  2741. return hr;
  2742. }
  2743. //
  2744. // Try to set the event filter mask to the TapiSrv level
  2745. //
  2746. hr = m_EventMasks.SetTapiSrvAddressEventMask(
  2747. pAddressLine->t3Line.hLine
  2748. );
  2749. if( FAILED(hr) )
  2750. {
  2751. LOG((TL_ERROR, "AddCallNotification - SetTapiSrvAddressEventMask failed 0x%08x", hr));
  2752. LineClose( &(pAddressLine->t3Line) );
  2753. ClientFree( pAddressLine );
  2754. return hr;
  2755. }
  2756. // Fix for bug 263866 & 250924
  2757. //
  2758. if ( m_dwAddressFlags & ADDRESSFLAG_MSP )
  2759. {
  2760. hr = LineCreateMSPInstance(
  2761. pAddressLine->t3Line.hLine,
  2762. GetAddressID()
  2763. );
  2764. if ( !SUCCEEDED(hr) )
  2765. {
  2766. LOG((TL_ERROR, "AddCallNotification - LineCreateMSPInstance failed %lx", hr));
  2767. LineClose( &(pAddressLine->t3Line) );
  2768. ClientFree( pAddressLine );
  2769. return hr;
  2770. }
  2771. }
  2772. // Tell TAPISRV what status messages we want
  2773. // NOTE : if the lines SP doesn't support TSPI_lineSetStatusMessages
  2774. // TAPISRV will fail this with LINEERR_OPERATIONUNAVAIL
  2775. LineSetStatusMessages(
  2776. &(pAddressLine->t3Line),
  2777. ALL_LINEDEVSTATE_MESSAGES,
  2778. ALL_LINEADDRESSSTATE_MESSAGES
  2779. );
  2780. //
  2781. // ZoltanS fix: enable callhub notifications only if the app
  2782. // hasn't already told us it doesn't want them
  2783. //
  2784. if ( m_fEnableCallHubTrackingOnLineOpen )
  2785. {
  2786. LINECALLHUBTRACKINGINFO lchti;
  2787. lchti.dwTotalSize = sizeof(lchti);
  2788. lchti.dwCurrentTracking = LINECALLHUBTRACKING_ALLCALLS;
  2789. hr = LineSetCallHubTracking(
  2790. &(pAddressLine->t3Line),
  2791. &lchti
  2792. );
  2793. if ( S_OK != hr )
  2794. {
  2795. LOG((TL_ERROR, "AddCallNotification - LineSetCallHubTracking failed %lx", hr ));
  2796. LineClose( &(pAddressLine->t3Line) );
  2797. ClientFree( pAddressLine );
  2798. return hr;
  2799. }
  2800. }
  2801. if (m_pCallHubTrackingLine)
  2802. {
  2803. MaybeCloseALine( &m_pCallHubTrackingLine );
  2804. m_pCallHubTrackingLine = pAddressLine;
  2805. pAddressLine->AddRef();
  2806. }
  2807. //
  2808. // save the line in the address's list
  2809. //
  2810. hr = AddNotificationLine( pAddressLine );;
  2811. if ( S_OK != hr )
  2812. {
  2813. LOG((TL_ERROR, "AddCallNotification - AddNotificationLine failed %lx", hr ));
  2814. LineClose( &(pAddressLine->t3Line) );
  2815. ClientFree( pAddressLine );
  2816. }
  2817. else
  2818. {
  2819. //
  2820. // fill in pulRegister
  2821. //
  2822. *ppRegister = (PVOID) pAddressLine;
  2823. }
  2824. return hr;
  2825. }
  2826. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2827. //
  2828. // RemoveCallNotification
  2829. //
  2830. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2831. HRESULT
  2832. CAddress::RemoveCallNotification(
  2833. PVOID pRegister
  2834. )
  2835. {
  2836. AddressLineStruct *pLine = (AddressLineStruct *)pRegister;
  2837. MaybeCloseALine( &pLine );
  2838. return S_OK;
  2839. }
  2840. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2841. //
  2842. // FindOrOpenALine
  2843. //
  2844. // Attempts to find a line that's already open. If so, keeps a
  2845. // pointer to it. If not opens the line.
  2846. //
  2847. // dwMediaModes
  2848. // IN tapi 2 style mediamodes
  2849. //
  2850. // ppAddressLine
  2851. // OUT addresslinestruct associated with this request
  2852. //
  2853. // RETURNS
  2854. //
  2855. //
  2856. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2857. HRESULT
  2858. CAddress::FindOrOpenALine(
  2859. DWORD dwMediaModes,
  2860. AddressLineStruct ** ppAddressLine
  2861. )
  2862. {
  2863. HRESULT hr = S_OK;
  2864. LOG((TL_TRACE, "FindOrOpenALine - enter" ));
  2865. Lock();
  2866. //
  2867. // if there is already a line,
  2868. // just return it
  2869. //
  2870. if ( m_AddressLinesPtrList.size() > 0 )
  2871. {
  2872. LOG((TL_INFO, "Found a line that is already open" ));
  2873. *ppAddressLine = (AddressLineStruct *)*( m_AddressLinesPtrList.begin() );
  2874. (*ppAddressLine)->AddRef();
  2875. LOG((TL_TRACE, "FindOrOpenALine - exit"));
  2876. Unlock();
  2877. return S_OK;
  2878. }
  2879. //
  2880. // we didn't have an address
  2881. //
  2882. LOG((TL_INFO, "Did not find an already open line"));
  2883. //
  2884. // create a new addressline
  2885. //
  2886. AddressLineStruct * pLine = (AddressLineStruct *)ClientAlloc( sizeof(AddressLineStruct) );
  2887. if (NULL == pLine)
  2888. {
  2889. LOG((TL_ERROR, "FindOrOpenALine - alloc pLine failed" ));
  2890. Unlock();
  2891. return E_OUTOFMEMORY;
  2892. }
  2893. //
  2894. // initialize
  2895. //
  2896. pLine->dwMediaModes = dwMediaModes;
  2897. pLine->dwPrivs = LINECALLPRIVILEGE_NONE;
  2898. pLine->t3Line.pAddress = this;
  2899. pLine->t3Line.dwAddressLineStructHandle = 0;
  2900. pLine->InitializeRefcount(1);
  2901. pLine->lCallbackInstance = 0;
  2902. LOG((TL_INFO, "FindOrOpenALine - Opening a line" ));
  2903. //
  2904. // open the line
  2905. //
  2906. hr = LineOpen(
  2907. GetHLineApp(),
  2908. GetDeviceID(),
  2909. GetAddressID(),
  2910. &(pLine->t3Line),
  2911. GetAPIVersion(),
  2912. LINECALLPRIVILEGE_NONE,
  2913. dwMediaModes,
  2914. pLine,
  2915. NULL,
  2916. this,
  2917. GetTapi()
  2918. );
  2919. if (S_OK != hr)
  2920. {
  2921. LOG((TL_ERROR, "FindOrOpenALine - LineOpen failed %lx", hr ));
  2922. ClientFree( pLine );
  2923. Unlock();
  2924. return hr;
  2925. }
  2926. //
  2927. // Try to set the event filter mask to the TapiSrv level
  2928. //
  2929. hr = m_EventMasks.SetTapiSrvAddressEventMask(
  2930. pLine->t3Line.hLine
  2931. );
  2932. if( FAILED(hr) )
  2933. {
  2934. LOG((TL_ERROR, "FindOrOpenALine - SetTapiSrvAddressEventMask failed 0x%08x", hr));
  2935. LineClose( &(pLine->t3Line) );
  2936. ClientFree( pLine );
  2937. Unlock();
  2938. return hr;
  2939. }
  2940. if ( m_dwAddressFlags & ADDRESSFLAG_MSP )
  2941. {
  2942. hr = LineCreateMSPInstance(
  2943. pLine->t3Line.hLine,
  2944. GetAddressID()
  2945. );
  2946. if ( !SUCCEEDED(hr) )
  2947. {
  2948. LOG((TL_ERROR, "FindOrOpenALine - LineCreateMSPInstance failed %lx", hr));
  2949. LineClose( &(pLine->t3Line) );
  2950. ClientFree( pLine );
  2951. Unlock();
  2952. return hr;
  2953. }
  2954. }
  2955. // Tell TAPISRV what status messages we want
  2956. // NOTE : if the lines SP doesn't support TSPI_lineSetStatusMessages
  2957. // TAPISRV will fail this with LINEERR_OPERATIONUNAVAIL
  2958. LineSetStatusMessages(
  2959. &(pLine->t3Line),
  2960. ALL_LINEDEVSTATE_MESSAGES,
  2961. ALL_LINEADDRESSSTATE_MESSAGES
  2962. );
  2963. //
  2964. // ZoltanS fix: enable callhub notifications only if the app
  2965. // hasn't already told us it doesn't want them
  2966. //
  2967. if ( m_fEnableCallHubTrackingOnLineOpen )
  2968. {
  2969. LINECALLHUBTRACKINGINFO lchti;
  2970. lchti.dwTotalSize = sizeof(lchti);
  2971. lchti.dwCurrentTracking = LINECALLHUBTRACKING_ALLCALLS;
  2972. hr = LineSetCallHubTracking(
  2973. &(pLine->t3Line),
  2974. &lchti
  2975. );
  2976. if ( S_OK != hr )
  2977. {
  2978. LOG((TL_ERROR, "FindOrOpenALine - LineSetCallHubTracking failed %lx", hr ));
  2979. LineClose( &(pLine->t3Line) );
  2980. ClientFree( pLine );
  2981. Unlock();
  2982. return hr;
  2983. }
  2984. }
  2985. //
  2986. // Add this line to our list of open lines. This is independent of
  2987. // the callhub tracking stuff above.
  2988. //
  2989. hr = AddNotificationLine( pLine );
  2990. if ( S_OK != hr )
  2991. {
  2992. LOG((TL_ERROR, "FindOrOpenALine - AddNotificationLine failed %lx", hr ));
  2993. LineClose( &(pLine->t3Line) );
  2994. ClientFree( pLine );
  2995. }
  2996. else
  2997. {
  2998. *ppAddressLine = pLine;
  2999. }
  3000. LOG((TL_TRACE, "FindOrOpenALine - exit"));
  3001. Unlock();
  3002. return hr;
  3003. }
  3004. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3005. //
  3006. // MaybeCloseALine
  3007. // After a call is done, close the line if it's not being
  3008. // used for monitoring.
  3009. //
  3010. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3011. HRESULT
  3012. CAddress::MaybeCloseALine(
  3013. AddressLineStruct ** ppAddressLine
  3014. )
  3015. {
  3016. HRESULT hr = S_OK;
  3017. LOG((TL_TRACE, "MaybeCloseALine - enter"));
  3018. Lock();
  3019. if (NULL == *ppAddressLine || !IsValidAddressLine(*ppAddressLine))
  3020. {
  3021. Unlock();
  3022. return S_OK;
  3023. }
  3024. //
  3025. // decrement reference count on the the address line
  3026. //
  3027. DWORD dwAddressLineRefcount = (*ppAddressLine)->Release();
  3028. //
  3029. // if ref count is 0, close line
  3030. //
  3031. if ( 0 == dwAddressLineRefcount )
  3032. {
  3033. m_AddressLinesPtrList.remove( (PVOID)*ppAddressLine );
  3034. //
  3035. // clean handle table.
  3036. //
  3037. try
  3038. {
  3039. DWORD dwAddressLineHandle = (*ppAddressLine)->t3Line.dwAddressLineStructHandle;
  3040. DeleteHandleTableEntry(dwAddressLineHandle);
  3041. }
  3042. catch(...)
  3043. {
  3044. LOG((TL_ERROR,
  3045. "MaybeCloseALine - exception accessing address line's handle" ));
  3046. _ASSERTE(FALSE);
  3047. }
  3048. //
  3049. // attempt to close the line
  3050. //
  3051. LOG((TL_INFO, "MaybeCloseALine - Calling LineClose" ));
  3052. if ( m_dwAddressFlags & ADDRESSFLAG_MSP )
  3053. {
  3054. LineCloseMSPInstance( (*ppAddressLine)->t3Line.hLine );
  3055. }
  3056. //
  3057. // release the lock and call lineclose
  3058. //
  3059. Unlock();
  3060. //
  3061. // now that the line was removed from our list of managed lines and all
  3062. // the calls have been notified, we can close the line and free the
  3063. // structure
  3064. //
  3065. hr = LineClose(
  3066. &((*ppAddressLine)->t3Line)
  3067. );
  3068. ClientFree( *ppAddressLine );
  3069. }
  3070. else
  3071. {
  3072. Unlock();
  3073. //
  3074. // otherwise, decrement the dwRefCount count
  3075. //
  3076. LOG((TL_INFO, "MaybeCloseALine - Not calling line close - decrementing number of addresses using line" ));
  3077. }
  3078. *ppAddressLine = NULL;
  3079. LOG((TL_TRACE, "MaybeCloseALine - exit"));
  3080. return hr;
  3081. }
  3082. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3083. //
  3084. // SetCallHubTracking
  3085. //
  3086. // called by the tapi object to start callhub tracking on this
  3087. // address
  3088. //
  3089. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3090. HRESULT
  3091. CAddress::SetCallHubTracking(
  3092. BOOL bSet
  3093. )
  3094. {
  3095. HRESULT hr = S_OK;
  3096. LINECALLHUBTRACKINGINFO lchti;
  3097. Lock();
  3098. //
  3099. // Check if we have to make any change now.
  3100. //
  3101. if ( bSet == m_fEnableCallHubTrackingOnLineOpen )
  3102. {
  3103. //
  3104. // already doing the right thing
  3105. //
  3106. Unlock();
  3107. return S_OK;
  3108. }
  3109. //
  3110. // ZoltanS: make sure we don't clobber this setting in other methods
  3111. // (e.g., AddCallNotification or FindOrOpenALine) where the
  3112. // default call notification behavior (true) would otherwise
  3113. // be set.
  3114. //
  3115. m_fEnableCallHubTrackingOnLineOpen = bSet;
  3116. //
  3117. // need an hline to call setcallhubtracking
  3118. //
  3119. if ( m_pCallHubTrackingLine == NULL )
  3120. {
  3121. hr = FindOrOpenALine(
  3122. LINEMEDIAMODE_INTERACTIVEVOICE,
  3123. &m_pCallHubTrackingLine
  3124. );
  3125. if ( !SUCCEEDED( hr ) )
  3126. {
  3127. LOG((TL_ERROR, "SCHT - FindOrOpen failed %lx", hr));
  3128. Unlock();
  3129. return hr;
  3130. }
  3131. //
  3132. // ZoltanS fix:
  3133. // FindOrOpenALine does callhubtracking for non-sp
  3134. // tracking capable lines, but only if it is opening
  3135. // the line (rather than just finding it in our list.
  3136. // So we still need to do LineSetCallHubTracking below,
  3137. // even for an SPCallHubTrackingAddress.
  3138. //
  3139. }
  3140. //
  3141. // tell the sp to track callhubs
  3142. //
  3143. ZeroMemory(
  3144. &lchti,
  3145. sizeof(lchti)
  3146. );
  3147. lchti.dwTotalSize = sizeof(lchti);
  3148. lchti.dwCurrentTracking = bSet? LINECALLHUBTRACKING_ALLCALLS :
  3149. LINECALLHUBTRACKING_NONE;
  3150. //
  3151. // ZoltanS: Only pass this flag if it applies.
  3152. // Note: LineSetCallHubTracking apparently ignores its first argument
  3153. // if the tracking level is set to LINECALLHUBTRACKING_NONE; otherwise
  3154. // we would also have to FindOrOpenALine if we are unsetting.
  3155. //
  3156. if ( bSet && IsSPCallHubTrackingAddress() )
  3157. {
  3158. lchti.dwCurrentTracking |= LINECALLHUBTRACKING_PROVIDERLEVEL;
  3159. }
  3160. hr = LineSetCallHubTracking(
  3161. &(m_pCallHubTrackingLine->t3Line),
  3162. &lchti
  3163. );
  3164. if ( !SUCCEEDED( hr ) )
  3165. {
  3166. LOG((TL_ERROR, "SCHT - LSCHT failed %lx", hr));
  3167. if ( bSet )
  3168. {
  3169. MaybeCloseALine( &m_pCallHubTrackingLine );
  3170. m_pCallHubTrackingLine = NULL;
  3171. }
  3172. Unlock();
  3173. return hr;
  3174. }
  3175. //
  3176. // Also properly set callhub tracking on lines that were
  3177. // opened for call notification
  3178. //
  3179. PtrList::iterator l;
  3180. for (l = m_AddressLinesPtrList.begin(); l != m_AddressLinesPtrList.end(); l++)
  3181. {
  3182. AddressLineStruct * pLine;
  3183. pLine = (AddressLineStruct *)(*l);
  3184. if (pLine != m_pCallHubTrackingLine)
  3185. {
  3186. LineSetCallHubTracking(
  3187. &(pLine->t3Line),
  3188. &lchti
  3189. );
  3190. }
  3191. }
  3192. //
  3193. // close it if unsetting
  3194. //
  3195. if ( !bSet )
  3196. {
  3197. MaybeCloseALine( &m_pCallHubTrackingLine );
  3198. m_pCallHubTrackingLine = NULL;
  3199. }
  3200. Unlock();
  3201. return S_OK;
  3202. }
  3203. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3204. //
  3205. // DoesThisAddressSupportCallHubs
  3206. //
  3207. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3208. DWORD
  3209. CAddress::DoesThisAddressSupportCallHubs(
  3210. CCall * pCall
  3211. )
  3212. {
  3213. HCALL hCall;
  3214. LINECALLINFO * pCallInfo;
  3215. HRESULT hr;
  3216. //
  3217. // does the sp tell us it supports callhubs?
  3218. //
  3219. if ( IsSPCallHubAddress() )
  3220. {
  3221. return CALLHUBSUPPORT_FULL;
  3222. }
  3223. //
  3224. // have we already determined that it doesn't?
  3225. //
  3226. if ( m_dwAddressFlags & ADDRESSFLAG_NOCALLHUB )
  3227. {
  3228. return CALLHUBSUPPORT_NONE;
  3229. }
  3230. //
  3231. // otherwise - hack - see if the dwrelatedcallid
  3232. // field is non-zero in this call.
  3233. // if so, it does callhubs,
  3234. //
  3235. hCall = pCall->GetHCall();
  3236. hr = LineGetCallInfo(
  3237. hCall,
  3238. &pCallInfo
  3239. );
  3240. if ( SUCCEEDED( hr ) )
  3241. {
  3242. if ( 0 != pCallInfo->dwCallID )
  3243. {
  3244. Lock();
  3245. m_dwAddressFlags |= ADDRESSFLAG_CALLHUB;
  3246. Unlock();
  3247. ClientFree( pCallInfo );
  3248. return CALLHUBSUPPORT_FULL;
  3249. }
  3250. else
  3251. {
  3252. Lock();
  3253. m_dwAddressFlags |= ADDRESSFLAG_NOCALLHUB;
  3254. Unlock();
  3255. ClientFree( pCallInfo );
  3256. return CALLHUBSUPPORT_NONE;
  3257. }
  3258. }
  3259. return CALLHUBSUPPORT_UNKNOWN;
  3260. }
  3261. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3262. //
  3263. // CreateForwardInfoObject
  3264. //
  3265. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3266. STDMETHODIMP
  3267. CAddress::CreateForwardInfoObject(
  3268. ITForwardInformation ** ppForwardInfo
  3269. )
  3270. {
  3271. CComObject< CForwardInfo > * p;
  3272. HRESULT hr = S_OK;
  3273. LOG((TL_TRACE, "CreatForwardInfoObject - enter"));
  3274. if (TAPIIsBadWritePtr(ppForwardInfo , sizeof(ITForwardInformation *) ) )
  3275. {
  3276. LOG((TL_ERROR, "CreateForwardInfoObject - bad pointer"));
  3277. return E_POINTER;
  3278. }
  3279. //
  3280. // create object
  3281. //
  3282. CComObject< CForwardInfo >::CreateInstance( &p );
  3283. if ( NULL == p )
  3284. {
  3285. LOG((TL_ERROR, "Create forward object failed"));
  3286. return E_OUTOFMEMORY;
  3287. }
  3288. //
  3289. // init
  3290. //
  3291. hr = p->Initialize();
  3292. if ( !SUCCEEDED(hr) )
  3293. {
  3294. LOG((TL_ERROR, "Initialize forward object failed"));
  3295. delete p;
  3296. return hr;
  3297. }
  3298. //
  3299. // return
  3300. //
  3301. hr = p->QueryInterface(
  3302. IID_ITForwardInformation,
  3303. (void**)ppForwardInfo
  3304. );
  3305. if ( !SUCCEEDED(hr) )
  3306. {
  3307. LOG((TL_ERROR, "CreateForwardobject failed - %lx", hr));
  3308. delete p;
  3309. return hr;
  3310. }
  3311. LOG((TL_TRACE, "CreateForwardObject - exit success"));
  3312. return S_OK;
  3313. }
  3314. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3315. //
  3316. // Forward - simple forwarding
  3317. //
  3318. // will unconditially forward to pDestAddress
  3319. //
  3320. // if pDestAddress is NULL, will cancel forwarding
  3321. //
  3322. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3323. STDMETHODIMP
  3324. CAddress::Forward(
  3325. ITForwardInformation * pForwardInfo,
  3326. ITBasicCallControl * pCall
  3327. )
  3328. {
  3329. HRESULT hr;
  3330. LINEFORWARDLIST * pList;
  3331. AddressLineStruct * pLine;
  3332. HCALL hCall = NULL;
  3333. CForwardInfo * pFI;
  3334. CCall * pCCall;
  3335. LONG lCap;
  3336. if ( IsBadReadPtr( pForwardInfo, sizeof( ITForwardInformation * ) ) )
  3337. {
  3338. LOG((TL_ERROR, "Forward - bad pForwardInfo"));
  3339. return E_POINTER;
  3340. }
  3341. hr = get_AddressCapability( AC_ADDRESSCAPFLAGS, &lCap );
  3342. if ( !SUCCEEDED(hr) )
  3343. {
  3344. return hr;
  3345. }
  3346. if ( lCap & LINEADDRCAPFLAGS_FWDCONSULT )
  3347. {
  3348. if ( IsBadReadPtr( pCall, sizeof(ITBasicCallControl *) ) )
  3349. {
  3350. LOG((TL_ERROR, "Forward - Need consultation call"));
  3351. return E_INVALIDARG;
  3352. }
  3353. pCCall = dynamic_cast<CCall *>(pCall);
  3354. if ( NULL == pCCall )
  3355. {
  3356. LOG((TL_ERROR, "Forward - invalid call"));
  3357. return E_POINTER;
  3358. }
  3359. }
  3360. pFI = dynamic_cast<CForwardInfo *>(pForwardInfo);
  3361. if ( NULL == pFI )
  3362. {
  3363. return E_POINTER;
  3364. }
  3365. hr = pFI->CreateForwardList( &pList );
  3366. if ( !SUCCEEDED(hr) )
  3367. {
  3368. return hr;
  3369. }
  3370. //
  3371. // get a line
  3372. //
  3373. hr = FindOrOpenALine(
  3374. LINEMEDIAMODE_INTERACTIVEVOICE,
  3375. &pLine
  3376. );
  3377. if ( !SUCCEEDED(hr) )
  3378. {
  3379. LOG((TL_ERROR, "Forward - FindOrOpen failed - %lx", hr));
  3380. ClientFree( pList );
  3381. return hr;
  3382. }
  3383. //
  3384. // call forward
  3385. //
  3386. //
  3387. DWORD dwRings;
  3388. pForwardInfo->get_NumRingsNoAnswer( (long *)&dwRings );
  3389. hr = LineForward(
  3390. &(pLine->t3Line),
  3391. m_dwAddressID,
  3392. pList,
  3393. dwRings,
  3394. &hCall
  3395. );
  3396. ClientFree( pList );
  3397. if ( ((long)hr) < 0 )
  3398. {
  3399. LOG((TL_ERROR, "Forward failed sync - %lx", hr));
  3400. MaybeCloseALine( &pLine );
  3401. return hr;
  3402. }
  3403. hr = WaitForReply( hr );
  3404. if ( lCap & LINEADDRCAPFLAGS_FWDCONSULT )
  3405. {
  3406. pCCall->Lock();
  3407. pCCall->FinishSettingUpCall( hCall );
  3408. pCCall->Unlock();
  3409. }
  3410. else
  3411. {
  3412. HRESULT hr2;
  3413. if( hCall != NULL )
  3414. {
  3415. hr2 = LineDrop(
  3416. hCall,
  3417. NULL,
  3418. 0
  3419. );
  3420. if ( ((long)hr2) > 0 )
  3421. {
  3422. hr2 = WaitForReply( hr2 );
  3423. }
  3424. LineDeallocateCall( hCall );
  3425. }
  3426. }
  3427. MaybeCloseALine( &pLine );
  3428. if ( !SUCCEEDED(hr) )
  3429. {
  3430. LOG((TL_ERROR, "Forward failed async - %lx", hr));
  3431. return hr;
  3432. }
  3433. LOG((TL_TRACE, "Forward - Exit"));
  3434. return S_OK;
  3435. }
  3436. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3437. // Class : CAddress
  3438. // Interface : ITAddress
  3439. // Method : get_CurrentForwardInfo
  3440. //
  3441. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3442. STDMETHODIMP
  3443. CAddress::get_CurrentForwardInfo(
  3444. ITForwardInformation ** ppForwardInfo
  3445. )
  3446. {
  3447. HRESULT hr = S_OK;
  3448. AddressLineStruct * pLine = NULL;
  3449. LINEADDRESSSTATUS * pAddressStatus = NULL;
  3450. LINEFORWARD * plineForward = NULL;
  3451. DWORD dwNumEntries = 0;
  3452. DWORD dwCount = 0;
  3453. PWSTR pszCallerAddress = NULL;
  3454. PWSTR pszDestddress = NULL;
  3455. LOG((TL_TRACE, "get_CurrentForwardInfo - enter"));
  3456. hr = CreateForwardInfoObject(ppForwardInfo);
  3457. if ( SUCCEEDED(hr) )
  3458. {
  3459. hr = FindOrOpenALine(
  3460. LINEMEDIAMODE_INTERACTIVEVOICE,
  3461. &pLine
  3462. );
  3463. if ( SUCCEEDED(hr) )
  3464. {
  3465. hr = LineGetAddressStatus(
  3466. &(pLine->t3Line),
  3467. m_dwAddressID,
  3468. &pAddressStatus
  3469. );
  3470. if ( SUCCEEDED(hr) )
  3471. {
  3472. (*ppForwardInfo)->put_NumRingsNoAnswer(pAddressStatus->dwNumRingsNoAnswer);
  3473. //
  3474. // if there is forwarding
  3475. //
  3476. dwNumEntries = pAddressStatus->dwForwardNumEntries;
  3477. plineForward = (LINEFORWARD *) (((LPBYTE)pAddressStatus) + pAddressStatus->dwForwardOffset);
  3478. for (dwCount = 0; dwCount != dwNumEntries; dwCount++)
  3479. {
  3480. if (plineForward->dwCallerAddressOffset > 0) // Caller address is not used for some forward modes
  3481. {
  3482. pszCallerAddress = (PWSTR) (((LPBYTE)pAddressStatus) + plineForward->dwCallerAddressOffset);
  3483. }
  3484. pszDestddress = (PWSTR) (((LPBYTE)pAddressStatus) + plineForward->dwDestAddressOffset);
  3485. if ( m_dwAPIVersion >= TAPI_VERSION3_1 )
  3486. {
  3487. //
  3488. // We have negotiated TAPI3.1 or better, so we can get the address types.
  3489. //
  3490. ITForwardInformation2 * pForwardInfo2;
  3491. //
  3492. // Get the ITForwardInformation2 interface so we can call SetForwardType2
  3493. //
  3494. hr = (*ppForwardInfo)->QueryInterface(
  3495. IID_ITForwardInformation2,
  3496. (void **)&pForwardInfo2
  3497. );
  3498. if ( SUCCEEDED(hr) )
  3499. {
  3500. pForwardInfo2->SetForwardType2(plineForward->dwForwardMode,
  3501. pszDestddress,
  3502. plineForward->dwDestAddressType,
  3503. pszCallerAddress,
  3504. plineForward->dwCallerAddressType
  3505. );
  3506. pForwardInfo2->Release();
  3507. }
  3508. else
  3509. {
  3510. LOG((TL_ERROR, "get_CurrentForwardInfo - QueryInterface failed - %lx", hr));
  3511. //
  3512. // If for some reason the QI fails, break out of the loop
  3513. //
  3514. break;
  3515. }
  3516. }
  3517. else
  3518. {
  3519. (*ppForwardInfo)->SetForwardType(plineForward->dwForwardMode,
  3520. pszDestddress,
  3521. pszCallerAddress
  3522. );
  3523. }
  3524. plineForward++;
  3525. }
  3526. ClientFree( pAddressStatus );
  3527. }
  3528. else
  3529. {
  3530. LOG((TL_ERROR, "get_CurrentForwardInfo - LineGetAddressStatus failed "));
  3531. }
  3532. MaybeCloseALine( &pLine );
  3533. }
  3534. else
  3535. {
  3536. LOG((TL_ERROR, "get_CurrentForwardInfo - FindOrOpen failed"));
  3537. }
  3538. }
  3539. else
  3540. {
  3541. LOG((TL_ERROR, "get_CurrentForwardInfo - failed to create"));
  3542. }
  3543. LOG((TL_TRACE, hr, "get_CurrentForwardInfo - exit"));
  3544. return hr;
  3545. }
  3546. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3547. //
  3548. //
  3549. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3550. STDMETHODIMP
  3551. CAddress::get_DialableAddress(
  3552. BSTR * ppDialableAddress
  3553. )
  3554. {
  3555. HRESULT hr = S_OK;
  3556. LOG((TL_TRACE, "get_DialableAddress - Enter"));
  3557. if (TAPIIsBadWritePtr(ppDialableAddress , sizeof(BSTR) ) )
  3558. {
  3559. LOG((TL_ERROR, "get_DialableAddress - bad pointer"));
  3560. return E_POINTER;
  3561. }
  3562. *ppDialableAddress = SysAllocString( m_szAddress );
  3563. if ( ( NULL == *ppDialableAddress ) && ( NULL != m_szAddress ) )
  3564. {
  3565. LOG((TL_TRACE, "SysAllocString Failed" ));
  3566. hr = E_OUTOFMEMORY;
  3567. }
  3568. LOG((TL_TRACE, "get_DialableAddress - Exit"));
  3569. return hr;
  3570. }
  3571. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3572. //
  3573. //
  3574. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3575. STDMETHODIMP
  3576. CAddress::put_MessageWaiting(
  3577. VARIANT_BOOL fMessageWaiting
  3578. )
  3579. {
  3580. HRESULT hr = S_OK;
  3581. AddressLineStruct * pLine = NULL;
  3582. LOG((TL_TRACE, "put_MessageWaiting - Enter"));
  3583. hr = FindOrOpenALine(
  3584. LINEMEDIAMODE_INTERACTIVEVOICE,
  3585. &pLine
  3586. );
  3587. if ( !SUCCEEDED(hr) )
  3588. {
  3589. LOG((TL_ERROR, "put_MessageWaiting - findoropen failed %lx", hr));
  3590. return hr;
  3591. }
  3592. hr = LineSetLineDevStatus(
  3593. &(pLine->t3Line),
  3594. LINEDEVSTATUSFLAGS_MSGWAIT,
  3595. fMessageWaiting?-1:0
  3596. );
  3597. if ( ((LONG)hr) < 0 )
  3598. {
  3599. LOG((TL_TRACE, "put_MessageWaiting failed sync - %lx", hr));
  3600. MaybeCloseALine( &pLine );
  3601. return hr;
  3602. }
  3603. // Wait for the async reply & map it's tapi2 code T3
  3604. hr = WaitForReply( hr );
  3605. MaybeCloseALine( &pLine );
  3606. if ( !SUCCEEDED(hr) )
  3607. {
  3608. LOG((TL_TRACE, "put_MessageWaiting failed async - %lx", hr));
  3609. return hr;
  3610. }
  3611. LOG((TL_TRACE, "put_MessageWaiting - Exit"));
  3612. return S_OK;
  3613. }
  3614. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3615. //
  3616. //
  3617. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3618. STDMETHODIMP
  3619. CAddress::get_MessageWaiting(
  3620. VARIANT_BOOL * pfMessageWaiting
  3621. )
  3622. {
  3623. HRESULT hr = S_OK;
  3624. LINEDEVSTATUS * pDevStatus;
  3625. AddressLineStruct * pLine;
  3626. LOG((TL_TRACE, "get_MessageWaiting - Enter"));
  3627. if (TAPIIsBadWritePtr(pfMessageWaiting , sizeof(VARIANT_BOOL) ) )
  3628. {
  3629. LOG((TL_ERROR, "get_MessageWaiting - bad pointer"));
  3630. return E_POINTER;
  3631. }
  3632. hr = FindOrOpenALine(
  3633. LINEMEDIAMODE_INTERACTIVEVOICE,
  3634. &pLine
  3635. );
  3636. if ( !SUCCEEDED(hr) )
  3637. {
  3638. LOG((TL_ERROR, "FindOrOpenALine failed - %lx", hr ));
  3639. return hr;
  3640. }
  3641. hr = LineGetLineDevStatus(
  3642. pLine->t3Line.hLine,
  3643. &pDevStatus
  3644. );
  3645. MaybeCloseALine( &pLine );
  3646. if ( !SUCCEEDED( hr ) )
  3647. {
  3648. LOG((TL_ERROR, "LineGetDevStatus failed - %lx", hr ));
  3649. return hr;
  3650. }
  3651. if ( pDevStatus->dwDevStatusFlags & LINEDEVSTATUSFLAGS_MSGWAIT )
  3652. {
  3653. *pfMessageWaiting = VARIANT_TRUE;
  3654. }
  3655. else
  3656. {
  3657. *pfMessageWaiting = VARIANT_FALSE;
  3658. }
  3659. ClientFree( pDevStatus );
  3660. if ( !SUCCEEDED(hr) )
  3661. {
  3662. LOG((TL_ERROR, "Bad pointer in get_MessageWaiting"));
  3663. return hr;
  3664. }
  3665. LOG((TL_TRACE, "get_MessageWaiting - Exit"));
  3666. return hr;
  3667. }
  3668. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3669. //
  3670. //
  3671. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3672. STDMETHODIMP
  3673. CAddress::put_DoNotDisturb(
  3674. VARIANT_BOOL fDoNotDisturb
  3675. )
  3676. {
  3677. HRESULT hr = S_OK;
  3678. LINEFORWARDLIST lfl;
  3679. HCALL hCall = NULL;
  3680. AddressLineStruct * pLine;
  3681. //
  3682. // do not disturb is acheived through calling
  3683. // lineforward with NULL for the dest address
  3684. //
  3685. LOG((TL_TRACE, "put_DoNotDisturb - Enter"));
  3686. //
  3687. // if we are setting dnd, create a lineforwardlist
  3688. // structure
  3689. //
  3690. if ( fDoNotDisturb )
  3691. {
  3692. ZeroMemory(
  3693. &lfl,
  3694. sizeof( LINEFORWARDLIST )
  3695. );
  3696. lfl.dwTotalSize = sizeof( LINEFORWARDLIST );
  3697. lfl.dwNumEntries = 1;
  3698. //
  3699. // there is only one item, and the dest address
  3700. // is NULL
  3701. //
  3702. lfl.ForwardList[0].dwForwardMode = LINEFORWARDMODE_UNCOND;
  3703. }
  3704. //
  3705. // get a line to use
  3706. //
  3707. hr = FindOrOpenALine(
  3708. LINEMEDIAMODE_INTERACTIVEVOICE,
  3709. &pLine
  3710. );
  3711. if ( !SUCCEEDED(hr) )
  3712. {
  3713. LOG((TL_ERROR, "put_DoNotDisturb - FindOrOpen failed %lx", hr));
  3714. return hr;
  3715. }
  3716. //
  3717. // call line forward
  3718. // if fDND is false, the LINEFORWARDLIST structure pointer
  3719. // should be NULL. This clears any fowarding on that
  3720. // line.
  3721. //
  3722. hr = LineForward(
  3723. &(pLine->t3Line),
  3724. m_dwAddressID,
  3725. fDoNotDisturb?&lfl:NULL,
  3726. 0,
  3727. &hCall
  3728. );
  3729. if ( ((long)hr) < 0 )
  3730. {
  3731. LOG((TL_ERROR, "put_DND - linefoward failed sync - %lx", hr));
  3732. MaybeCloseALine( &pLine );
  3733. return hr;
  3734. }
  3735. // Wait for the async reply & map it's tapi2 code T3
  3736. hr = WaitForReply( hr );
  3737. if ( NULL != hCall )
  3738. {
  3739. T3CALL t3Call;
  3740. HRESULT hr2;
  3741. t3Call.hCall = hCall;
  3742. hr2 = LineDrop(
  3743. t3Call.hCall,
  3744. NULL,
  3745. 0
  3746. );
  3747. if ( ((long)hr2) > 0 )
  3748. {
  3749. hr2 = WaitForReply( hr2 ) ;
  3750. }
  3751. hr2 = LineDeallocateCall(
  3752. t3Call.hCall
  3753. );
  3754. }
  3755. //
  3756. // we are no longer using the line
  3757. //
  3758. MaybeCloseALine( &pLine );
  3759. if ( !SUCCEEDED(hr) )
  3760. {
  3761. LOG((TL_ERROR, "put_DND - linefoware failed async - %lx", hr));
  3762. return hr;
  3763. }
  3764. LOG((TL_TRACE, "put_DoNotDisturb - Exit"));
  3765. return hr;
  3766. }
  3767. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3768. //
  3769. //
  3770. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3771. STDMETHODIMP
  3772. CAddress::get_DoNotDisturb(
  3773. VARIANT_BOOL * pfDoNotDisturb
  3774. )
  3775. {
  3776. HRESULT hr = S_OK;
  3777. AddressLineStruct * pLine;
  3778. LINEADDRESSSTATUS * pAddressStatus;
  3779. LINEDEVSTATUS * pDevStatus = NULL;
  3780. //
  3781. // donotdisturb is implemented through
  3782. // line forward.
  3783. // to get_DND, check the forward state
  3784. //
  3785. LOG((TL_TRACE, "get_DoNotDisturb - Enter"));
  3786. if (TAPIIsBadWritePtr(pfDoNotDisturb , sizeof(VARIANT_BOOL) ) )
  3787. {
  3788. LOG((TL_ERROR, "pfDoNotDisturb - bad pointer"));
  3789. return E_POINTER;
  3790. }
  3791. hr = FindOrOpenALine(
  3792. LINEMEDIAMODE_INTERACTIVEVOICE,
  3793. &pLine
  3794. );
  3795. if ( !SUCCEEDED(hr) )
  3796. {
  3797. LOG((TL_ERROR, "get_DND - FindOrOpen failed %lx", hr));
  3798. return hr;
  3799. }
  3800. hr = LineGetLineDevStatus(
  3801. pLine->t3Line.hLine,
  3802. &pDevStatus
  3803. );
  3804. if ( !SUCCEEDED(hr) )
  3805. {
  3806. MaybeCloseALine( &pLine );
  3807. return hr;
  3808. }
  3809. if ( !(pDevStatus->dwLineFeatures & LINEFEATURE_FORWARD ) )
  3810. {
  3811. LOG((TL_INFO, "get_DND - not supported"));
  3812. MaybeCloseALine( &pLine );
  3813. if(pDevStatus != NULL)
  3814. {
  3815. ClientFree(pDevStatus);
  3816. }
  3817. return TAPI_E_NOTSUPPORTED;
  3818. }
  3819. // finished with pDevStatus
  3820. if(pDevStatus != NULL)
  3821. {
  3822. ClientFree(pDevStatus);
  3823. }
  3824. //
  3825. // get the addresss status
  3826. //
  3827. hr = LineGetAddressStatus(
  3828. &(pLine->t3Line),
  3829. m_dwAddressID,
  3830. &pAddressStatus
  3831. );
  3832. if ( !SUCCEEDED(hr) )
  3833. {
  3834. LOG((TL_ERROR, "get_DND - LineGetAddressStatus failed - %lx", hr));
  3835. MaybeCloseALine( &pLine );
  3836. return hr;
  3837. }
  3838. //
  3839. // initialize to false
  3840. //
  3841. *pfDoNotDisturb = VARIANT_FALSE;
  3842. if ( !SUCCEEDED(hr) )
  3843. {
  3844. LOG((TL_ERROR, "get_DND - bad pointer"));
  3845. return E_OUTOFMEMORY;
  3846. }
  3847. //
  3848. // if there is forwarding
  3849. //
  3850. if ( 0 != pAddressStatus->dwForwardNumEntries )
  3851. {
  3852. LINEFORWARD * pfl;
  3853. pfl = (LINEFORWARD *) (((LPBYTE)pAddressStatus) + pAddressStatus->dwForwardOffset);
  3854. //
  3855. // and the dest address is NULL
  3856. //
  3857. if ( 0 == pfl->dwDestAddressOffset )
  3858. {
  3859. //
  3860. // DND is set
  3861. //
  3862. *pfDoNotDisturb = VARIANT_TRUE;
  3863. }
  3864. }
  3865. MaybeCloseALine( &pLine );
  3866. ClientFree( pAddressStatus );
  3867. LOG((TL_TRACE, "get_DoNotDisturb - Exit"));
  3868. return S_OK;
  3869. }
  3870. //
  3871. // itaddresscapabilities
  3872. //
  3873. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3874. //
  3875. //
  3876. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  3877. STDMETHODIMP
  3878. CAddress::get_AddressCapability(
  3879. ADDRESS_CAPABILITY AddressCap,
  3880. long * plCapability
  3881. )
  3882. {
  3883. HRESULT hr = S_OK;
  3884. LOG((TL_TRACE, "get_AddressCapability - Enter"));
  3885. if ( TAPIIsBadWritePtr( plCapability, sizeof(long) ) )
  3886. {
  3887. LOG((TL_ERROR, "get_AddressCapability - bad pointer"));
  3888. return E_POINTER;
  3889. }
  3890. Lock();
  3891. hr = UpdateAddressCaps();
  3892. if ( !SUCCEEDED(hr) )
  3893. {
  3894. LOG((TL_ERROR, "get_AddressCapability - could not get addresscaps"));
  3895. Unlock();
  3896. return E_UNEXPECTED;
  3897. }
  3898. switch (AddressCap)
  3899. {
  3900. case AC_LINEID:
  3901. *plCapability = m_dwDeviceID;
  3902. break;
  3903. case AC_ADDRESSID:
  3904. *plCapability = m_dwAddressID;
  3905. break;
  3906. case AC_ADDRESSTYPES:
  3907. case AC_BEARERMODES:
  3908. case AC_MONITORDIGITSUPPORT:
  3909. case AC_GENERATEDIGITSUPPORT:
  3910. case AC_GENERATETONEMODES:
  3911. case AC_GENERATETONEMAXNUMFREQ:
  3912. case AC_MONITORTONEMAXNUMFREQ:
  3913. case AC_MONITORTONEMAXNUMENTRIES:
  3914. case AC_DEVCAPFLAGS:
  3915. case AC_ANSWERMODES:
  3916. case AC_LINEFEATURES:
  3917. case AC_SETTABLEDEVSTATUS:
  3918. case AC_PERMANENTDEVICEID:
  3919. case AC_GATHERDIGITSMINTIMEOUT:
  3920. case AC_GATHERDIGITSMAXTIMEOUT:
  3921. case AC_GENERATEDIGITMINDURATION:
  3922. case AC_GENERATEDIGITMAXDURATION:
  3923. case AC_GENERATEDIGITDEFAULTDURATION:
  3924. {
  3925. hr = UpdateLineDevCaps();
  3926. if ( !SUCCEEDED(hr) )
  3927. {
  3928. LOG((TL_ERROR, "get_AddressCap - could not get devcaps - %lx", hr));
  3929. Unlock();
  3930. return hr;
  3931. }
  3932. switch (AddressCap)
  3933. {
  3934. case AC_ADDRESSTYPES:
  3935. if ( m_dwAPIVersion >= TAPI_VERSION3_0 )
  3936. {
  3937. *plCapability = m_pDevCaps->dwAddressTypes;
  3938. }
  3939. else
  3940. {
  3941. *plCapability = LINEADDRESSTYPE_PHONENUMBER;
  3942. }
  3943. break;
  3944. case AC_BEARERMODES:
  3945. *plCapability = m_pDevCaps->dwBearerModes;
  3946. break;
  3947. case AC_MONITORDIGITSUPPORT:
  3948. *plCapability = m_pDevCaps->dwMonitorDigitModes;
  3949. break;
  3950. case AC_GENERATEDIGITSUPPORT:
  3951. *plCapability = m_pDevCaps->dwGenerateDigitModes;
  3952. break;
  3953. case AC_GENERATETONEMODES:
  3954. *plCapability = m_pDevCaps->dwGenerateToneModes;
  3955. break;
  3956. case AC_GENERATETONEMAXNUMFREQ:
  3957. *plCapability = m_pDevCaps->dwGenerateToneMaxNumFreq;
  3958. break;
  3959. case AC_MONITORTONEMAXNUMFREQ:
  3960. *plCapability = m_pDevCaps->dwMonitorToneMaxNumFreq;
  3961. break;
  3962. case AC_MONITORTONEMAXNUMENTRIES:
  3963. *plCapability = m_pDevCaps->dwMonitorToneMaxNumEntries;
  3964. break;
  3965. case AC_DEVCAPFLAGS:
  3966. *plCapability = m_pDevCaps->dwDevCapFlags;
  3967. break;
  3968. case AC_ANSWERMODES:
  3969. *plCapability = m_pDevCaps->dwAnswerMode;
  3970. break;
  3971. case AC_LINEFEATURES:
  3972. *plCapability = m_pDevCaps->dwLineFeatures;
  3973. break;
  3974. case AC_SETTABLEDEVSTATUS:
  3975. if ( m_dwAPIVersion >= TAPI_VERSION2_0 )
  3976. {
  3977. *plCapability = m_pDevCaps->dwSettableDevStatus;
  3978. }
  3979. else
  3980. hr = TAPI_E_NOTSUPPORTED;
  3981. break;
  3982. case AC_PERMANENTDEVICEID:
  3983. *plCapability = m_pDevCaps->dwPermanentLineID;
  3984. break;
  3985. case AC_GATHERDIGITSMINTIMEOUT:
  3986. *plCapability = m_pDevCaps->dwGatherDigitsMinTimeout;
  3987. break;
  3988. case AC_GATHERDIGITSMAXTIMEOUT:
  3989. *plCapability = m_pDevCaps->dwGatherDigitsMaxTimeout;
  3990. break;
  3991. case AC_GENERATEDIGITMINDURATION:
  3992. *plCapability = m_pDevCaps->MinDialParams.dwDigitDuration;
  3993. break;
  3994. case AC_GENERATEDIGITMAXDURATION:
  3995. *plCapability = m_pDevCaps->MaxDialParams.dwDigitDuration;
  3996. break;
  3997. case AC_GENERATEDIGITDEFAULTDURATION:
  3998. *plCapability = m_pDevCaps->DefaultDialParams.dwDigitDuration;
  3999. break;
  4000. }
  4001. break;
  4002. }
  4003. case AC_MAXACTIVECALLS:
  4004. case AC_MAXONHOLDCALLS:
  4005. case AC_MAXONHOLDPENDINGCALLS:
  4006. case AC_MAXNUMCONFERENCE:
  4007. case AC_MAXNUMTRANSCONF:
  4008. case AC_PARKSUPPORT:
  4009. case AC_CALLERIDSUPPORT:
  4010. case AC_CALLEDIDSUPPORT:
  4011. case AC_CONNECTEDIDSUPPORT:
  4012. case AC_REDIRECTIONIDSUPPORT:
  4013. case AC_REDIRECTINGIDSUPPORT:
  4014. case AC_ADDRESSCAPFLAGS:
  4015. case AC_CALLFEATURES1:
  4016. case AC_CALLFEATURES2:
  4017. case AC_REMOVEFROMCONFCAPS:
  4018. case AC_REMOVEFROMCONFSTATE:
  4019. case AC_TRANSFERMODES:
  4020. case AC_ADDRESSFEATURES:
  4021. case AC_PREDICTIVEAUTOTRANSFERSTATES:
  4022. case AC_MAXCALLDATASIZE:
  4023. case AC_FORWARDMODES:
  4024. case AC_MAXFORWARDENTRIES:
  4025. case AC_MAXSPECIFICENTRIES:
  4026. case AC_MINFWDNUMRINGS:
  4027. case AC_MAXFWDNUMRINGS:
  4028. case AC_MAXCALLCOMPLETIONS:
  4029. case AC_CALLCOMPLETIONCONDITIONS:
  4030. case AC_CALLCOMPLETIONMODES:
  4031. {
  4032. hr = UpdateAddressCaps();
  4033. if ( !SUCCEEDED(hr) )
  4034. {
  4035. LOG((TL_ERROR, "get_AddressCaps - could not update caps - %lx", hr));
  4036. Unlock();
  4037. return hr;
  4038. }
  4039. switch (AddressCap)
  4040. {
  4041. case AC_MAXACTIVECALLS:
  4042. *plCapability = m_pAddressCaps->dwMaxNumActiveCalls;
  4043. break;
  4044. case AC_MAXONHOLDCALLS:
  4045. *plCapability = m_pAddressCaps->dwMaxNumOnHoldCalls;
  4046. break;
  4047. case AC_MAXONHOLDPENDINGCALLS:
  4048. *plCapability = m_pAddressCaps->dwMaxNumOnHoldPendingCalls;
  4049. break;
  4050. case AC_MAXNUMCONFERENCE:
  4051. *plCapability = m_pAddressCaps->dwMaxNumConference;
  4052. break;
  4053. case AC_MAXNUMTRANSCONF:
  4054. *plCapability = m_pAddressCaps->dwMaxNumTransConf;
  4055. break;
  4056. case AC_PARKSUPPORT:
  4057. *plCapability = m_pAddressCaps->dwParkModes;
  4058. break;
  4059. case AC_CALLERIDSUPPORT:
  4060. *plCapability = m_pAddressCaps->dwCallerIDFlags;
  4061. break;
  4062. case AC_CALLEDIDSUPPORT:
  4063. *plCapability = m_pAddressCaps->dwCalledIDFlags;
  4064. break;
  4065. case AC_CONNECTEDIDSUPPORT:
  4066. *plCapability = m_pAddressCaps->dwConnectedIDFlags;
  4067. break;
  4068. case AC_REDIRECTIONIDSUPPORT:
  4069. *plCapability = m_pAddressCaps->dwRedirectionIDFlags;
  4070. break;
  4071. case AC_REDIRECTINGIDSUPPORT:
  4072. *plCapability = m_pAddressCaps->dwRedirectingIDFlags;
  4073. break;
  4074. case AC_ADDRESSCAPFLAGS:
  4075. *plCapability = m_pAddressCaps->dwAddrCapFlags;
  4076. break;
  4077. case AC_CALLFEATURES1:
  4078. *plCapability = m_pAddressCaps->dwCallFeatures;
  4079. break;
  4080. case AC_CALLFEATURES2:
  4081. if ( m_dwAPIVersion < TAPI_VERSION2_0 )
  4082. {
  4083. hr = TAPI_E_NOTSUPPORTED;
  4084. }
  4085. else
  4086. {
  4087. *plCapability = m_pAddressCaps->dwCallFeatures2;
  4088. }
  4089. break;
  4090. case AC_REMOVEFROMCONFCAPS:
  4091. *plCapability = m_pAddressCaps->dwRemoveFromConfCaps;
  4092. break;
  4093. case AC_REMOVEFROMCONFSTATE:
  4094. *plCapability = m_pAddressCaps->dwRemoveFromConfState;
  4095. break;
  4096. case AC_TRANSFERMODES:
  4097. *plCapability = m_pAddressCaps->dwTransferModes;
  4098. break;
  4099. case AC_ADDRESSFEATURES:
  4100. *plCapability = m_pAddressCaps->dwAddressFeatures;
  4101. break;
  4102. case AC_PREDICTIVEAUTOTRANSFERSTATES:
  4103. if ( m_dwAPIVersion < TAPI_VERSION2_0 )
  4104. {
  4105. hr = TAPI_E_NOTSUPPORTED;
  4106. }
  4107. else
  4108. {
  4109. *plCapability = m_pAddressCaps->dwPredictiveAutoTransferStates;
  4110. }
  4111. break;
  4112. case AC_MAXCALLDATASIZE:
  4113. if ( m_dwAPIVersion < TAPI_VERSION2_0 )
  4114. {
  4115. hr = TAPI_E_NOTSUPPORTED;
  4116. }
  4117. else
  4118. {
  4119. *plCapability = m_pAddressCaps->dwMaxCallDataSize;
  4120. }
  4121. break;
  4122. case AC_FORWARDMODES:
  4123. *plCapability = m_pAddressCaps->dwForwardModes;
  4124. break;
  4125. case AC_MAXFORWARDENTRIES:
  4126. *plCapability = m_pAddressCaps->dwMaxForwardEntries;
  4127. break;
  4128. case AC_MAXSPECIFICENTRIES:
  4129. *plCapability = m_pAddressCaps->dwMaxSpecificEntries;
  4130. break;
  4131. case AC_MINFWDNUMRINGS:
  4132. *plCapability = m_pAddressCaps->dwMinFwdNumRings;
  4133. break;
  4134. case AC_MAXFWDNUMRINGS:
  4135. *plCapability = m_pAddressCaps->dwMaxFwdNumRings;
  4136. break;
  4137. case AC_MAXCALLCOMPLETIONS:
  4138. *plCapability = m_pAddressCaps->dwMaxCallCompletions;
  4139. break;
  4140. case AC_CALLCOMPLETIONCONDITIONS:
  4141. *plCapability = m_pAddressCaps->dwCallCompletionConds;
  4142. break;
  4143. case AC_CALLCOMPLETIONMODES:
  4144. *plCapability = m_pAddressCaps->dwCallCompletionModes;
  4145. break;
  4146. }
  4147. break;
  4148. }
  4149. default:
  4150. LOG((TL_ERROR, "get_AddressCapability - bad addrcap"));
  4151. Unlock();
  4152. return E_INVALIDARG;
  4153. }
  4154. LOG((TL_TRACE, "get_AddressCapability - Exit - result - %lx", hr));
  4155. Unlock();
  4156. return hr;
  4157. }
  4158. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4159. //
  4160. //
  4161. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4162. STDMETHODIMP
  4163. CAddress::get_AddressCapabilityString(
  4164. ADDRESS_CAPABILITY_STRING AddressCapString,
  4165. BSTR * ppCapabilityString
  4166. )
  4167. {
  4168. HRESULT hr = S_OK;
  4169. LOG((TL_TRACE, "get_AddressCapabilityString - Enter"));
  4170. if ( TAPIIsBadWritePtr( ppCapabilityString, sizeof(BSTR) ) )
  4171. {
  4172. LOG((TL_ERROR, "get_AddressCapabilityString - bad pointer"));
  4173. return E_POINTER;
  4174. }
  4175. *ppCapabilityString = NULL;
  4176. Lock();
  4177. //
  4178. // note on:
  4179. // ACS_ADDRESSDEVICESPECIFIC
  4180. // ACS_LINEDEVICESPECIFIC:
  4181. // ACS_PROVIDERSPECIFIC:
  4182. // ACS_SWITCHSPECIFIC:
  4183. //
  4184. // These buffers in LINEDEVCAPS and LINEADDRESSCAPS may or may not be strings. However
  4185. // in TAPI 3, we define them as strings.
  4186. //
  4187. // The algorithm for moving them from the tapi 2 structure to the tapi 3 string is:
  4188. //
  4189. // if the size of the buffer is evenly divisible by sizeof(WCHAR)
  4190. // then it's definitely not a string (we only support WCHAR)
  4191. // so we will just copy the buffer straight into the returned string, and append a
  4192. // NULL at the end
  4193. // else
  4194. // if the buffer already has a null at the end, don't copy it, because SysAllocStringByteLen
  4195. // always appends a NULL and double NULLs frighten VB.
  4196. //
  4197. // It can still be a "buffer" and not a string even if it is divisible by sizeof WCHAR, but for
  4198. // this it does matter, since we are using SysAllocStringByteLen to copy the buffer no
  4199. // matter what
  4200. //
  4201. switch (AddressCapString)
  4202. {
  4203. case ACS_ADDRESSDEVICESPECIFIC:
  4204. hr = UpdateAddressCaps();
  4205. if ( !SUCCEEDED(hr) )
  4206. {
  4207. Unlock();
  4208. return hr;
  4209. }
  4210. if ( m_pAddressCaps->dwDevSpecificSize != 0 )
  4211. {
  4212. LPWSTR pHold;
  4213. DWORD dwSize;
  4214. dwSize = m_pAddressCaps->dwDevSpecificSize;
  4215. //
  4216. // is size divisible by sizeof(WCHAR)?
  4217. //
  4218. if (0 == (dwSize % sizeof(WCHAR)))
  4219. {
  4220. //
  4221. // yes - get to the last character in the string
  4222. //
  4223. pHold = (LPWSTR)(((LPBYTE)(m_pAddressCaps)) + m_pAddressCaps->dwDevSpecificOffset),
  4224. (dwSize-sizeof(WCHAR))/sizeof(WCHAR);
  4225. //
  4226. // is the last character a NULL?
  4227. //
  4228. if (*pHold == NULL)
  4229. {
  4230. //
  4231. // yes, so don't copy it
  4232. //
  4233. dwSize-=sizeof(WCHAR);
  4234. }
  4235. }
  4236. //
  4237. // Alloc and copy into return string. SysAllocStringByteLen always
  4238. // appends a NULL
  4239. //
  4240. *ppCapabilityString = SysAllocStringByteLen(
  4241. (LPSTR)(((LPBYTE)(m_pAddressCaps)) + m_pAddressCaps->dwDevSpecificOffset),
  4242. dwSize
  4243. );
  4244. if ( NULL == *ppCapabilityString )
  4245. {
  4246. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4247. hr = E_OUTOFMEMORY;
  4248. }
  4249. }
  4250. break;
  4251. case ACS_PROTOCOL:
  4252. case ACS_LINEDEVICESPECIFIC:
  4253. case ACS_PROVIDERSPECIFIC:
  4254. case ACS_SWITCHSPECIFIC:
  4255. case ACS_PERMANENTDEVICEGUID:
  4256. {
  4257. hr = UpdateLineDevCaps();
  4258. if ( !SUCCEEDED(hr) )
  4259. {
  4260. Unlock();
  4261. return hr;
  4262. }
  4263. switch (AddressCapString)
  4264. {
  4265. case ACS_PROTOCOL:
  4266. {
  4267. LPWSTR pwstr;
  4268. if ( m_dwAPIVersion >= TAPI_VERSION3_0 )
  4269. {
  4270. IID iid;
  4271. iid = m_pDevCaps->ProtocolGuid;
  4272. StringFromIID(iid, &pwstr);
  4273. }
  4274. else
  4275. {
  4276. StringFromIID( TAPIPROTOCOL_PSTN, &pwstr );
  4277. }
  4278. *ppCapabilityString = SysAllocString( pwstr );
  4279. if ( NULL == *ppCapabilityString )
  4280. {
  4281. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4282. hr = E_OUTOFMEMORY;
  4283. }
  4284. CoTaskMemFree( pwstr );
  4285. break;
  4286. }
  4287. case ACS_LINEDEVICESPECIFIC:
  4288. if ( m_pDevCaps->dwDevSpecificSize != 0 )
  4289. {
  4290. LPWSTR pHold;
  4291. DWORD dwSize;
  4292. dwSize = m_pDevCaps->dwDevSpecificSize;
  4293. //
  4294. // is size divisible by sizeof(WCHAR)?
  4295. //
  4296. if (0 == (dwSize % sizeof(WCHAR)))
  4297. {
  4298. //
  4299. // yes - get to the last character in the string
  4300. //
  4301. pHold = (LPWSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwDevSpecificOffset) +
  4302. (dwSize-sizeof(WCHAR))/sizeof(WCHAR);
  4303. //
  4304. // is the last character a NULL?
  4305. //
  4306. if (*pHold == NULL)
  4307. {
  4308. //
  4309. // yes, so don't copy it
  4310. //
  4311. dwSize-=sizeof(WCHAR);
  4312. }
  4313. }
  4314. //
  4315. // Alloc and copy into return string. SysAllocStringByteLen always
  4316. // appends a NULL
  4317. //
  4318. *ppCapabilityString = SysAllocStringByteLen(
  4319. (LPSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwDevSpecificOffset),
  4320. dwSize
  4321. );
  4322. if ( NULL == *ppCapabilityString )
  4323. {
  4324. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4325. hr = E_OUTOFMEMORY;
  4326. }
  4327. }
  4328. break;
  4329. case ACS_PROVIDERSPECIFIC:
  4330. if ( m_pDevCaps->dwProviderInfoSize != 0 )
  4331. {
  4332. LPWSTR pHold;
  4333. DWORD dwSize;
  4334. dwSize = m_pDevCaps->dwProviderInfoSize;
  4335. //
  4336. // is size divisible by sizeof(WCHAR)?
  4337. //
  4338. if (0 == (dwSize % sizeof(WCHAR)))
  4339. {
  4340. //
  4341. // yes - get to the last character in the string
  4342. //
  4343. pHold = (LPWSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwProviderInfoOffset) +
  4344. (dwSize-sizeof(WCHAR))/sizeof(WCHAR);
  4345. //
  4346. // is the last character a NULL?
  4347. //
  4348. if (*pHold == NULL)
  4349. {
  4350. //
  4351. // yes, so don't copy it
  4352. //
  4353. dwSize-=sizeof(WCHAR);
  4354. }
  4355. }
  4356. //
  4357. // Alloc and copy into return string. SysAllocStringByteLen always
  4358. // appends a NULL
  4359. //
  4360. *ppCapabilityString = SysAllocStringByteLen(
  4361. (LPSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwProviderInfoOffset),
  4362. dwSize
  4363. );
  4364. if ( NULL == *ppCapabilityString )
  4365. {
  4366. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4367. hr = E_OUTOFMEMORY;
  4368. }
  4369. }
  4370. break;
  4371. case ACS_SWITCHSPECIFIC:
  4372. if ( m_pDevCaps->dwSwitchInfoSize != 0 )
  4373. {
  4374. LPWSTR pHold;
  4375. DWORD dwSize;
  4376. dwSize = m_pDevCaps->dwSwitchInfoSize;
  4377. //
  4378. // is size divisible by sizeof(WCHAR)?
  4379. //
  4380. if (0 == (dwSize % sizeof(WCHAR)))
  4381. {
  4382. //
  4383. // yes - get to the last character in the string
  4384. //
  4385. pHold = (LPWSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwSwitchInfoOffset) +
  4386. (dwSize-sizeof(WCHAR))/sizeof(WCHAR);
  4387. //
  4388. // is the last character a NULL?
  4389. //
  4390. if (*pHold == NULL)
  4391. {
  4392. //
  4393. // yes, so don't copy it
  4394. //
  4395. dwSize-=sizeof(WCHAR);
  4396. }
  4397. }
  4398. //
  4399. // Alloc and copy into return string. SysAllocStringByteLen always
  4400. // appends a NULL
  4401. //
  4402. *ppCapabilityString = SysAllocStringByteLen(
  4403. (LPSTR)(((LPBYTE)(m_pDevCaps)) + m_pDevCaps->dwSwitchInfoOffset),
  4404. dwSize
  4405. );
  4406. if ( NULL == *ppCapabilityString )
  4407. {
  4408. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4409. hr = E_OUTOFMEMORY;
  4410. }
  4411. }
  4412. break;
  4413. case ACS_PERMANENTDEVICEGUID:
  4414. {
  4415. LPWSTR pwstrGUIDText;
  4416. if ( m_dwAPIVersion >= TAPI_VERSION2_2 )
  4417. {
  4418. IID iid;
  4419. iid = m_pDevCaps->PermanentLineGuid;
  4420. StringFromIID(iid, &pwstrGUIDText);
  4421. *ppCapabilityString = SysAllocString( pwstrGUIDText );
  4422. if ( NULL == *ppCapabilityString )
  4423. {
  4424. LOG((TL_ERROR, "get_AddressCapabilityString - SysAllocString Failed"));
  4425. hr = E_OUTOFMEMORY;
  4426. }
  4427. CoTaskMemFree( pwstrGUIDText );
  4428. }
  4429. else
  4430. {
  4431. // return with NULL string & error code
  4432. hr = TAPI_E_NOTSUPPORTED;
  4433. }
  4434. break;
  4435. }
  4436. default:
  4437. break;
  4438. }
  4439. break;
  4440. }
  4441. default:
  4442. LOG((TL_ERROR, "get_AddressCapabilityString - invalid cap"));
  4443. Unlock();
  4444. return E_INVALIDARG;
  4445. }
  4446. LOG((TL_TRACE, "get_AddressCapabilityString - Exit - result - %lx", hr));
  4447. Unlock();
  4448. return hr;
  4449. }
  4450. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4451. //
  4452. //
  4453. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4454. HRESULT CreateBstrCollection(
  4455. IN BSTR * pBstr,
  4456. IN DWORD dwCount,
  4457. OUT VARIANT * pVariant
  4458. )
  4459. {
  4460. //
  4461. // create the collection object
  4462. //
  4463. CComObject<CTapiBstrCollection> * pCollection;
  4464. HRESULT hr = CComObject<CTapiBstrCollection>::CreateInstance( &pCollection );
  4465. if ( FAILED(hr) )
  4466. {
  4467. LOG((TL_ERROR, "CreateBstrCollection - "
  4468. "can't create collection - exit 0x%lx", hr));
  4469. return hr;
  4470. }
  4471. //
  4472. // get the Collection's IDispatch interface
  4473. //
  4474. IDispatch * pDispatch;
  4475. hr = pCollection->_InternalQueryInterface(__uuidof(IDispatch),
  4476. (void **) &pDispatch );
  4477. if ( FAILED(hr) )
  4478. {
  4479. LOG((TL_ERROR, "CreateBstrCollection - "
  4480. "QI for IDispatch on collection failed - exit 0x%lx", hr));
  4481. delete pCollection;
  4482. return hr;
  4483. }
  4484. //
  4485. // Init the collection using an iterator -- pointers to the beginning and
  4486. // the ending element plus one.
  4487. //
  4488. hr = pCollection->Initialize( dwCount,
  4489. pBstr,
  4490. pBstr + dwCount);
  4491. if ( FAILED(hr) )
  4492. {
  4493. LOG((TL_ERROR, "CreateBstrCollection - "
  4494. "Initialize on collection failed - exit 0x%lx", hr));
  4495. pDispatch->Release();
  4496. return hr;
  4497. }
  4498. //
  4499. // put the IDispatch interface pointer into the variant
  4500. //
  4501. LOG((TL_ERROR, "CreateBstrCollection - "
  4502. "placing IDispatch value %p in variant", pDispatch));
  4503. VariantInit(pVariant);
  4504. pVariant->vt = VT_DISPATCH;
  4505. pVariant->pdispVal = pDispatch;
  4506. LOG((TL_TRACE, "CreateBstrCollection - exit S_OK"));
  4507. return S_OK;
  4508. }
  4509. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4510. //
  4511. //
  4512. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4513. HRESULT CreateBstrEnumerator(
  4514. IN BSTR * begin,
  4515. IN BSTR * end,
  4516. OUT IEnumBstr ** ppIEnum
  4517. )
  4518. {
  4519. typedef CSafeComEnum<IEnumBstr, &__uuidof(IEnumBstr), BSTR, _CopyBSTR> CEnumerator;
  4520. HRESULT hr;
  4521. CComObject<CEnumerator> *pEnum = NULL;
  4522. hr = CComObject<CEnumerator>::CreateInstance(&pEnum);
  4523. if (pEnum == NULL)
  4524. {
  4525. LOG((TL_ERROR, "CreateBstrEnumerator - "
  4526. "Could not create enumerator object, 0x%lx", hr));
  4527. return hr;
  4528. }
  4529. IEnumBstr * pIEnum;
  4530. hr = pEnum->_InternalQueryInterface(
  4531. __uuidof(IEnumBstr),
  4532. (void**)&pIEnum
  4533. );
  4534. if (FAILED(hr))
  4535. {
  4536. LOG((TL_ERROR, "CreateBstrEnumerator - "
  4537. "query enum interface failed, 0x%lx", hr));
  4538. delete pEnum;
  4539. return hr;
  4540. }
  4541. hr = pEnum->Init(begin, end, NULL, AtlFlagCopy);
  4542. if (FAILED(hr))
  4543. {
  4544. LOG((TL_ERROR, "CreateBstrEnumerator - "
  4545. "init enumerator object failed, 0x%lx", hr));
  4546. pIEnum->Release();
  4547. return hr;
  4548. }
  4549. *ppIEnum = pIEnum;
  4550. LOG((TL_TRACE, "CreateBstrEnumerator - exit S_OK"));
  4551. return S_OK;
  4552. }
  4553. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4554. //
  4555. //
  4556. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4557. STDMETHODIMP
  4558. CAddress::get_CallTreatments(VARIANT * pVariant )
  4559. {
  4560. HRESULT hr = E_NOTIMPL;
  4561. LOG((TL_TRACE, "get_CallTreatments - Enter"));
  4562. LOG((TL_TRACE, "get_CallTreatments - Exit - result - %lx", hr));
  4563. return hr;
  4564. }
  4565. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4566. //
  4567. //
  4568. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4569. STDMETHODIMP
  4570. CAddress::EnumerateCallTreatments(IEnumBstr ** ppEnumCallTreatment )
  4571. {
  4572. HRESULT hr = E_NOTIMPL;
  4573. LOG((TL_TRACE, "EnumerateCallTreatments - Enter"));
  4574. LOG((TL_TRACE, "EnumerateCallTreatments - Exit - result - %lx", hr));
  4575. return hr;
  4576. }
  4577. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4578. //
  4579. //
  4580. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4581. STDMETHODIMP
  4582. CAddress::get_CompletionMessages(VARIANT * pVariant)
  4583. {
  4584. HRESULT hr = E_NOTIMPL;
  4585. LOG((TL_TRACE, "get_CompletionMessages - Enter"));
  4586. LOG((TL_TRACE, "get_CompletionMessages - Exit - result - %lx", hr));
  4587. return hr;
  4588. }
  4589. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4590. //
  4591. //
  4592. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4593. STDMETHODIMP
  4594. CAddress::EnumerateCompletionMessages(IEnumBstr ** ppEnumCompletionMessage)
  4595. {
  4596. HRESULT hr = E_NOTIMPL;
  4597. LOG((TL_TRACE, "EnumerateCompletionMessages - Enter"));
  4598. LOG((TL_TRACE, "EnumerateCompletionMessages - Exit - result - %lx", hr));
  4599. return hr;
  4600. }
  4601. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4602. //
  4603. //
  4604. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4605. STDMETHODIMP
  4606. CAddress::get_DeviceClasses(VARIANT * pVariant)
  4607. {
  4608. HRESULT hr = S_OK;
  4609. LOG((TL_TRACE, "get_DeviceClasses - Enter"));
  4610. //
  4611. // Check arguments.
  4612. //
  4613. if ( TAPIIsBadWritePtr( pVariant, sizeof( VARIANT ) ) )
  4614. {
  4615. LOG((TL_ERROR, "get_DeviceClasses - bad pointer"));
  4616. return E_POINTER;
  4617. }
  4618. Lock();
  4619. hr = UpdateLineDevCaps();
  4620. if ( SUCCEEDED(hr) )
  4621. {
  4622. if ( (m_dwAPIVersion >= TAPI_VERSION2_0) && (0 != m_pDevCaps->dwDeviceClassesOffset) )
  4623. {
  4624. PWSTR pszDevices;
  4625. DWORD dwNumDeviceClasses;
  4626. //
  4627. // first count the device classes
  4628. //
  4629. dwNumDeviceClasses = 0;
  4630. pszDevices = (PWSTR)( ( (PBYTE)m_pDevCaps ) + m_pDevCaps->dwDeviceClassesOffset );
  4631. while (NULL != *pszDevices)
  4632. {
  4633. dwNumDeviceClasses++;
  4634. pszDevices += (lstrlenW(pszDevices) + 1 );
  4635. }
  4636. //
  4637. // allocate an array of BSTR pointers
  4638. //
  4639. BSTR *DeviceClasses =
  4640. (BSTR *)ClientAlloc(sizeof(BSTR *) * dwNumDeviceClasses);
  4641. if (DeviceClasses == NULL)
  4642. {
  4643. LOG((TL_ERROR, "get_DeviceClasses - out of memory"));
  4644. Unlock();
  4645. return E_OUTOFMEMORY;
  4646. }
  4647. //
  4648. // allocate all the BSTRs, copying the device class names
  4649. //
  4650. DWORD dwCount = 0;
  4651. pszDevices = (PWSTR)( ( (PBYTE)m_pDevCaps ) + m_pDevCaps->dwDeviceClassesOffset );
  4652. for (DWORD i = 0; i < dwNumDeviceClasses; i++)
  4653. {
  4654. LOG((TL_INFO, "get_DeviceClasses - got '%ws'", pszDevices));
  4655. DeviceClasses[i] = SysAllocString(pszDevices);
  4656. if (DeviceClasses[i] == NULL)
  4657. {
  4658. LOG((TL_ERROR, "get_DeviceClasses - out of memory"));
  4659. hr = E_OUTOFMEMORY;
  4660. break;
  4661. }
  4662. dwCount++;
  4663. pszDevices += (lstrlenW(pszDevices) + 1 );
  4664. }
  4665. if ( FAILED(hr) )
  4666. {
  4667. // release all the BSTRs and the array.
  4668. for (i = 0; i < dwCount; i ++)
  4669. {
  4670. SysFreeString(DeviceClasses[i]);
  4671. }
  4672. ClientFree(DeviceClasses);
  4673. Unlock();
  4674. return hr;
  4675. }
  4676. hr = CreateBstrCollection(DeviceClasses, dwCount, pVariant);
  4677. // if the collection is not created, release all the BSTRs.
  4678. if (FAILED(hr))
  4679. {
  4680. LOG((TL_ERROR, "get_DeviceClasses - unable to create collection"));
  4681. for (i = 0; i < dwCount; i ++)
  4682. {
  4683. SysFreeString(DeviceClasses[i]);
  4684. }
  4685. }
  4686. // delete the pointer array.
  4687. ClientFree(DeviceClasses);
  4688. }
  4689. else
  4690. {
  4691. LOG((TL_ERROR, "get_DeviceClasses - no device classes"));
  4692. //
  4693. // create an empty collection
  4694. //
  4695. hr = CreateBstrCollection(NULL, 0, pVariant);
  4696. if (FAILED(hr))
  4697. {
  4698. LOG((TL_ERROR, "get_DeviceClasses - unable to create collection"));
  4699. }
  4700. }
  4701. }
  4702. Unlock();
  4703. LOG((TL_TRACE, "get_DeviceClasses - Exit - result - %lx", hr));
  4704. return hr;
  4705. }
  4706. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4707. //
  4708. //
  4709. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4710. STDMETHODIMP
  4711. CAddress::EnumerateDeviceClasses(IEnumBstr ** ppEnumDeviceClass)
  4712. {
  4713. HRESULT hr = S_OK;
  4714. LOG((TL_TRACE, "EnumerateDeviceClasses - Enter"));
  4715. //
  4716. // Check arguments.
  4717. //
  4718. if ( TAPIIsBadWritePtr( ppEnumDeviceClass, sizeof( IEnumBstr * ) ) )
  4719. {
  4720. LOG((TL_ERROR, "EnumerateDeviceClasses - bad pointer"));
  4721. return E_POINTER;
  4722. }
  4723. Lock();
  4724. hr = UpdateLineDevCaps();
  4725. if ( SUCCEEDED(hr) )
  4726. {
  4727. if ( (m_dwAPIVersion >= TAPI_VERSION2_0) && (0 != m_pDevCaps->dwDeviceClassesOffset) )
  4728. {
  4729. PWSTR pszDevices;
  4730. DWORD dwNumDeviceClasses;
  4731. //
  4732. // first count the device classes
  4733. //
  4734. dwNumDeviceClasses = 0;
  4735. pszDevices = (PWSTR)( ( (PBYTE)m_pDevCaps ) + m_pDevCaps->dwDeviceClassesOffset );
  4736. while (NULL != *pszDevices)
  4737. {
  4738. dwNumDeviceClasses++;
  4739. pszDevices += (lstrlenW(pszDevices) + 1 );
  4740. }
  4741. //
  4742. // allocate an array of BSTR pointers
  4743. //
  4744. BSTR *DeviceClasses =
  4745. (BSTR *)ClientAlloc(sizeof(BSTR *) * dwNumDeviceClasses);
  4746. if (DeviceClasses == NULL)
  4747. {
  4748. LOG((TL_ERROR, "EnumerateDeviceClasses - out of memory"));
  4749. Unlock();
  4750. return E_OUTOFMEMORY;
  4751. }
  4752. //
  4753. // allocate all the BSTRs, copying the device class names
  4754. //
  4755. DWORD dwCount = 0;
  4756. pszDevices = (PWSTR)( ( (PBYTE)m_pDevCaps ) + m_pDevCaps->dwDeviceClassesOffset );
  4757. for (DWORD i = 0; i < dwNumDeviceClasses; i++)
  4758. {
  4759. LOG((TL_INFO, "EnumerateDeviceClasses - got '%ws'", pszDevices));
  4760. DeviceClasses[i] = SysAllocString(pszDevices);
  4761. if (DeviceClasses[i] == NULL)
  4762. {
  4763. LOG((TL_ERROR, "EnumerateDeviceClasses - out of memory"));
  4764. hr = E_OUTOFMEMORY;
  4765. break;
  4766. }
  4767. dwCount++;
  4768. pszDevices += (lstrlenW(pszDevices) + 1 );
  4769. }
  4770. if ( FAILED(hr) )
  4771. {
  4772. // release all the BSTRs and the array.
  4773. for (i = 0; i < dwCount; i ++)
  4774. {
  4775. SysFreeString(DeviceClasses[i]);
  4776. }
  4777. ClientFree(DeviceClasses);
  4778. Unlock();
  4779. return hr;
  4780. }
  4781. hr = CreateBstrEnumerator(DeviceClasses, DeviceClasses + dwCount, ppEnumDeviceClass);
  4782. if (FAILED(hr))
  4783. {
  4784. LOG((TL_ERROR, "EnumerateDeviceClasses - unable to create enum"));
  4785. }
  4786. // release all the BSTRs as the enumerator made a copy of them
  4787. for (i = 0; i < dwCount; i ++)
  4788. {
  4789. SysFreeString(DeviceClasses[i]);
  4790. }
  4791. // delete the pointer array.
  4792. ClientFree(DeviceClasses);
  4793. }
  4794. else
  4795. {
  4796. LOG((TL_ERROR, "EnumerateDeviceClasses - no device classes"));
  4797. //
  4798. // create an empty enumeration
  4799. //
  4800. hr = CreateBstrEnumerator(NULL, NULL, ppEnumDeviceClass);
  4801. if (FAILED(hr))
  4802. {
  4803. LOG((TL_ERROR, "EnumerateDeviceClasses - unable to create enumeration"));
  4804. }
  4805. }
  4806. }
  4807. Unlock();
  4808. LOG((TL_TRACE, "EnumerateDeviceClasses - Exit - result - %lx", hr));
  4809. return hr;
  4810. }
  4811. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4812. //
  4813. //
  4814. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4815. void
  4816. HandleLineDevStateMessage(
  4817. CTAPI * pTapi,
  4818. PASYNCEVENTMSG pParams
  4819. )
  4820. {
  4821. LOG((TL_TRACE, "HandleLineDevStateMessage - enter. tapi[%p]", pTapi));
  4822. CAddress * pAddress;
  4823. if ( LINEDEVSTATE_REINIT == pParams->Param1 )
  4824. {
  4825. LOG((TL_TRACE, "HandleLineDevStateMessage - LINEDEVSTATE_REINIT"));
  4826. pTapi->AddRef();
  4827. //This is to remove the bug, 4 RE_INIT messages for two objects.
  4828. pTapi->HandleReinit();
  4829. pTapi->Release();
  4830. LOG((TL_TRACE, "HandleLineDevStateMessage - exit"));
  4831. return;
  4832. }
  4833. if ( LINEDEVSTATE_TRANSLATECHANGE == pParams->Param1 )
  4834. {
  4835. CTapiObjectEvent::FireEvent(
  4836. pTapi,
  4837. TE_TRANSLATECHANGE,
  4838. NULL,
  4839. 0,
  4840. NULL
  4841. );
  4842. return;
  4843. }
  4844. if ( !FindAddressObject(
  4845. (HLINE)(pParams->hDevice),
  4846. &pAddress
  4847. ) )
  4848. {
  4849. LOG((TL_WARN, "Can't process LINE_LINEDEVSTATE message"));
  4850. LOG((TL_WARN, " - cannot find hLine %lx", pParams->hDevice));
  4851. return;
  4852. }
  4853. switch ( pParams->Param1 )
  4854. {
  4855. case LINEDEVSTATE_CONNECTED:
  4856. case LINEDEVSTATE_INSERVICE:
  4857. {
  4858. pAddress->InService( pParams->Param1 );
  4859. break;
  4860. }
  4861. case LINEDEVSTATE_OUTOFSERVICE:
  4862. case LINEDEVSTATE_MAINTENANCE:
  4863. case LINEDEVSTATE_REMOVED:
  4864. case LINEDEVSTATE_DISCONNECTED:
  4865. case LINEDEVSTATE_LOCK:
  4866. pAddress->OutOfService( pParams->Param1 );
  4867. break;
  4868. case LINEDEVSTATE_MSGWAITON:
  4869. CAddressEvent::FireEvent(
  4870. pAddress,
  4871. AE_MSGWAITON,
  4872. NULL
  4873. );
  4874. break;
  4875. case LINEDEVSTATE_MSGWAITOFF:
  4876. CAddressEvent::FireEvent(
  4877. pAddress,
  4878. AE_MSGWAITOFF,
  4879. NULL
  4880. );
  4881. break;
  4882. // the line has been opened or
  4883. // closed by another app
  4884. case LINEDEVSTATE_OPEN:
  4885. case LINEDEVSTATE_CLOSE:
  4886. break;
  4887. case LINEDEVSTATE_CAPSCHANGE:
  4888. pAddress->CapsChange( FALSE );
  4889. break;
  4890. case LINEDEVSTATE_CONFIGCHANGE:
  4891. {
  4892. CAddressEvent::FireEvent(
  4893. pAddress,
  4894. AE_CONFIGCHANGE,
  4895. NULL
  4896. );
  4897. break;
  4898. }
  4899. case LINEDEVSTATE_RINGING:
  4900. {
  4901. CAddressEvent::FireEvent(
  4902. pAddress,
  4903. AE_RINGING,
  4904. NULL
  4905. );
  4906. break;
  4907. }
  4908. case LINEDEVSTATE_DEVSPECIFIC:
  4909. case LINEDEVSTATE_OTHER:
  4910. case LINEDEVSTATE_NUMCALLS:
  4911. case LINEDEVSTATE_NUMCOMPLETIONS:
  4912. case LINEDEVSTATE_TERMINALS:
  4913. case LINEDEVSTATE_ROAMMODE:
  4914. case LINEDEVSTATE_BATTERY:
  4915. case LINEDEVSTATE_SIGNAL:
  4916. case LINEDEVSTATE_COMPLCANCEL:
  4917. LOG((TL_INFO, "LINE_LINEDEVSTATE message not handled - %lx", pParams->Param1));
  4918. break;
  4919. default:
  4920. LOG((TL_WARN, "Unknown LINE_LINEDEVSTATE message - %lx", pParams->Param1));
  4921. break;
  4922. }
  4923. //FindAddressObject addrefs the address objct
  4924. pAddress->Release();
  4925. LOG((TL_TRACE, "HandleLineDevStateMessage - exit."));
  4926. return;
  4927. }
  4928. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4929. //
  4930. //
  4931. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4932. void
  4933. HandleAddressStateMessage(
  4934. PASYNCEVENTMSG pParams
  4935. )
  4936. {
  4937. CAddress * pAddress;
  4938. if ( !FindAddressObject(
  4939. (HLINE)(pParams->hDevice),
  4940. &pAddress
  4941. ) )
  4942. {
  4943. LOG((TL_WARN, "Can't process LINE_LINEDEVSTATE message"));
  4944. LOG((TL_WARN, " - cannot find hLine %lx", pParams->hDevice));
  4945. return;
  4946. }
  4947. switch ( pParams->Param1 )
  4948. {
  4949. case LINEADDRESSSTATE_FORWARD:
  4950. {
  4951. CAddressEvent::FireEvent(
  4952. pAddress,
  4953. AE_FORWARD,
  4954. NULL
  4955. );
  4956. break;
  4957. }
  4958. case LINEADDRESSSTATE_CAPSCHANGE:
  4959. {
  4960. pAddress->CapsChange( TRUE );
  4961. break;
  4962. }
  4963. case LINEADDRESSSTATE_OTHER:
  4964. case LINEADDRESSSTATE_DEVSPECIFIC:
  4965. case LINEADDRESSSTATE_INUSEZERO:
  4966. case LINEADDRESSSTATE_INUSEONE:
  4967. case LINEADDRESSSTATE_INUSEMANY:
  4968. case LINEADDRESSSTATE_NUMCALLS:
  4969. case LINEADDRESSSTATE_TERMINALS:
  4970. LOG((TL_WARN, "HandleAddressStateMessage - not handled %lx", pParams->Param1));
  4971. break;
  4972. default:
  4973. LOG((TL_WARN, "HandleAddressStateMessage - Unknown %lx", pParams->Param1));
  4974. break;
  4975. }
  4976. //FindAddressObject addrefs the address objct
  4977. pAddress->Release();
  4978. return;
  4979. }
  4980. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4981. //
  4982. //
  4983. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  4984. void
  4985. CAddress::InService(
  4986. DWORD dwType
  4987. )
  4988. {
  4989. BOOL bEvent = FALSE;
  4990. switch( dwType )
  4991. {
  4992. case LINEDEVSTATE_CONNECTED:
  4993. case LINEDEVSTATE_INSERVICE:
  4994. default:
  4995. break;
  4996. }
  4997. Lock();
  4998. if ( AS_INSERVICE != m_AddressState )
  4999. {
  5000. m_AddressState = AS_INSERVICE;
  5001. bEvent = TRUE;
  5002. }
  5003. Unlock();
  5004. if (bEvent)
  5005. {
  5006. CAddressEvent::FireEvent(
  5007. this,
  5008. AE_STATE,
  5009. NULL
  5010. );
  5011. }
  5012. }
  5013. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5014. //
  5015. //
  5016. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5017. void
  5018. CAddress::OutOfService(
  5019. DWORD dwType
  5020. )
  5021. {
  5022. BOOL bEvent = FALSE;
  5023. switch ( dwType )
  5024. {
  5025. case LINEDEVSTATE_OUTOFSERVICE:
  5026. case LINEDEVSTATE_MAINTENANCE:
  5027. case LINEDEVSTATE_REMOVED:
  5028. case LINEDEVSTATE_DISCONNECTED:
  5029. case LINEDEVSTATE_LOCK:
  5030. default:
  5031. break;
  5032. }
  5033. Lock();
  5034. if ( AS_OUTOFSERVICE != m_AddressState )
  5035. {
  5036. m_AddressState = AS_OUTOFSERVICE;
  5037. bEvent = TRUE;
  5038. }
  5039. Unlock();
  5040. if ( bEvent )
  5041. {
  5042. CAddressEvent::FireEvent(
  5043. this,
  5044. AE_STATE,
  5045. NULL
  5046. );
  5047. }
  5048. }
  5049. void
  5050. CAddress::CapsChange( BOOL bAddress )
  5051. {
  5052. Lock();
  5053. if (bAddress)
  5054. {
  5055. m_dwAddressFlags |= ADDRESSFLAG_ADDRESSCAPSCHANGE;
  5056. }
  5057. else
  5058. {
  5059. m_dwAddressFlags |= ADDRESSFLAG_DEVCAPSCHANGE;
  5060. }
  5061. Unlock();
  5062. CAddressEvent::FireEvent(
  5063. this,
  5064. AE_CAPSCHANGE,
  5065. NULL
  5066. );
  5067. }
  5068. //
  5069. // CAddressEvent
  5070. //
  5071. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5072. //
  5073. //
  5074. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5075. HRESULT
  5076. CAddressEvent::FireEvent(
  5077. CAddress * pCAddress,
  5078. ADDRESS_EVENT Event,
  5079. ITTerminal * pTerminal
  5080. )
  5081. {
  5082. HRESULT hr = S_OK;
  5083. CComObject<CAddressEvent> * p;
  5084. IDispatch * pDisp;
  5085. //
  5086. // Check the event filter mask
  5087. // just if is the AE_NEWTERMINAL or AE_REMOVETERMINAL
  5088. // These two events are MSP events and are not filtered by
  5089. // TapiSrv
  5090. //
  5091. DWORD dwEventFilterMask = 0;
  5092. dwEventFilterMask = pCAddress->GetSubEventsMask( TE_ADDRESS );
  5093. if( !( dwEventFilterMask & GET_SUBEVENT_FLAG(Event)))
  5094. {
  5095. STATICLOG((TL_ERROR, "This event is filtered - %lx", Event));
  5096. return S_OK;
  5097. }
  5098. //
  5099. // create event
  5100. //
  5101. hr = CComObject<CAddressEvent>::CreateInstance( &p );
  5102. if ( !SUCCEEDED(hr) )
  5103. {
  5104. STATICLOG((TL_ERROR, "Could not create AddressEvent object - %lx", hr));
  5105. return hr;
  5106. }
  5107. //
  5108. // initialize
  5109. //
  5110. p->m_Event = Event;
  5111. p->m_pAddress = dynamic_cast<ITAddress *>(pCAddress);
  5112. p->m_pAddress->AddRef();
  5113. p->m_pTerminal = pTerminal;
  5114. if ( NULL != pTerminal )
  5115. {
  5116. pTerminal->AddRef();
  5117. }
  5118. #if DBG
  5119. p->m_pDebug = (PWSTR) ClientAlloc( 1 );
  5120. #endif
  5121. //
  5122. // get idisp interface
  5123. //
  5124. hr = p->QueryInterface(
  5125. IID_IDispatch,
  5126. (void **)&pDisp
  5127. );
  5128. if ( !SUCCEEDED(hr) )
  5129. {
  5130. STATICLOG((TL_ERROR, "Could not get disp interface of AddressEvent object %lx", hr));
  5131. delete p;
  5132. return hr;
  5133. }
  5134. //
  5135. // get callback
  5136. //
  5137. //
  5138. // fire event
  5139. //
  5140. (pCAddress->GetTapi())->Event(
  5141. TE_ADDRESS,
  5142. pDisp
  5143. );
  5144. //
  5145. // release stuff
  5146. //
  5147. pDisp->Release();
  5148. return S_OK;
  5149. }
  5150. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5151. //
  5152. // finalrelease
  5153. //
  5154. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5155. void
  5156. CAddressEvent::FinalRelease()
  5157. {
  5158. LOG((TL_INFO, "CAddressEvent - FinalRelease"));
  5159. m_pAddress->Release();
  5160. if ( NULL != m_pTerminal )
  5161. {
  5162. m_pTerminal->Release();
  5163. }
  5164. #if DBG
  5165. ClientFree( m_pDebug );
  5166. #endif
  5167. }
  5168. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5169. //
  5170. // get_Address
  5171. //
  5172. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5173. STDMETHODIMP
  5174. CAddressEvent::get_Address(
  5175. ITAddress ** ppAddress
  5176. )
  5177. {
  5178. if (TAPIIsBadWritePtr(ppAddress , sizeof(ITAddress *) ) )
  5179. {
  5180. LOG((TL_ERROR, "get_Address - bad pointer"));
  5181. return E_POINTER;
  5182. }
  5183. *ppAddress = m_pAddress;
  5184. (*ppAddress)->AddRef();
  5185. return S_OK;
  5186. }
  5187. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5188. //
  5189. // get_Terminal
  5190. //
  5191. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5192. STDMETHODIMP
  5193. CAddressEvent::get_Terminal(
  5194. ITTerminal ** ppTerminal
  5195. )
  5196. {
  5197. if ( TAPIIsBadWritePtr( ppTerminal , sizeof(ITTerminal *) ) )
  5198. {
  5199. LOG((TL_ERROR, "get_Terminal - bad pointer"));
  5200. return E_POINTER;
  5201. }
  5202. if ((m_Event != AE_NEWTERMINAL) && (m_Event != AE_REMOVETERMINAL))
  5203. {
  5204. LOG((TL_ERROR, "get_Terminal - wrong event"));
  5205. return TAPI_E_WRONGEVENT;
  5206. }
  5207. *ppTerminal = m_pTerminal;
  5208. if ( NULL != m_pTerminal )
  5209. {
  5210. m_pTerminal->AddRef();
  5211. }
  5212. return S_OK;
  5213. }
  5214. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5215. //
  5216. // get_Event
  5217. //
  5218. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5219. STDMETHODIMP
  5220. CAddressEvent::get_Event(
  5221. ADDRESS_EVENT * pEvent
  5222. )
  5223. {
  5224. if (TAPIIsBadWritePtr(pEvent , sizeof(ADDRESS_EVENT) ) )
  5225. {
  5226. LOG((TL_ERROR, "get_Event - bad pointer"));
  5227. return E_POINTER;
  5228. }
  5229. *pEvent = m_Event;
  5230. return S_OK;
  5231. }
  5232. ///////////////////////////////////////////////////////////////////////////////
  5233. //
  5234. // CAddressDevSpecificEvent
  5235. //
  5236. // static
  5237. HRESULT CAddressDevSpecificEvent::FireEvent( CAddress * pCAddress,
  5238. CCall * pCall,
  5239. long l1,
  5240. long l2,
  5241. long l3
  5242. )
  5243. {
  5244. STATICLOG((TL_INFO, "CAddressDevSpecificEvent::FireEvent - enter"));
  5245. //
  5246. // try to create the event
  5247. //
  5248. CComObject<CAddressDevSpecificEvent> *pEventObject = NULL;
  5249. HRESULT hr = CComObject<CAddressDevSpecificEvent>::CreateInstance(&pEventObject);
  5250. if ( FAILED(hr) )
  5251. {
  5252. STATICLOG((TL_ERROR,
  5253. "CAddressDevSpecificEvent::FireEvent - failed to create CAddressDevSpecificEvent. hr = %lx",
  5254. hr));
  5255. return hr;
  5256. }
  5257. //
  5258. // initialize the event with the data we received
  5259. //
  5260. //
  5261. // get ITAddress from CAddress we received
  5262. //
  5263. hr = pCAddress->_InternalQueryInterface(IID_ITAddress, (void**)(&(pEventObject->m_pAddress)) );
  5264. if (FAILED(hr))
  5265. {
  5266. STATICLOG((TL_ERROR,
  5267. "CAddressDevSpecificEvent::FireEvent - failed to create get ITAddress interface from address. hr = %lx",
  5268. hr));
  5269. delete pEventObject;
  5270. return hr;
  5271. }
  5272. //
  5273. // get ITCallInfo interface from CCall we received
  5274. //
  5275. if (NULL != pCall)
  5276. {
  5277. hr = pCall->_InternalQueryInterface(IID_ITCallInfo, (void**)(&(pEventObject->m_pCall)) );
  5278. if (FAILED(hr))
  5279. {
  5280. STATICLOG((TL_ERROR,
  5281. "CAddressDevSpecificEvent::FireEvent - failed to create get ITAddress interface from address. hr = %lx",
  5282. hr));
  5283. //
  5284. // no need to release event's data members, event's destructor will do
  5285. // this for us
  5286. //
  5287. delete pEventObject;
  5288. return hr;
  5289. }
  5290. }
  5291. //
  5292. // keep the actual data
  5293. //
  5294. pEventObject->m_l1 = l1;
  5295. pEventObject->m_l2 = l2;
  5296. pEventObject->m_l3 = l3;
  5297. #if DBG
  5298. pEventObject->m_pDebug = (PWSTR) ClientAlloc( 1 );
  5299. #endif
  5300. //
  5301. // get event's idispatch interface
  5302. //
  5303. IDispatch *pDispatch = NULL;
  5304. hr = pEventObject->QueryInterface( IID_IDispatch,
  5305. (void **)&pDispatch );
  5306. if ( FAILED(hr) )
  5307. {
  5308. STATICLOG((TL_ERROR,
  5309. "CAddressDevSpecificEvent::FireEvent - Could not get disp interface of AddressEvent object %lx",
  5310. hr));
  5311. //
  5312. // no need to release event's data members, event's destructor will do
  5313. // this for us
  5314. //
  5315. //
  5316. // delete the event object
  5317. //
  5318. delete pEventObject;
  5319. return hr;
  5320. }
  5321. //
  5322. // from this point on, we will be using events pDispatch
  5323. //
  5324. pEventObject = NULL;
  5325. //
  5326. // get callback
  5327. //
  5328. //
  5329. // fire event to tapi
  5330. //
  5331. hr = (pCAddress->GetTapi())->Event(TE_ADDRESSDEVSPECIFIC, pDispatch);
  5332. //
  5333. // succeeded or not, we no longer need a reference to the event object
  5334. //
  5335. pDispatch->Release();
  5336. pDispatch = NULL;
  5337. STATICLOG((TL_INFO, "CAddressDevSpecificEvent::FireEvent - exit, hr = %lx", hr));
  5338. return hr;
  5339. }
  5340. ///////////////////////////////////////////////////////////////////////////////
  5341. CAddressDevSpecificEvent::CAddressDevSpecificEvent()
  5342. :m_pAddress(NULL),
  5343. m_pCall(NULL)
  5344. {
  5345. LOG((TL_INFO, "CAddressDevSpecificEvent - enter"));
  5346. #if DBG
  5347. m_pDebug = NULL;
  5348. #endif
  5349. LOG((TL_INFO, "CAddressDevSpecificEvent - exit"));
  5350. }
  5351. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5352. //
  5353. // ~CAddressDevSpecificEvent
  5354. //
  5355. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5356. CAddressDevSpecificEvent::~CAddressDevSpecificEvent()
  5357. {
  5358. LOG((TL_INFO, "~CAddressDevSpecificEvent - enter"));
  5359. if (NULL != m_pAddress)
  5360. {
  5361. m_pAddress->Release();
  5362. m_pAddress = NULL;
  5363. }
  5364. if (NULL != m_pCall)
  5365. {
  5366. m_pCall->Release();
  5367. m_pCall = NULL;
  5368. }
  5369. #if DBG
  5370. ClientFree( m_pDebug );
  5371. #endif
  5372. LOG((TL_INFO, "~CAddressDevSpecificEvent - exit"));
  5373. }
  5374. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5375. //
  5376. // get_Address
  5377. //
  5378. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5379. STDMETHODIMP
  5380. CAddressDevSpecificEvent::get_Address(
  5381. ITAddress ** ppAddress
  5382. )
  5383. {
  5384. LOG((TL_TRACE, "get_Address - enter"));
  5385. //
  5386. // good out pointer?
  5387. //
  5388. if (TAPIIsBadWritePtr(ppAddress , sizeof(ITAddress *) ) )
  5389. {
  5390. LOG((TL_ERROR, "get_Address - bad pointer at [%p]", ppAddress));
  5391. return E_POINTER;
  5392. }
  5393. //
  5394. // return addreff'd address
  5395. //
  5396. _ASSERTE(NULL != m_pAddress);
  5397. *ppAddress = m_pAddress;
  5398. (*ppAddress)->AddRef();
  5399. LOG((TL_TRACE, "get_Address - enter. address[%p]", (*ppAddress) ));
  5400. return S_OK;
  5401. }
  5402. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5403. //
  5404. // get_Address
  5405. //
  5406. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5407. STDMETHODIMP
  5408. CAddressDevSpecificEvent::get_Call(
  5409. ITCallInfo ** ppCall
  5410. )
  5411. {
  5412. LOG((TL_TRACE, "get_Call - enter"));
  5413. //
  5414. // good out pointer?
  5415. //
  5416. if (TAPIIsBadWritePtr(ppCall, sizeof(ITCallInfo*) ) )
  5417. {
  5418. LOG((TL_ERROR, "get_Call - bad pointer at [%p]", ppCall));
  5419. return E_POINTER;
  5420. }
  5421. //
  5422. // return addreff'd call
  5423. //
  5424. HRESULT hr = S_OK;
  5425. if ( NULL != m_pCall )
  5426. {
  5427. //
  5428. // this event is call specific
  5429. //
  5430. *ppCall = m_pCall;
  5431. (*ppCall)->AddRef();
  5432. }
  5433. else
  5434. {
  5435. //
  5436. // this event was not call specific
  5437. //
  5438. LOG((TL_WARN, "get_Call - no call"));
  5439. hr = TAPI_E_CALLUNAVAIL;
  5440. }
  5441. LOG(( TL_TRACE, "get_Call - enter. call [%p]. hr = %lx", (*ppCall), hr ));
  5442. return hr;
  5443. }
  5444. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5445. //
  5446. // get_lParam1
  5447. //
  5448. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5449. STDMETHODIMP CAddressDevSpecificEvent::get_lParam1( long *pl1 )
  5450. {
  5451. LOG((TL_TRACE, "get_lParam1 - enter"));
  5452. //
  5453. // good out pointer?
  5454. //
  5455. if (TAPIIsBadWritePtr(pl1, sizeof(long) ) )
  5456. {
  5457. LOG((TL_ERROR, "get_lParam1 - bad pointer at %p", pl1));
  5458. return E_POINTER;
  5459. }
  5460. //
  5461. // log and return the value
  5462. //
  5463. *pl1 = m_l1;
  5464. LOG((TL_TRACE, "get_lParam1 - exit. p1[%ld]", *pl1));
  5465. return S_OK;
  5466. }
  5467. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5468. //
  5469. // get_lParam2
  5470. //
  5471. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5472. STDMETHODIMP CAddressDevSpecificEvent::get_lParam2( long *pl2 )
  5473. {
  5474. LOG((TL_TRACE, "get_lParam2 - enter"));
  5475. //
  5476. // good out pointer?
  5477. //
  5478. if (TAPIIsBadWritePtr(pl2, sizeof(long) ) )
  5479. {
  5480. LOG((TL_ERROR, "get_lParam2 - bad pointer at %p", pl2));
  5481. return E_POINTER;
  5482. }
  5483. //
  5484. // log and return the value
  5485. //
  5486. *pl2 = m_l2;
  5487. LOG((TL_TRACE, "get_lParam2 - exit. p2[%ld]", *pl2));
  5488. return S_OK;
  5489. }
  5490. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5491. //
  5492. // get_lParam3
  5493. //
  5494. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5495. STDMETHODIMP CAddressDevSpecificEvent::get_lParam3( long *pl3 )
  5496. {
  5497. LOG((TL_TRACE, "get_lParam3 - enter"));
  5498. //
  5499. // good out pointer?
  5500. //
  5501. if ( TAPIIsBadWritePtr(pl3, sizeof(long)) )
  5502. {
  5503. LOG((TL_ERROR, "get_lParam3 - bad pointer at %p", pl3));
  5504. return E_POINTER;
  5505. }
  5506. //
  5507. // log and return the value
  5508. //
  5509. *pl3 = m_l3;
  5510. LOG((TL_TRACE, "get_lParam3 - exit. p3[%ld]", *pl3));
  5511. return S_OK;
  5512. }
  5513. ///////////////////////////////////////////////////////////////////////////////
  5514. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5515. //
  5516. // Initialize
  5517. //
  5518. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5519. HRESULT
  5520. CForwardInfo::Initialize()
  5521. {
  5522. HRESULT hr = S_OK;
  5523. ZeroMemory(
  5524. m_ForwardStructs,
  5525. sizeof( MYFORWARDSTRUCT ) * NUMFORWARDTYPES
  5526. );
  5527. m_lNumRings = 0;
  5528. return hr;
  5529. }
  5530. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5531. //
  5532. // GetForwardOffset
  5533. //
  5534. // maps a forward type to an offset for the array
  5535. // in the
  5536. //
  5537. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5538. DWORD
  5539. GetForwardOffset(
  5540. DWORD dwForwardType
  5541. )
  5542. {
  5543. switch (dwForwardType)
  5544. {
  5545. case LINEFORWARDMODE_UNCOND:
  5546. return 0;
  5547. case LINEFORWARDMODE_UNCONDINTERNAL:
  5548. return 1;
  5549. case LINEFORWARDMODE_UNCONDEXTERNAL:
  5550. return 2;
  5551. case LINEFORWARDMODE_UNCONDSPECIFIC:
  5552. return 3;
  5553. case LINEFORWARDMODE_BUSY:
  5554. return 4;
  5555. case LINEFORWARDMODE_BUSYINTERNAL:
  5556. return 5;
  5557. case LINEFORWARDMODE_BUSYEXTERNAL:
  5558. return 6;
  5559. case LINEFORWARDMODE_BUSYSPECIFIC:
  5560. return 7;
  5561. case LINEFORWARDMODE_NOANSW:
  5562. return 8;
  5563. case LINEFORWARDMODE_NOANSWINTERNAL:
  5564. return 9;
  5565. case LINEFORWARDMODE_NOANSWEXTERNAL:
  5566. return 10;
  5567. case LINEFORWARDMODE_NOANSWSPECIFIC:
  5568. return 11;
  5569. case LINEFORWARDMODE_BUSYNA:
  5570. return 12;
  5571. case LINEFORWARDMODE_BUSYNAINTERNAL:
  5572. return 13;
  5573. case LINEFORWARDMODE_BUSYNAEXTERNAL:
  5574. return 14;
  5575. case LINEFORWARDMODE_BUSYNASPECIFIC:
  5576. return 15;
  5577. case LINEFORWARDMODE_UNKNOWN:
  5578. return 16;
  5579. case LINEFORWARDMODE_UNAVAIL:
  5580. return 17;
  5581. default:
  5582. return 0;
  5583. }
  5584. return 0;
  5585. }
  5586. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5587. //
  5588. // put_NumRingsNoAnswer
  5589. //
  5590. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5591. STDMETHODIMP
  5592. CForwardInfo::put_NumRingsNoAnswer(
  5593. long lNumRings
  5594. )
  5595. {
  5596. HRESULT hr = S_OK;
  5597. Lock();
  5598. m_lNumRings = lNumRings;
  5599. Unlock();
  5600. return hr;
  5601. }
  5602. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5603. //
  5604. // get_NumRingsNoAnswer
  5605. //
  5606. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5607. STDMETHODIMP
  5608. CForwardInfo::get_NumRingsNoAnswer(
  5609. long * plNumRings
  5610. )
  5611. {
  5612. if (TAPIIsBadWritePtr(plNumRings , sizeof(long) ) )
  5613. {
  5614. LOG((TL_ERROR, "get_NumRingsNoAnswer - bad pointer"));
  5615. return E_POINTER;
  5616. }
  5617. Lock();
  5618. *plNumRings = m_lNumRings;
  5619. Unlock();
  5620. return S_OK;
  5621. }
  5622. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5623. //
  5624. // SetForwardType
  5625. //
  5626. // save the forward type. overwrite and free is there is already
  5627. // a matching type
  5628. //
  5629. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5630. STDMETHODIMP
  5631. CForwardInfo::SetForwardType(
  5632. long ForwardType,
  5633. BSTR pDestAddress,
  5634. BSTR pCallerAddress
  5635. )
  5636. {
  5637. HRESULT hr;
  5638. LOG((TL_TRACE, "SetForwardType - enter"));
  5639. hr = SetForwardType2(
  5640. ForwardType,
  5641. pDestAddress,
  5642. 0,
  5643. pCallerAddress,
  5644. 0
  5645. );
  5646. LOG((TL_TRACE, "SetForwardType - exit - %lx", hr));
  5647. return hr;
  5648. }
  5649. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5650. //
  5651. // SetForwardType2
  5652. //
  5653. // save the forward type. overwrite and free is there is already
  5654. // a matching type
  5655. //
  5656. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5657. STDMETHODIMP
  5658. CForwardInfo::SetForwardType2(
  5659. long ForwardType,
  5660. BSTR pDestAddress,
  5661. long DestAddressType,
  5662. BSTR pCallerAddress,
  5663. long CallerAddressType
  5664. )
  5665. {
  5666. HRESULT hr = S_OK;
  5667. DWORD dwCount;
  5668. LOG((TL_TRACE, "SetForwardType2 - enter"));
  5669. //
  5670. // check forwardtype
  5671. //
  5672. if ( !IsOnlyOneBitSetInDWORD( ForwardType ) )
  5673. {
  5674. LOG((TL_ERROR, "ForwardType has more than one bit set"));
  5675. return E_INVALIDARG;
  5676. }
  5677. //
  5678. // check destaddress
  5679. //
  5680. if ( pDestAddress == NULL )
  5681. {
  5682. LOG((TL_ERROR, "Forward destaddress cannot be NULL"));
  5683. return E_INVALIDARG;
  5684. }
  5685. if ( IsBadStringPtrW( pDestAddress, -1 ) )
  5686. {
  5687. LOG((TL_ERROR, "Forward destaddress invalid"));
  5688. return E_POINTER;
  5689. }
  5690. //
  5691. // check calleraddress
  5692. //
  5693. if ( FORWARDMODENEEDSCALLER( ForwardType ) )
  5694. {
  5695. if ( NULL == pCallerAddress )
  5696. {
  5697. LOG((TL_ERROR, "Forward type needs calleraddress"));
  5698. return E_INVALIDARG;
  5699. }
  5700. if ( IsBadStringPtrW( pCallerAddress, -1 ) )
  5701. {
  5702. LOG((TL_ERROR, "Forward calleraddress invalid"));
  5703. return E_POINTER;
  5704. }
  5705. }
  5706. Lock();
  5707. //
  5708. // find correct structure in array
  5709. //
  5710. MYFORWARDSTRUCT * pStruct = NULL;
  5711. pStruct = &(m_ForwardStructs[GetForwardOffset(ForwardType)]);
  5712. //
  5713. // free alloced stuff
  5714. //
  5715. if ( NULL != pStruct->bstrDestination )
  5716. {
  5717. SysFreeString( pStruct->bstrDestination );
  5718. pStruct->bstrDestination = NULL;
  5719. }
  5720. if ( NULL != pStruct->bstrCaller )
  5721. {
  5722. SysFreeString( pStruct->bstrCaller );
  5723. pStruct->bstrCaller = NULL;
  5724. }
  5725. //
  5726. // save stuff
  5727. //
  5728. pStruct->bstrDestination = SysAllocString( pDestAddress );
  5729. if ( NULL == pStruct->bstrDestination )
  5730. {
  5731. Unlock();
  5732. LOG((TL_ERROR, "Could not alloc dest in put_Forward"));
  5733. return E_OUTOFMEMORY;
  5734. }
  5735. if ( NULL != pCallerAddress )
  5736. {
  5737. pStruct->bstrCaller = SysAllocString( pCallerAddress );
  5738. if ( NULL == pStruct->bstrCaller )
  5739. {
  5740. LOG((TL_ERROR, "Could not calloc caller in put_Forward"));
  5741. SysFreeString( pStruct->bstrDestination );
  5742. Unlock();
  5743. return E_OUTOFMEMORY;
  5744. }
  5745. }
  5746. pStruct->dwDestAddressType = DestAddressType;
  5747. pStruct->dwCallerAddressType = CallerAddressType;
  5748. pStruct->dwForwardType = ForwardType;
  5749. Unlock();
  5750. LOG((TL_TRACE, "SetForwardType2 - exit - success"));
  5751. return S_OK;
  5752. }
  5753. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5754. //
  5755. // get_ForwardTypeDestination
  5756. //
  5757. // will return null if nothing saved
  5758. //
  5759. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5760. STDMETHODIMP
  5761. CForwardInfo::get_ForwardTypeDestination(
  5762. long ForwardType,
  5763. BSTR * ppDestAddress
  5764. )
  5765. {
  5766. HRESULT hr = S_OK;
  5767. DWORD dwCount;
  5768. LOG((TL_TRACE, "get_ForwardTypeDest - enter"));
  5769. //
  5770. // check forwardtype
  5771. //
  5772. if ( !IsOnlyOneBitSetInDWORD( ForwardType ) )
  5773. {
  5774. LOG((TL_ERROR, "ForwardType has more than one bit set"));
  5775. return E_INVALIDARG;
  5776. }
  5777. if ( TAPIIsBadWritePtr( ppDestAddress, sizeof( BSTR ) ) )
  5778. {
  5779. LOG((TL_ERROR, "Bad pointer in get_ForwardTypeDest"));
  5780. return E_POINTER;
  5781. }
  5782. *ppDestAddress = NULL;
  5783. Lock();
  5784. dwCount = GetForwardOffset( ForwardType );
  5785. if ( NULL != m_ForwardStructs[dwCount].bstrDestination )
  5786. {
  5787. *ppDestAddress = SysAllocString(
  5788. m_ForwardStructs[dwCount].bstrDestination
  5789. );
  5790. if ( NULL == *ppDestAddress )
  5791. {
  5792. LOG((TL_ERROR, "OutOfMemory in get_ForwardTypeDest"));
  5793. Unlock();
  5794. return E_POINTER;
  5795. }
  5796. }
  5797. Unlock();
  5798. LOG((TL_TRACE, "get_ForwardTypeDest - exit"));
  5799. return S_OK;
  5800. }
  5801. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5802. //
  5803. // get_ForwardTypeDestinationAddressType
  5804. //
  5805. // will return null if nothing saved
  5806. //
  5807. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5808. STDMETHODIMP
  5809. CForwardInfo::get_ForwardTypeDestinationAddressType(
  5810. long ForwardType,
  5811. long * pDestAddressType
  5812. )
  5813. {
  5814. HRESULT hr = S_OK;
  5815. DWORD dwCount;
  5816. LOG((TL_TRACE, "get_ForwardTypeDestinationAddressType - enter"));
  5817. //
  5818. // check forwardtype
  5819. //
  5820. if ( !IsOnlyOneBitSetInDWORD( ForwardType ) )
  5821. {
  5822. LOG((TL_ERROR, "ForwardType has more than one bit set"));
  5823. return E_INVALIDARG;
  5824. }
  5825. if ( TAPIIsBadWritePtr( pDestAddressType, sizeof( long ) ) )
  5826. {
  5827. LOG((TL_ERROR, "Bad pointer in get_ForwardTypeDestinationAddressType"));
  5828. return E_POINTER;
  5829. }
  5830. Lock();
  5831. dwCount = GetForwardOffset( ForwardType );
  5832. *pDestAddressType = m_ForwardStructs[dwCount].dwDestAddressType;
  5833. Unlock();
  5834. LOG((TL_TRACE, "get_ForwardTypeDestinationAddressType - exit"));
  5835. return S_OK;
  5836. }
  5837. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5838. //
  5839. // get_ForwardTypeCaller
  5840. //
  5841. // gets the caller save for the specifies forward type
  5842. //
  5843. // will return NULL in ppCallerAddress if nothing saved
  5844. //
  5845. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5846. STDMETHODIMP
  5847. CForwardInfo::get_ForwardTypeCaller(
  5848. long ForwardType,
  5849. BSTR * ppCallerAddress
  5850. )
  5851. {
  5852. HRESULT hr = S_OK;
  5853. DWORD dwCount;
  5854. LOG((TL_TRACE, "get_ForwardTypeCaller - enter"));
  5855. //
  5856. // check forwardtype
  5857. //
  5858. if ( !IsOnlyOneBitSetInDWORD( ForwardType ) )
  5859. {
  5860. LOG((TL_ERROR, "ForwardType has more than one bit set"));
  5861. return E_INVALIDARG;
  5862. }
  5863. if ( TAPIIsBadWritePtr( ppCallerAddress, sizeof( BSTR ) ) )
  5864. {
  5865. LOG((TL_ERROR, "Bad pointer in get_ForwardTypeCaller"));
  5866. return E_POINTER;
  5867. }
  5868. *ppCallerAddress = NULL;
  5869. Lock();
  5870. dwCount = GetForwardOffset( ForwardType );
  5871. if ( NULL != m_ForwardStructs[dwCount].bstrCaller )
  5872. {
  5873. *ppCallerAddress = SysAllocString(
  5874. m_ForwardStructs[dwCount].bstrCaller
  5875. );
  5876. if ( NULL == *ppCallerAddress )
  5877. {
  5878. LOG((TL_ERROR, "OutOfMemory in get_ForwardTypeCaller"));
  5879. Unlock();
  5880. return E_POINTER;
  5881. }
  5882. }
  5883. Unlock();
  5884. LOG((TL_TRACE, "get_ForwardTypeDest - exit"));
  5885. return S_OK;
  5886. }
  5887. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5888. //
  5889. // get_ForwardTypeCallerAddressType
  5890. //
  5891. // will return null if nothing saved
  5892. //
  5893. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5894. STDMETHODIMP
  5895. CForwardInfo::get_ForwardTypeCallerAddressType(
  5896. long ForwardType,
  5897. long * pCallerAddressType
  5898. )
  5899. {
  5900. HRESULT hr = S_OK;
  5901. DWORD dwCount;
  5902. LOG((TL_TRACE, "get_ForwardTypeCallerAddressType - enter"));
  5903. //
  5904. // check forwardtype
  5905. //
  5906. if ( !IsOnlyOneBitSetInDWORD( ForwardType ) )
  5907. {
  5908. LOG((TL_ERROR, "ForwardType has more than one bit set"));
  5909. return E_INVALIDARG;
  5910. }
  5911. if ( TAPIIsBadWritePtr( pCallerAddressType, sizeof( long ) ) )
  5912. {
  5913. LOG((TL_ERROR, "Bad pointer in get_ForwardTypeCallerAddressType"));
  5914. return E_POINTER;
  5915. }
  5916. Lock();
  5917. dwCount = GetForwardOffset( ForwardType );
  5918. *pCallerAddressType = m_ForwardStructs[dwCount].dwCallerAddressType;
  5919. Unlock();
  5920. LOG((TL_TRACE, "get_ForwardTypeCallerAddressType - exit"));
  5921. return S_OK;
  5922. }
  5923. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5924. //
  5925. // GetForwardType
  5926. //
  5927. // get the destination and caller based on the type
  5928. //
  5929. // simply use the vb functions to do this.
  5930. //
  5931. // will return success even if no info - both addresses will
  5932. // be NULL in that case
  5933. //
  5934. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5935. STDMETHODIMP
  5936. CForwardInfo::GetForwardType(
  5937. long ForwardType,
  5938. BSTR * ppDestinationAddress,
  5939. BSTR * ppCallerAddress
  5940. )
  5941. {
  5942. HRESULT hr = S_OK;
  5943. LOG((TL_TRACE, "GetForwardType - enter"));
  5944. hr = get_ForwardTypeDestination( ForwardType, ppDestinationAddress );
  5945. if ( !SUCCEEDED(hr) )
  5946. {
  5947. return hr;
  5948. }
  5949. hr = get_ForwardTypeCaller( ForwardType, ppCallerAddress );
  5950. if ( !SUCCEEDED(hr) )
  5951. {
  5952. SysFreeString( *ppDestinationAddress );
  5953. return hr;
  5954. }
  5955. LOG((TL_TRACE, "GetForwardType - exit"));
  5956. return hr;
  5957. }
  5958. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5959. //
  5960. // GetForwardType2
  5961. //
  5962. // get the destination and caller based on the type
  5963. //
  5964. // simply use the vb functions to do this.
  5965. //
  5966. // will return success even if no info - both addresses will
  5967. // be NULL in that case
  5968. //
  5969. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  5970. STDMETHODIMP
  5971. CForwardInfo::GetForwardType2(
  5972. long ForwardType,
  5973. BSTR * ppDestinationAddress,
  5974. long * pDestAddressType,
  5975. BSTR * ppCallerAddress,
  5976. long * pCallerAddressType
  5977. )
  5978. {
  5979. HRESULT hr = S_OK;
  5980. LOG((TL_TRACE, "GetForwardType2 - enter"));
  5981. hr = get_ForwardTypeDestination( ForwardType, ppDestinationAddress );
  5982. if ( !SUCCEEDED(hr) )
  5983. {
  5984. return hr;
  5985. }
  5986. hr = get_ForwardTypeDestinationAddressType( ForwardType, pDestAddressType );
  5987. if ( !SUCCEEDED(hr) )
  5988. {
  5989. SysFreeString( *ppDestinationAddress );
  5990. return hr;
  5991. }
  5992. hr = get_ForwardTypeCaller( ForwardType, ppCallerAddress );
  5993. if ( !SUCCEEDED(hr) )
  5994. {
  5995. SysFreeString( *ppDestinationAddress );
  5996. return hr;
  5997. }
  5998. hr = get_ForwardTypeCallerAddressType( ForwardType, pCallerAddressType );
  5999. if ( !SUCCEEDED(hr) )
  6000. {
  6001. SysFreeString( *ppDestinationAddress );
  6002. SysFreeString( *ppCallerAddress );
  6003. return hr;
  6004. }
  6005. LOG((TL_TRACE, "GetForwardType2 - exit"));
  6006. return hr;
  6007. }
  6008. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6009. //
  6010. // Clear
  6011. //
  6012. // clears & frees all info in the forward object
  6013. //
  6014. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6015. STDMETHODIMP
  6016. CForwardInfo::Clear()
  6017. {
  6018. HRESULT hr = S_OK;
  6019. DWORD dwCount;
  6020. LOG((TL_TRACE, "Clear - enter"));
  6021. Lock();
  6022. //
  6023. // go through all the structs and free
  6024. // related memory
  6025. //
  6026. for (dwCount = 0; dwCount < NUMFORWARDTYPES; dwCount++)
  6027. {
  6028. if ( NULL != m_ForwardStructs[dwCount].bstrDestination )
  6029. {
  6030. SysFreeString( m_ForwardStructs[dwCount].bstrDestination );
  6031. }
  6032. if ( NULL != m_ForwardStructs[dwCount].bstrCaller )
  6033. {
  6034. SysFreeString( m_ForwardStructs[dwCount].bstrCaller );
  6035. }
  6036. }
  6037. //
  6038. // zero out stuff
  6039. //
  6040. ZeroMemory(
  6041. m_ForwardStructs,
  6042. sizeof( MYFORWARDSTRUCT ) * NUMFORWARDTYPES
  6043. );
  6044. Unlock();
  6045. LOG((TL_TRACE, "Clear - exit"));
  6046. return S_OK;
  6047. }
  6048. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6049. //
  6050. // CreateForwardList
  6051. //
  6052. // Creates a LINEFORWARDLIST structure based on the info
  6053. // in the object
  6054. //
  6055. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6056. HRESULT
  6057. CForwardInfo::CreateForwardList(
  6058. LINEFORWARDLIST ** ppList
  6059. )
  6060. {
  6061. LINEFORWARDLIST * pList;
  6062. DWORD dwCount;
  6063. DWORD dwSize = 0;
  6064. DWORD dwOffset;
  6065. DWORD dwNumEntries = 0;
  6066. Lock();
  6067. //
  6068. // count the number of entries that are filled
  6069. //
  6070. for (dwCount = 0; dwCount < NUMFORWARDTYPES; dwCount++)
  6071. {
  6072. if ( 0 != m_ForwardStructs[dwCount].dwForwardType )
  6073. {
  6074. dwSize += ( (lstrlenW( m_ForwardStructs[dwCount].bstrDestination ) + 1) * sizeof(WCHAR*));
  6075. dwSize += ( (lstrlenW( m_ForwardStructs[dwCount].bstrCaller ) + 1) * sizeof(WCHAR*));
  6076. dwNumEntries++;
  6077. }
  6078. }
  6079. if ( 0 == dwNumEntries )
  6080. {
  6081. Unlock();
  6082. *ppList = NULL;
  6083. return S_OK;
  6084. }
  6085. dwSize += sizeof (LINEFORWARDLIST) +
  6086. sizeof (LINEFORWARD) * dwNumEntries +
  6087. dwSize;
  6088. //
  6089. // alloc structure
  6090. //
  6091. pList = (LINEFORWARDLIST *)ClientAlloc( dwSize );
  6092. if ( NULL == pList )
  6093. {
  6094. LOG((TL_ERROR, "CreateForwardList - OutOfMemory"));
  6095. Unlock();
  6096. return E_OUTOFMEMORY;
  6097. }
  6098. //
  6099. // init
  6100. //
  6101. pList->dwTotalSize = dwSize;
  6102. pList->dwNumEntries = dwNumEntries;
  6103. //
  6104. // offset should be past the fixed part of the structure
  6105. //
  6106. dwOffset = sizeof( LINEFORWARDLIST ) + sizeof( LINEFORWARD ) * dwNumEntries;
  6107. dwNumEntries = 0;
  6108. //
  6109. // go through entries again
  6110. //
  6111. for (dwCount = 0; dwCount < NUMFORWARDTYPES; dwCount++)
  6112. {
  6113. if ( 0 != m_ForwardStructs[dwCount].dwForwardType )
  6114. {
  6115. DWORD dwSize;
  6116. LINEFORWARD * pEntry = &(pList->ForwardList[dwNumEntries]);
  6117. //
  6118. // save the type
  6119. //
  6120. pEntry->dwForwardMode = m_ForwardStructs[dwCount].dwForwardType;
  6121. //
  6122. // save destination - should always be a destination if
  6123. // there is a type
  6124. //
  6125. pEntry->dwDestAddressType = m_ForwardStructs[dwCount].dwDestAddressType;
  6126. pEntry->dwDestAddressSize = (lstrlenW(m_ForwardStructs[dwCount].bstrDestination) + 1)
  6127. * sizeof( WCHAR );
  6128. pEntry->dwDestAddressOffset = dwOffset;
  6129. lstrcpyW(
  6130. (PWSTR)(((PBYTE)pList)+dwOffset),
  6131. m_ForwardStructs[dwCount].bstrDestination
  6132. );
  6133. //
  6134. // fixup offset
  6135. //
  6136. dwOffset += pEntry->dwDestAddressSize;
  6137. //
  6138. // if there is a caller, do the same
  6139. //
  6140. if ( NULL != m_ForwardStructs[dwCount].bstrCaller )
  6141. {
  6142. pEntry->dwCallerAddressType = m_ForwardStructs[dwCount].dwCallerAddressType;
  6143. pEntry->dwCallerAddressSize = (lstrlenW(m_ForwardStructs[dwCount].bstrCaller) + 1)
  6144. * sizeof( WCHAR );
  6145. pEntry->dwCallerAddressOffset = dwOffset;
  6146. lstrcpyW(
  6147. (PWSTR)(((PBYTE)pList)+dwOffset),
  6148. m_ForwardStructs[dwCount].bstrCaller
  6149. );
  6150. dwOffset += pEntry->dwCallerAddressSize;
  6151. }
  6152. dwNumEntries++;
  6153. }
  6154. }
  6155. //
  6156. // return it
  6157. //
  6158. *ppList = pList;
  6159. Unlock();
  6160. return S_OK;
  6161. }
  6162. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6163. //
  6164. // FinalRelease()
  6165. //
  6166. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6167. void
  6168. CForwardInfo::FinalRelease()
  6169. {
  6170. //
  6171. // simply clear it
  6172. //
  6173. Clear();
  6174. }
  6175. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6176. // Class : CAddress
  6177. // Interface : ITAddressTranslation
  6178. // Method : TranslateAddress
  6179. //
  6180. //
  6181. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6182. STDMETHODIMP
  6183. CAddress::TranslateAddress(
  6184. BSTR pAddressToTranslate,
  6185. long ulCard,
  6186. long ulTranslateOptions,
  6187. ITAddressTranslationInfo ** ppTranslated
  6188. )
  6189. {
  6190. HRESULT hr = S_OK;
  6191. LPLINETRANSLATEOUTPUT pTranslateOutput = NULL;
  6192. PWSTR pszDialableString = NULL;
  6193. PWSTR pszDisplayableString = NULL;
  6194. CComObject< CAddressTranslationInfo > * pTranslationInfo = NULL;
  6195. long lCap = 0;
  6196. BOOL bUsePSTNAddressTranslation = TRUE;
  6197. HLINEAPP hLineApp;
  6198. DWORD dwDeviceID;
  6199. LOG((TL_TRACE, "TranslateAddress - enter" ));
  6200. if ( !TAPIIsBadWritePtr( ppTranslated, sizeof(ITAddressTranslationInfo *) ) )
  6201. {
  6202. // ppTranslated OK
  6203. if ( !IsBadStringPtrW( pAddressToTranslate, -1 ) )
  6204. {
  6205. // pAddressToTranslate OK
  6206. // Check Addresscap bit
  6207. hr = get_AddressCapability( AC_ADDRESSCAPFLAGS, &lCap );
  6208. if ( SUCCEEDED(hr) )
  6209. {
  6210. if ( lCap & LINEADDRCAPFLAGS_NOPSTNADDRESSTRANSLATION )
  6211. {
  6212. bUsePSTNAddressTranslation = FALSE;
  6213. }
  6214. }
  6215. // create Translate Info object
  6216. hr = CComObject< CAddressTranslationInfo >::CreateInstance( &pTranslationInfo );
  6217. if ( SUCCEEDED(hr) )
  6218. {
  6219. // Translate or copy ?
  6220. if (bUsePSTNAddressTranslation)
  6221. {
  6222. LOG((TL_INFO, "TranslateAddress - Do address translation" ));
  6223. Lock();
  6224. hLineApp = m_hLineApp;
  6225. dwDeviceID = m_dwDeviceID;
  6226. Unlock();
  6227. hr= LineTranslateAddress(
  6228. hLineApp,
  6229. dwDeviceID,
  6230. TAPI_CURRENT_VERSION,
  6231. (LPCWSTR)pAddressToTranslate,
  6232. ulCard,
  6233. ulTranslateOptions,
  6234. &pTranslateOutput);
  6235. if(SUCCEEDED(hr) )
  6236. {
  6237. // Pull String info out of LPLINETRANSLATEOUTPUT structure
  6238. pszDialableString = (PWSTR) ((BYTE*)(pTranslateOutput) + pTranslateOutput->dwDialableStringOffset);
  6239. pszDisplayableString = (PWSTR) ((BYTE*)(pTranslateOutput) + pTranslateOutput->dwDisplayableStringOffset);
  6240. hr = pTranslationInfo->Initialize(pszDialableString,
  6241. pszDisplayableString,
  6242. pTranslateOutput->dwCurrentCountry,
  6243. pTranslateOutput->dwDestCountry,
  6244. pTranslateOutput->dwTranslateResults
  6245. );
  6246. }
  6247. else // LinetranslateAddress failed
  6248. {
  6249. LOG((TL_ERROR, "TranslateAddress - LineTranslateAddress failed" ));
  6250. }
  6251. }
  6252. else // copy input string unmodified
  6253. {
  6254. LOG((TL_INFO, "TranslateAddress - No address translation" ));
  6255. hr = pTranslationInfo->Initialize(pAddressToTranslate,
  6256. pAddressToTranslate,
  6257. 0,
  6258. 0,
  6259. LINETRANSLATERESULT_NOTRANSLATION
  6260. );
  6261. } // end if (bUsePSTNAddressTranslation)
  6262. //
  6263. // Did we translate & initialize output object ?
  6264. if ( SUCCEEDED(hr) )
  6265. {
  6266. hr = pTranslationInfo->QueryInterface(IID_ITAddressTranslationInfo,(void**)ppTranslated);
  6267. if ( SUCCEEDED(hr) )
  6268. {
  6269. LOG((TL_TRACE, "TranslateAddress - success"));
  6270. hr = S_OK;
  6271. }
  6272. else
  6273. {
  6274. LOG((TL_ERROR, "TranslateAddress - Bad pointer" ));
  6275. delete pTranslationInfo;
  6276. }
  6277. }
  6278. else // object failed to initialize
  6279. {
  6280. LOG((TL_ERROR, "TranslateAddress - Initialize TranslateInfo object failed" ));
  6281. delete pTranslationInfo;
  6282. }
  6283. }
  6284. else // Create instance failed
  6285. {
  6286. LOG((TL_ERROR, "TranslateAddress - Create TranslateInfo object failed" ));
  6287. }
  6288. }
  6289. else // pAddressToTranslate bad
  6290. {
  6291. LOG((TL_ERROR, "TranslateAddress -pAddressToTranslate invalid"));
  6292. hr = E_POINTER;
  6293. }
  6294. }
  6295. else // ppTranslated bad
  6296. {
  6297. LOG((TL_ERROR, "TranslateAddress - Bad ppTranslated Pointer" ));
  6298. hr = E_POINTER;
  6299. }
  6300. if(pTranslateOutput != NULL)
  6301. {
  6302. ClientFree(pTranslateOutput);
  6303. }
  6304. LOG((TL_TRACE, hr, "TranslateAddress - exit" ));
  6305. return hr;
  6306. }
  6307. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6308. //
  6309. // TranslateDialog
  6310. //
  6311. // simply call LineTranslateDialog
  6312. //
  6313. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6314. STDMETHODIMP
  6315. CAddress::TranslateDialog(
  6316. TAPIHWND hwndOwner,
  6317. BSTR pAddressIn
  6318. )
  6319. {
  6320. HRESULT hr = E_NOTIMPL;
  6321. HLINEAPP hLineApp;
  6322. DWORD dwDeviceID;
  6323. DWORD dwAPIVersion;
  6324. LOG((TL_TRACE, "TranslateDialog - enter:%p", hwndOwner ));
  6325. Lock();
  6326. hLineApp = m_hLineApp;
  6327. dwDeviceID = m_dwDeviceID;
  6328. dwAPIVersion = m_dwAPIVersion;
  6329. Unlock();
  6330. hr = LineTranslateDialog(
  6331. dwDeviceID,
  6332. dwAPIVersion,
  6333. (HWND)hwndOwner,
  6334. pAddressIn
  6335. );
  6336. LOG((TL_TRACE, "TranslateDialog - exit - return %lx", hr));
  6337. return hr;
  6338. }
  6339. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6340. // Class : CAddress
  6341. // Interface : ITAddressTranslation
  6342. // Method : EnumerateLocations
  6343. //
  6344. //
  6345. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6346. STDMETHODIMP CAddress::EnumerateLocations (IEnumLocation ** ppEnumLocation )
  6347. {
  6348. HRESULT hr = S_OK;
  6349. LPLINETRANSLATECAPS pTranslateCaps = NULL;
  6350. DWORD dwNumLocations;
  6351. DWORD dwCount;
  6352. LPLINELOCATIONENTRY pEntry = NULL;
  6353. PWSTR pszLocationName;
  6354. PWSTR pszCityCode;
  6355. PWSTR pszLocalAccessCode;
  6356. PWSTR pszLongDistanceAccessCode;
  6357. PWSTR pszTollPrefixList;
  6358. PWSTR pszCancelCallWaitingCode;
  6359. DWORD dwPermanentLocationID;
  6360. DWORD dwCountryCode;
  6361. DWORD dwPreferredCardID;
  6362. DWORD dwCountryID;
  6363. DWORD dwOptions;
  6364. LOG((TL_TRACE, "EnumerateLocations - enter" ));
  6365. if ( TAPIIsBadWritePtr( ppEnumLocation, sizeof(IEnumLocation *) ) )
  6366. {
  6367. LOG((TL_ERROR, "EnumerateLocations - Bad Pointer" ));
  6368. hr = E_POINTER;
  6369. }
  6370. else // Ok Pointer
  6371. {
  6372. //
  6373. // create the enumerator
  6374. //
  6375. CComObject< CTapiEnum<IEnumLocation, ITLocationInfo, &IID_IEnumLocation> > * pEnum;
  6376. hr = CComObject< CTapiEnum<IEnumLocation, ITLocationInfo, &IID_IEnumLocation> > ::CreateInstance( &pEnum );
  6377. if (SUCCEEDED(hr) )
  6378. {
  6379. //
  6380. // initialize it with our Locations list
  6381. //
  6382. pEnum->Initialize();
  6383. hr = LineGetTranslateCaps(m_hLineApp, TAPI_CURRENT_VERSION, &pTranslateCaps);
  6384. if(SUCCEEDED(hr) )
  6385. {
  6386. dwNumLocations = pTranslateCaps->dwNumLocations ;
  6387. // Find positionn of 1st LINELOCATIONENTRY structure in the LINETRANSLATECAPS structure
  6388. pEntry = (LPLINELOCATIONENTRY) ((BYTE*)(pTranslateCaps) + pTranslateCaps->dwLocationListOffset );
  6389. for (dwCount = 0; dwCount < dwNumLocations; dwCount++)
  6390. {
  6391. // Pull Location Info out of LINELOCATIONENTRY structure
  6392. pszLocationName = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwLocationNameOffset);
  6393. pszCityCode = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwCityCodeOffset);
  6394. pszLocalAccessCode = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwLocalAccessCodeOffset);
  6395. pszLongDistanceAccessCode = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwLongDistanceAccessCodeOffset);
  6396. pszTollPrefixList = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwTollPrefixListOffset);
  6397. pszCancelCallWaitingCode = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwCancelCallWaitingOffset);
  6398. dwPermanentLocationID = pEntry->dwPermanentLocationID;
  6399. dwCountryCode = pEntry->dwCountryCode;
  6400. dwPreferredCardID = pEntry->dwPreferredCardID;
  6401. dwCountryID = pEntry->dwCountryID;
  6402. dwOptions = pEntry->dwOptions;
  6403. // create our new LocationInfo Object
  6404. CComObject<CLocationInfo> * pLocationInfo;
  6405. CComObject<CLocationInfo>::CreateInstance( &pLocationInfo );
  6406. if (SUCCEEDED(hr) )
  6407. {
  6408. // initialize the new LocationInfo Object
  6409. hr = pLocationInfo->Initialize(
  6410. pszLocationName,
  6411. pszCityCode,
  6412. pszLocalAccessCode,
  6413. pszLongDistanceAccessCode,
  6414. pszTollPrefixList,
  6415. pszCancelCallWaitingCode ,
  6416. dwPermanentLocationID,
  6417. dwCountryCode,
  6418. dwPreferredCardID,
  6419. dwCountryID,
  6420. dwOptions
  6421. );
  6422. if (SUCCEEDED(hr) )
  6423. {
  6424. // Add it to the enumerator
  6425. hr = pEnum->Add(pLocationInfo);
  6426. if (SUCCEEDED(hr))
  6427. {
  6428. LOG((TL_INFO, "EnumerateLocations - Added LocationInfo object to enum"));
  6429. }
  6430. else
  6431. {
  6432. LOG((TL_INFO, "EnumerateLocations - Add LocationInfo object failed"));
  6433. delete pLocationInfo;
  6434. }
  6435. }
  6436. else
  6437. {
  6438. LOG((TL_ERROR, "EnumerateLocations - Init LocationInfo object failed"));
  6439. delete pLocationInfo;
  6440. }
  6441. }
  6442. else // CComObject::CreateInstance failed
  6443. {
  6444. LOG((TL_ERROR, "EnumerateLocations - Create LocationInfo object failed"));
  6445. }
  6446. // Try next location in list
  6447. pEntry++;
  6448. } //for(dwCount.....)
  6449. //
  6450. // return the Enumerator
  6451. //
  6452. *ppEnumLocation = pEnum;
  6453. }
  6454. else // LineGetTranslateCaps failed
  6455. {
  6456. LOG((TL_ERROR, "EnumerateLocations - LineGetTranslateCaps failed" ));
  6457. pEnum->Release();
  6458. }
  6459. }
  6460. else // CComObject::CreateInstance failed
  6461. {
  6462. LOG((TL_ERROR, "EnumerateLocations - could not create enum" ));
  6463. }
  6464. // finished with TAPI memory block so release
  6465. if ( pTranslateCaps != NULL )
  6466. ClientFree( pTranslateCaps );
  6467. }
  6468. LOG((TL_TRACE, hr, "EnumerateLocations - exit" ));
  6469. return hr;
  6470. }
  6471. STDMETHODIMP
  6472. CAddress::get_Locations(
  6473. VARIANT * pVariant
  6474. )
  6475. {
  6476. IEnumLocation * pEnumLocation;
  6477. HRESULT hr;
  6478. CComObject< CTapiCollection< ITLocationInfo > > * p;
  6479. LocationArray TempLocationArray;
  6480. ITLocationInfo * pLocation;
  6481. if ( TAPIIsBadWritePtr( pVariant, sizeof( VARIANT ) ) )
  6482. {
  6483. LOG((TL_ERROR, "get_locations - bad pointer"));
  6484. return E_POINTER;
  6485. }
  6486. //
  6487. // create collection object
  6488. //
  6489. CComObject< CTapiCollection< ITLocationInfo > >::CreateInstance( &p );
  6490. if (NULL == p)
  6491. {
  6492. LOG((TL_ERROR, "get_Locations - could not create collection" ));
  6493. return E_OUTOFMEMORY;
  6494. }
  6495. hr = EnumerateLocations ( &pEnumLocation );
  6496. if ( !SUCCEEDED(hr) )
  6497. {
  6498. delete p;
  6499. LOG((TL_ERROR, "get_locations - enumerate locations failed"));
  6500. return hr;
  6501. }
  6502. while (TRUE)
  6503. {
  6504. hr = pEnumLocation->Next(1, &pLocation, NULL);
  6505. if ( S_OK != hr )
  6506. {
  6507. break;
  6508. }
  6509. TempLocationArray.Add( pLocation );
  6510. pLocation->Release();
  6511. }
  6512. pEnumLocation->Release();
  6513. p->Initialize( TempLocationArray );
  6514. TempLocationArray.Shutdown();
  6515. IDispatch * pDisp;
  6516. //
  6517. // get the IDispatch interface
  6518. //
  6519. hr = p->_InternalQueryInterface( IID_IDispatch, (void **) &pDisp );
  6520. if (S_OK != hr)
  6521. {
  6522. LOG((TL_ERROR, "get_Locations - could not get IDispatch interface" ));
  6523. delete p;
  6524. return hr;
  6525. }
  6526. //
  6527. // put it in the variant
  6528. //
  6529. VariantInit(pVariant);
  6530. pVariant->vt = VT_DISPATCH;
  6531. pVariant->pdispVal = pDisp;
  6532. LOG((TL_TRACE, "get_Locations exit - return success"));
  6533. return S_OK;
  6534. }
  6535. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6536. // Class : CAddress
  6537. // Interface : ITAddressTranslation
  6538. // Method : EnumerateCallingCards
  6539. //
  6540. //
  6541. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6542. STDMETHODIMP CAddress::EnumerateCallingCards (IEnumCallingCard ** ppCallingCards )
  6543. {
  6544. HRESULT hr = S_OK;
  6545. LPLINETRANSLATECAPS pTranslateCaps = NULL;
  6546. DWORD dwNumCards;
  6547. DWORD dwCount;
  6548. LPLINECARDENTRY pEntry = NULL;
  6549. PWSTR pszCardName;
  6550. PWSTR pszSameAreaDialingRule;
  6551. PWSTR pszLongDistanceDialingRule;
  6552. PWSTR pszInternationalDialingRule;
  6553. DWORD dwPermanentCardID;
  6554. DWORD dwNumberOfDigits;
  6555. DWORD dwOptions;
  6556. LOG((TL_TRACE, "EnumerateCallingCards - enter" ));
  6557. if ( TAPIIsBadWritePtr( ppCallingCards, sizeof(IEnumCallingCard *) ) )
  6558. {
  6559. LOG((TL_ERROR, "EnumerateCallingCards - Bad Pointer" ));
  6560. hr = E_POINTER;
  6561. }
  6562. else // Ok Pointer
  6563. {
  6564. //
  6565. // create the enumerator
  6566. //
  6567. CComObject< CTapiEnum<IEnumCallingCard, ITCallingCard, &IID_IEnumCallingCard> > * pEnum;
  6568. hr = CComObject< CTapiEnum<IEnumCallingCard, ITCallingCard, &IID_IEnumCallingCard> > ::CreateInstance( &pEnum );
  6569. if ( SUCCEEDED(hr) )
  6570. {
  6571. //
  6572. // initialize it with our Locations list
  6573. //
  6574. pEnum->Initialize();
  6575. hr = LineGetTranslateCaps(m_hLineApp, TAPI_CURRENT_VERSION, &pTranslateCaps);
  6576. if( SUCCEEDED(hr) )
  6577. {
  6578. dwNumCards = pTranslateCaps->dwNumCards ;
  6579. // Find positionn of 1st LINECARDENTRY structure in the LINETRANSLATECAPS structure
  6580. pEntry = (LPLINECARDENTRY) ((BYTE*)(pTranslateCaps) + pTranslateCaps->dwCardListOffset );
  6581. for (dwCount = 0; dwCount < dwNumCards; dwCount++)
  6582. {
  6583. // Pull Location Info out of LINECARDENTRY structure
  6584. pszCardName = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwCardNameOffset);
  6585. pszSameAreaDialingRule = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwSameAreaRuleOffset);
  6586. pszLongDistanceDialingRule = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwLongDistanceRuleOffset);
  6587. pszInternationalDialingRule = (PWSTR) ((BYTE*)(pTranslateCaps) + pEntry->dwInternationalRuleOffset);
  6588. dwPermanentCardID = pEntry->dwPermanentCardID;
  6589. dwNumberOfDigits = pEntry->dwCardNumberDigits;
  6590. dwOptions = pEntry->dwOptions;
  6591. // create our new CallingCard Object
  6592. CComObject<CCallingCard> * pCallingCard;
  6593. CComObject<CCallingCard>::CreateInstance( &pCallingCard );
  6594. if (SUCCEEDED(hr) )
  6595. {
  6596. // initialize the new CallingCard Object
  6597. hr = pCallingCard->Initialize(
  6598. pszCardName,
  6599. pszSameAreaDialingRule,
  6600. pszLongDistanceDialingRule,
  6601. pszInternationalDialingRule,
  6602. dwPermanentCardID,
  6603. dwNumberOfDigits,
  6604. dwOptions
  6605. );
  6606. if (SUCCEEDED(hr) )
  6607. {
  6608. // Add it to the enumerator
  6609. hr = pEnum->Add(pCallingCard);
  6610. if (SUCCEEDED(hr))
  6611. {
  6612. LOG((TL_INFO, "EnumerateCallingCards - Added CallingCard object to enum"));
  6613. }
  6614. else
  6615. {
  6616. LOG((TL_INFO, "EnumertateCallingCards - Add CallingCard object failed"));
  6617. delete pCallingCard;
  6618. }
  6619. }
  6620. else
  6621. {
  6622. LOG((TL_ERROR, "EnumerateCallingCards - Init CallingCard object failed"));
  6623. delete pCallingCard;
  6624. }
  6625. }
  6626. else // CComObject::CreateInstance failed
  6627. {
  6628. LOG((TL_ERROR, "EnumerateCallingCards - Create CallingCard object failed"));
  6629. }
  6630. // Try next card in list
  6631. pEntry++;
  6632. } //for(dwCount.....)
  6633. //
  6634. // return the Enumerator
  6635. //
  6636. *ppCallingCards = pEnum;
  6637. }
  6638. else // LineGetTranslateCaps failed
  6639. {
  6640. LOG((TL_ERROR, "EnumerateCallingCards - LineGetTranslateCaps failed" ));
  6641. pEnum->Release();
  6642. }
  6643. }
  6644. else // CComObject::CreateInstance failed
  6645. {
  6646. LOG((TL_ERROR, "EnumerateCallingCards - could not create enum" ));
  6647. }
  6648. // finished with TAPI memory block so release
  6649. if ( pTranslateCaps != NULL )
  6650. ClientFree( pTranslateCaps );
  6651. }
  6652. LOG((TL_TRACE, hr, "EnumerateCallingCards - exit" ));
  6653. return hr;
  6654. }
  6655. STDMETHODIMP
  6656. CAddress::get_CallingCards(
  6657. VARIANT * pVariant
  6658. )
  6659. {
  6660. IEnumCallingCard * pEnumCallingCards;
  6661. HRESULT hr;
  6662. CComObject< CTapiCollection< ITCallingCard > > * p;
  6663. CallingCardArray TempCallingCardArray;
  6664. ITCallingCard * pCallingCard;
  6665. if ( TAPIIsBadWritePtr( pVariant, sizeof( VARIANT ) ) )
  6666. {
  6667. LOG((TL_ERROR, "get_CallingCard - bad pointer"));
  6668. return E_POINTER;
  6669. }
  6670. //
  6671. // create collection object
  6672. //
  6673. CComObject< CTapiCollection< ITCallingCard > >::CreateInstance( &p );
  6674. if (NULL == p)
  6675. {
  6676. LOG((TL_ERROR, "get_CallingCards - could not create collection" ));
  6677. return E_OUTOFMEMORY;
  6678. }
  6679. hr = EnumerateCallingCards ( &pEnumCallingCards );
  6680. if ( !SUCCEEDED(hr) )
  6681. {
  6682. delete p;
  6683. LOG((TL_ERROR, "get_CallingCards - enumerate callingcards failed"));
  6684. return hr;
  6685. }
  6686. while (TRUE)
  6687. {
  6688. hr = pEnumCallingCards->Next(1, &pCallingCard, NULL);
  6689. if ( S_OK != hr )
  6690. {
  6691. break;
  6692. }
  6693. TempCallingCardArray.Add( pCallingCard );
  6694. pCallingCard->Release();
  6695. }
  6696. pEnumCallingCards->Release();
  6697. p->Initialize( TempCallingCardArray );
  6698. TempCallingCardArray.Shutdown();
  6699. IDispatch * pDisp;
  6700. //
  6701. // get the IDispatch interface
  6702. //
  6703. hr = p->_InternalQueryInterface( IID_IDispatch, (void **) &pDisp );
  6704. if (S_OK != hr)
  6705. {
  6706. LOG((TL_ERROR, "get_CallingCards - could not get IDispatch interface" ));
  6707. delete p;
  6708. return hr;
  6709. }
  6710. //
  6711. // put it in the variant
  6712. //
  6713. VariantInit(pVariant);
  6714. pVariant->vt = VT_DISPATCH;
  6715. pVariant->pdispVal = pDisp;
  6716. LOG((TL_TRACE, "get_CallingCards exit - return success"));
  6717. return S_OK;
  6718. }
  6719. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6720. // Class : CAddress
  6721. // Method : GetPhoneArrayFromTapiAndPrune
  6722. //
  6723. //
  6724. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6725. HRESULT
  6726. CAddress::GetPhoneArrayFromTapiAndPrune(
  6727. PhoneArray *pPhoneArray,
  6728. BOOL bPreferredOnly
  6729. )
  6730. {
  6731. HRESULT hr;
  6732. CTAPI * pCTapi;
  6733. ITPhone * pPhone;
  6734. CPhone * pCPhone;
  6735. LOG((TL_TRACE, "GetPhoneArrayFromTapiAndPrune enter"));
  6736. if ( IsBadReadPtr( pPhoneArray, sizeof( PhoneArray ) ) )
  6737. {
  6738. LOG((TL_ERROR, "GetPhoneArrayFromTapiAndPrune - bad pointer"));
  6739. return E_POINTER;
  6740. }
  6741. pCTapi = GetTapi();
  6742. if( NULL == pCTapi )
  6743. {
  6744. LOG((TL_ERROR, "dynamic cast operation failed"));
  6745. hr = E_POINTER;
  6746. }
  6747. else
  6748. {
  6749. hr = pCTapi->GetPhoneArray( pPhoneArray );
  6750. if ( SUCCEEDED(hr) )
  6751. {
  6752. //
  6753. // Go through the phones
  6754. //
  6755. for(int iCount = 0; iCount < pPhoneArray->GetSize(); iCount++)
  6756. {
  6757. pPhone = (*pPhoneArray)[iCount];
  6758. pCPhone = dynamic_cast<CPhone *>(pPhone);
  6759. if ( NULL == pCPhone )
  6760. {
  6761. //
  6762. // We have a bad pointer in our phone array.
  6763. // Lets skip it and move on.
  6764. //
  6765. _ASSERTE(FALSE);
  6766. continue;
  6767. }
  6768. //
  6769. // Is the phone on this address?
  6770. //
  6771. if ( bPreferredOnly ? pCPhone->IsPhoneOnPreferredAddress(this) : pCPhone->IsPhoneOnAddress(this) )
  6772. {
  6773. LOG((TL_INFO, "GetPhoneArrayFromTapiAndPrune - found matching phone - %p", pPhone));
  6774. }
  6775. else
  6776. {
  6777. // No, remove it from the array
  6778. pPhoneArray->RemoveAt(iCount);
  6779. iCount--;
  6780. }
  6781. }
  6782. }
  6783. }
  6784. LOG((TL_TRACE, "GetPhoneArrayFromTapiAndPrune - exit - return %lx", hr ));
  6785. return hr;
  6786. }
  6787. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6788. // Class : CAddress
  6789. // Interface : ITAddress2
  6790. // Method : get_Phones
  6791. //
  6792. //
  6793. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6794. HRESULT
  6795. CAddress::get_Phones(
  6796. VARIANT * pPhones
  6797. )
  6798. {
  6799. HRESULT hr;
  6800. IDispatch * pDisp;
  6801. PhoneArray PhoneArray;
  6802. LOG((TL_TRACE, "get_Phones enter"));
  6803. if ( TAPIIsBadWritePtr( pPhones, sizeof( VARIANT ) ) )
  6804. {
  6805. LOG((TL_ERROR, "get_Phones - bad pointer"));
  6806. return E_POINTER;
  6807. }
  6808. hr = GetPhoneArrayFromTapiAndPrune( &PhoneArray, FALSE );
  6809. if ( SUCCEEDED(hr) )
  6810. {
  6811. CComObject< CTapiCollection< ITPhone > > * p;
  6812. CComObject< CTapiCollection< ITPhone > >::CreateInstance( &p );
  6813. if (NULL == p)
  6814. {
  6815. LOG((TL_ERROR, "get_Phones - could not create collection" ));
  6816. PhoneArray.Shutdown();
  6817. return E_OUTOFMEMORY;
  6818. }
  6819. // get the IDispatch interface
  6820. hr = p->_InternalQueryInterface( IID_IDispatch, (void **) &pDisp );
  6821. if (S_OK != hr)
  6822. {
  6823. LOG((TL_ERROR, "get_Phones - could not get IDispatch interface" ));
  6824. delete p;
  6825. return hr;
  6826. }
  6827. Lock();
  6828. // initialize
  6829. hr = p->Initialize( PhoneArray );
  6830. Unlock();
  6831. PhoneArray.Shutdown();
  6832. if (S_OK != hr)
  6833. {
  6834. LOG((TL_ERROR, "get_Phones - could not initialize collection" ));
  6835. pDisp->Release();
  6836. return hr;
  6837. }
  6838. // put it in the variant
  6839. VariantInit(pPhones);
  6840. pPhones->vt = VT_DISPATCH;
  6841. pPhones->pdispVal = pDisp;
  6842. }
  6843. LOG((TL_TRACE, "get_Phones - exit - return %lx", hr ));
  6844. return hr;
  6845. }
  6846. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6847. // Class : CAddress
  6848. // Interface : ITAddress2
  6849. // Method : EnumeratePhones
  6850. //
  6851. //
  6852. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6853. HRESULT
  6854. CAddress::EnumeratePhones(
  6855. IEnumPhone ** ppEnumPhone
  6856. )
  6857. {
  6858. HRESULT hr;
  6859. PhoneArray PhoneArray;
  6860. LOG((TL_TRACE, "EnumeratePhones - enter"));
  6861. LOG((TL_TRACE, " ppEnumPhone----->%p", ppEnumPhone ));
  6862. if ( TAPIIsBadWritePtr( ppEnumPhone, sizeof( IEnumPhone * ) ) )
  6863. {
  6864. LOG((TL_ERROR, "EnumeratePhones - bad pointer"));
  6865. return E_POINTER;
  6866. }
  6867. hr = GetPhoneArrayFromTapiAndPrune( &PhoneArray, FALSE );
  6868. if ( SUCCEEDED(hr) )
  6869. {
  6870. //
  6871. // create the enumerator
  6872. //
  6873. CComObject< CTapiEnum< IEnumPhone, ITPhone, &IID_IEnumPhone > > * p;
  6874. hr = CComObject< CTapiEnum< IEnumPhone, ITPhone, &IID_IEnumPhone > >
  6875. ::CreateInstance( &p );
  6876. if (S_OK != hr)
  6877. {
  6878. LOG((TL_ERROR, "EnumeratePhones - could not create enum" ));
  6879. PhoneArray.Shutdown();
  6880. return hr;
  6881. }
  6882. Lock();
  6883. // initialize it with our phone list
  6884. p->Initialize( PhoneArray );
  6885. Unlock();
  6886. PhoneArray.Shutdown();
  6887. //
  6888. // return it
  6889. //
  6890. *ppEnumPhone = p;
  6891. }
  6892. LOG((TL_TRACE, "EnumeratePhones - exit - return %lx", hr ));
  6893. return hr;
  6894. }
  6895. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6896. // Class : CAddress
  6897. // Interface : ITAddress2
  6898. // Method : get_PreferredPhones
  6899. //
  6900. //
  6901. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6902. HRESULT
  6903. CAddress::get_PreferredPhones(
  6904. VARIANT * pPhones
  6905. )
  6906. {
  6907. HRESULT hr;
  6908. IDispatch * pDisp;
  6909. PhoneArray PhoneArray;
  6910. LOG((TL_TRACE, "get_PreferredPhones enter"));
  6911. if ( TAPIIsBadWritePtr( pPhones, sizeof( VARIANT ) ) )
  6912. {
  6913. LOG((TL_ERROR, "get_PreferredPhones - bad pointer"));
  6914. return E_POINTER;
  6915. }
  6916. hr = GetPhoneArrayFromTapiAndPrune( &PhoneArray, TRUE );
  6917. if ( SUCCEEDED(hr) )
  6918. {
  6919. CComObject< CTapiCollection< ITPhone > > * p;
  6920. CComObject< CTapiCollection< ITPhone > >::CreateInstance( &p );
  6921. if (NULL == p)
  6922. {
  6923. LOG((TL_ERROR, "get_PreferredPhones - could not create collection" ));
  6924. PhoneArray.Shutdown();
  6925. return E_OUTOFMEMORY;
  6926. }
  6927. // get the IDispatch interface
  6928. hr = p->_InternalQueryInterface( IID_IDispatch, (void **) &pDisp );
  6929. if (S_OK != hr)
  6930. {
  6931. LOG((TL_ERROR, "get_PreferredPhones - could not get IDispatch interface" ));
  6932. delete p;
  6933. return hr;
  6934. }
  6935. Lock();
  6936. // initialize
  6937. hr = p->Initialize( PhoneArray );
  6938. Unlock();
  6939. PhoneArray.Shutdown();
  6940. if (S_OK != hr)
  6941. {
  6942. LOG((TL_ERROR, "get_PreferredPhones - could not initialize collection" ));
  6943. pDisp->Release();
  6944. return hr;
  6945. }
  6946. // put it in the variant
  6947. VariantInit(pPhones);
  6948. pPhones->vt = VT_DISPATCH;
  6949. pPhones->pdispVal = pDisp;
  6950. }
  6951. LOG((TL_TRACE, "get_PreferredPhones - exit - return %lx", hr ));
  6952. return hr;
  6953. }
  6954. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6955. // Class : CAddress
  6956. // Interface : ITAddress2
  6957. // Method : EnumeratePreferredPhones
  6958. //
  6959. //
  6960. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  6961. HRESULT
  6962. CAddress::EnumeratePreferredPhones(
  6963. IEnumPhone ** ppEnumPhone
  6964. )
  6965. {
  6966. HRESULT hr;
  6967. PhoneArray PhoneArray;
  6968. LOG((TL_TRACE, "EnumeratePreferredPhones - enter"));
  6969. LOG((TL_TRACE, " ppEnumPhone----->%p", ppEnumPhone ));
  6970. if ( TAPIIsBadWritePtr( ppEnumPhone, sizeof( IEnumPhone * ) ) )
  6971. {
  6972. LOG((TL_ERROR, "EnumeratePreferredPhones - bad pointer"));
  6973. return E_POINTER;
  6974. }
  6975. hr = GetPhoneArrayFromTapiAndPrune( &PhoneArray, TRUE );
  6976. if ( SUCCEEDED(hr) )
  6977. {
  6978. //
  6979. // create the enumerator
  6980. //
  6981. CComObject< CTapiEnum< IEnumPhone, ITPhone, &IID_IEnumPhone > > * p;
  6982. hr = CComObject< CTapiEnum< IEnumPhone, ITPhone, &IID_IEnumPhone > >
  6983. ::CreateInstance( &p );
  6984. if (S_OK != hr)
  6985. {
  6986. LOG((TL_ERROR, "EnumeratePreferredPhones - could not create enum" ));
  6987. PhoneArray.Shutdown();
  6988. return hr;
  6989. }
  6990. Lock();
  6991. // initialize it with our phone list
  6992. p->Initialize( PhoneArray );
  6993. Unlock();
  6994. PhoneArray.Shutdown();
  6995. //
  6996. // return it
  6997. //
  6998. *ppEnumPhone = p;
  6999. }
  7000. LOG((TL_TRACE, "EnumeratePreferredPhones - exit - return %lx", hr ));
  7001. return hr;
  7002. }
  7003. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7004. // Class : CAddress
  7005. // Interface : ITAddress2
  7006. // Method : GetPhoneFromTerminal
  7007. //
  7008. //
  7009. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7010. HRESULT CAddress::GetPhoneFromTerminal(
  7011. ITTerminal * pTerminal,
  7012. ITPhone ** ppPhone
  7013. )
  7014. {
  7015. HRESULT hr = E_FAIL;
  7016. PhoneArray PhoneArray;
  7017. ITStaticAudioTerminal * pStaticAudioTerminal;
  7018. LONG lMediaType;
  7019. TERMINAL_DIRECTION nDir;
  7020. LOG((TL_TRACE, "GetPhoneFromTerminal - enter"));
  7021. if ( TAPIIsBadWritePtr( ppPhone, sizeof( ITPhone * ) ) ||
  7022. IsBadReadPtr( pTerminal, sizeof( ITTerminal ) ) )
  7023. {
  7024. LOG((TL_ERROR, "GetPhoneFromTerminal - bad pointer"));
  7025. return E_POINTER;
  7026. }
  7027. *ppPhone = NULL;
  7028. if ( SUCCEEDED(pTerminal->get_MediaType(&lMediaType)) &&
  7029. SUCCEEDED(pTerminal->get_Direction(&nDir)) &&
  7030. (lMediaType == TAPIMEDIATYPE_AUDIO) )
  7031. {
  7032. hr = pTerminal->QueryInterface(IID_ITStaticAudioTerminal, (void **) &pStaticAudioTerminal);
  7033. if ( SUCCEEDED(hr) )
  7034. {
  7035. LONG lWaveId;
  7036. hr = pStaticAudioTerminal->get_WaveId(&lWaveId);
  7037. if ( SUCCEEDED(hr) )
  7038. {
  7039. LOG((TL_INFO, "GetPhoneFromTerminal - got terminal wave id %d", lWaveId));
  7040. hr = GetPhoneArrayFromTapiAndPrune( &PhoneArray, FALSE );
  7041. if ( SUCCEEDED(hr) )
  7042. {
  7043. ITPhone * pPhone;
  7044. CPhone * pCPhone;
  7045. int iPhoneCount;
  7046. hr = TAPI_E_NODEVICE;
  7047. for(iPhoneCount = 0; iPhoneCount < PhoneArray.GetSize(); iPhoneCount++)
  7048. {
  7049. pPhone = PhoneArray[iPhoneCount];
  7050. pCPhone = dynamic_cast<CPhone *>(pPhone);
  7051. if ( NULL == pCPhone )
  7052. {
  7053. //
  7054. // We have a bad pointer in our phone array.
  7055. // Lets skip it and move on.
  7056. //
  7057. _ASSERTE(FALSE);
  7058. continue;
  7059. }
  7060. if (pCPhone->IsPhoneUsingWaveID( lWaveId, nDir ))
  7061. {
  7062. *ppPhone = pPhone;
  7063. pPhone->AddRef();
  7064. hr = S_OK;
  7065. break;
  7066. }
  7067. }
  7068. PhoneArray.Shutdown();
  7069. }
  7070. }
  7071. pStaticAudioTerminal->Release();
  7072. }
  7073. }
  7074. LOG((TL_TRACE, "GetPhoneFromTerminal - exit - return %lx", hr ));
  7075. return hr;
  7076. }
  7077. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7078. // Class : CAddress
  7079. // Interface : ITAddress2
  7080. // Method : put_EventFilter
  7081. //
  7082. //
  7083. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7084. HRESULT
  7085. CAddress::put_EventFilter(
  7086. TAPI_EVENT TapiEvent,
  7087. long lSubEvent,
  7088. VARIANT_BOOL bEnable
  7089. )
  7090. {
  7091. LOG((TL_TRACE, "put_EventFilter - enter"));
  7092. //
  7093. // Validates the pair TapiEvent - lSubEvent
  7094. // Accept also all subevents
  7095. //
  7096. if( !m_EventMasks.IsSubEventValid( TapiEvent, lSubEvent, TRUE, FALSE) )
  7097. {
  7098. LOG((TL_ERROR, "put_EventFilter - "
  7099. "This event can't be set: %x, return E_INVALIDARG", TapiEvent ));
  7100. return E_INVALIDARG;
  7101. }
  7102. // Enter critical section
  7103. Lock();
  7104. // Set the subevent flag
  7105. HRESULT hr = E_FAIL;
  7106. hr = SetSubEventFlag(
  7107. TapiEvent,
  7108. (DWORD)lSubEvent,
  7109. (bEnable == VARIANT_TRUE)
  7110. );
  7111. // Leave critical section
  7112. Unlock();
  7113. return hr;
  7114. }
  7115. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7116. // Class : CAddress
  7117. // Interface : ITAddress2
  7118. // Method : get_EventFilter
  7119. //
  7120. //
  7121. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7122. HRESULT
  7123. CAddress::get_EventFilter(
  7124. TAPI_EVENT TapiEvent,
  7125. long lSubEvent,
  7126. VARIANT_BOOL* pEnable
  7127. )
  7128. {
  7129. LOG((TL_TRACE, "get_EventFilter - enter"));
  7130. //
  7131. // Validates output argument
  7132. //
  7133. if( IsBadReadPtr(pEnable, sizeof(VARIANT_BOOL)) )
  7134. {
  7135. LOG((TL_ERROR, "get_EventFilter - "
  7136. "invalid VARIANT_BOOL pointer, return E_POINTER" ));
  7137. return E_POINTER;
  7138. }
  7139. //
  7140. // Validates the pair TapiEvent - lSubEvent
  7141. // Don't accept all subevents
  7142. //
  7143. if( !m_EventMasks.IsSubEventValid( TapiEvent, lSubEvent, FALSE, FALSE) )
  7144. {
  7145. LOG((TL_ERROR, "get_EventFilter - "
  7146. "This event can't be set: %x, return E_INVALIDARG", TapiEvent ));
  7147. return E_INVALIDARG;
  7148. }
  7149. // Enter critical section
  7150. Lock();
  7151. //
  7152. // Get the subevent mask for that (event, subevent) pair
  7153. //
  7154. BOOL bEnable = FALSE;
  7155. HRESULT hr = GetSubEventFlag(
  7156. TapiEvent,
  7157. (DWORD)lSubEvent,
  7158. &bEnable);
  7159. if( FAILED(hr) )
  7160. {
  7161. LOG((TL_ERROR, "get_EventFilter - "
  7162. "GetSubEventFlag failed, return 0x%08x", hr ));
  7163. // Leave critical section
  7164. Unlock();
  7165. return hr;
  7166. }
  7167. //
  7168. // Set the output argument
  7169. //
  7170. *pEnable = bEnable ? VARIANT_TRUE : VARIANT_FALSE;
  7171. // Leave critical section
  7172. Unlock();
  7173. LOG((TL_TRACE, "get_EventFilter - exit S_OK"));
  7174. return S_OK;
  7175. }
  7176. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7177. // Class : CAddress
  7178. // Interface : ITAddress2
  7179. // Method : DeviceSpecific
  7180. //
  7181. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7182. HRESULT CAddress::DeviceSpecific(
  7183. IN ITCallInfo *pCall,
  7184. IN BYTE *pbDataArray,
  7185. IN DWORD dwSize
  7186. )
  7187. {
  7188. LOG((TL_TRACE, "DeviceSpecific - enter"));
  7189. //
  7190. // check if arguments are any good
  7191. //
  7192. if ( NULL == pbDataArray )
  7193. {
  7194. LOG((TL_ERROR, "DeviceSpecific - pbDataArray is NULL. E_INVALIDARG"));
  7195. return E_INVALIDARG;
  7196. }
  7197. if ( 0 == dwSize )
  7198. {
  7199. LOG((TL_ERROR, "DeviceSpecific - dwSize is 0. E_INVALIDARG"));
  7200. return E_INVALIDARG;
  7201. }
  7202. //
  7203. // check if the buffer is valid
  7204. //
  7205. if ( IsBadReadPtr(pbDataArray, dwSize) )
  7206. {
  7207. LOG((TL_ERROR,
  7208. "DeviceSpecific - bad array passed in [%p] of size %ld",
  7209. pbDataArray, dwSize));
  7210. return E_POINTER;
  7211. }
  7212. //
  7213. // see if the call is obviously bad, and try to get call object pointer if
  7214. // it is good
  7215. //
  7216. CCall *pCallObject = NULL;
  7217. if ( (NULL != pCall) )
  7218. {
  7219. //
  7220. // does it point to readable memory at all?
  7221. //
  7222. if ( IsBadReadPtr(pCall, sizeof(ITCallInfo)) )
  7223. {
  7224. LOG((TL_ERROR, "DeviceSpecific - unreadable call pointer [%p]", pCall));
  7225. return E_POINTER;
  7226. }
  7227. //
  7228. // see if call is pointing to a real call object
  7229. //
  7230. try
  7231. {
  7232. pCallObject = dynamic_cast<CCall*>(pCall);
  7233. }
  7234. catch (...)
  7235. {
  7236. //
  7237. // call pointer is really really bad
  7238. //
  7239. LOG((TL_ERROR,
  7240. "DeviceSpecific - exception casting call pointer to a call object, bad call [%p]",
  7241. pCall));
  7242. }
  7243. //
  7244. // if we could not get the call object pointer, this is not a good call
  7245. //
  7246. if (NULL == pCallObject)
  7247. {
  7248. LOG((TL_ERROR,
  7249. "DeviceSpecific - could not get call object from call pointer -- bad call pointer argument [%p]",
  7250. pCall));
  7251. return E_POINTER;
  7252. }
  7253. } // received call pointer that is NULL?
  7254. //
  7255. // by this point we know pCall is either NULL or we have a call pointer
  7256. // that seems (but not guaranteed) to be good
  7257. //
  7258. //
  7259. // prepare all the data for the call to lineDevSpecific
  7260. //
  7261. //
  7262. // get hcall from the call
  7263. //
  7264. HCALL hCall = NULL;
  7265. if (NULL != pCallObject)
  7266. {
  7267. hCall = pCallObject->GetHCall();
  7268. //
  7269. // if we there is no call handle, return an error -- the app did not
  7270. // called Connect on the call
  7271. //
  7272. if (NULL == hCall)
  7273. {
  7274. LOG((TL_ERROR,
  7275. "DeviceSpecific - no call handle. hr = TAPI_E_INVALCALLSTATE",
  7276. pCall));
  7277. return TAPI_E_INVALCALLSTATE;
  7278. }
  7279. }
  7280. //
  7281. // starting to access data members. lock.
  7282. //
  7283. Lock();
  7284. //
  7285. // get a line to use to communicate devspecific information
  7286. //
  7287. AddressLineStruct *pAddressLine = NULL;
  7288. HRESULT hr = FindOrOpenALine(m_dwMediaModesSupported, &pAddressLine);
  7289. if (FAILED(hr))
  7290. {
  7291. Unlock();
  7292. LOG((TL_TRACE, "DeviceSpecific - FindOrOpenALine failed. hr = %lx", hr));
  7293. return hr;
  7294. }
  7295. DWORD dwAddressID = m_dwAddressID;
  7296. Unlock();
  7297. //
  7298. // make the tapisrv call
  7299. //
  7300. hr = lineDevSpecific( pAddressLine->t3Line.hLine,
  7301. dwAddressID,
  7302. hCall,
  7303. pbDataArray,
  7304. dwSize
  7305. );
  7306. //
  7307. // no longer need the line. if registered for address notifications, the
  7308. // line will remain opened. otherwise, if no one has the line open, it will
  7309. // close -- we are not processing incoming events anyway.
  7310. //
  7311. MaybeCloseALine(&pAddressLine);
  7312. LOG((TL_TRACE, "DeviceSpecific - exit. hr = %lx", hr));
  7313. return hr;
  7314. }
  7315. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7316. // Class : CAddress
  7317. // Interface : ITAddress2
  7318. // Method : DeviceSpecificVariant
  7319. //
  7320. // this is the scriptable version of DeviceSpecific
  7321. //
  7322. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7323. HRESULT CAddress::DeviceSpecificVariant(
  7324. IN ITCallInfo *pCall,
  7325. IN VARIANT varDevSpecificByteArray
  7326. )
  7327. {
  7328. LOG((TL_TRACE, "DeviceSpecificVariant - enter"));
  7329. //
  7330. // extract buffer from the variant
  7331. //
  7332. DWORD dwByteArraySize = 0;
  7333. BYTE *pBuffer = NULL;
  7334. HRESULT hr = E_FAIL;
  7335. hr = MakeBufferFromVariant(varDevSpecificByteArray, &dwByteArraySize, &pBuffer);
  7336. if (FAILED(hr))
  7337. {
  7338. LOG((TL_TRACE, "DeviceSpecificVariant - MakeBufferFromVariant failed. hr = %lx", hr));
  7339. return hr;
  7340. }
  7341. //
  7342. // call the non-scriptable version and pass it the nonscriptable implementation
  7343. //
  7344. hr = DeviceSpecific(pCall, pBuffer, dwByteArraySize);
  7345. //
  7346. // success or failure, free the buffer allocated by MakeBufferFromVariant
  7347. //
  7348. ClientFree(pBuffer);
  7349. pBuffer = NULL;
  7350. //
  7351. // log rc and exit
  7352. //
  7353. LOG((TL_TRACE, "DeviceSpecificVariant - exit. hr = %lx", hr));
  7354. return hr;
  7355. }
  7356. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7357. // Class : CAddress
  7358. // Interface : ITAddress2
  7359. // Method : NegotiateExtVersion
  7360. //
  7361. //
  7362. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7363. HRESULT CAddress::NegotiateExtVersion (
  7364. IN long lLowVersion,
  7365. IN long lHighVersion,
  7366. OUT long *plExtVersion
  7367. )
  7368. {
  7369. LOG((TL_TRACE, "NegotiateExtVersion - enter"));
  7370. //
  7371. // make sure the out parameter is writable
  7372. //
  7373. if (IsBadWritePtr(plExtVersion, sizeof(long)) )
  7374. {
  7375. LOG((TL_ERROR, "NegotiateExtVersion - output arg [%p] not writeable", plExtVersion));
  7376. return E_POINTER;
  7377. }
  7378. Lock();
  7379. //
  7380. // make a call to tapisrv
  7381. //
  7382. DWORD dwNegotiatedVersion = 0;
  7383. LONG lResult = lineNegotiateExtVersion( m_hLineApp,
  7384. m_dwDeviceID,
  7385. m_dwAPIVersion,
  7386. lLowVersion,
  7387. lHighVersion,
  7388. &dwNegotiatedVersion );
  7389. Unlock();
  7390. HRESULT hr = mapTAPIErrorCode(lResult);
  7391. //
  7392. // return the value on success
  7393. //
  7394. if ( SUCCEEDED(hr) )
  7395. {
  7396. LOG((TL_TRACE, "NegotiateExtVersion - negotiated version %ld", dwNegotiatedVersion));
  7397. *plExtVersion = dwNegotiatedVersion;
  7398. }
  7399. LOG((TL_TRACE, "NegotiateExtVersion - exit. hr = %lx", hr));
  7400. return hr;
  7401. }
  7402. //
  7403. // ----------------------- CAddressTranslationInfo -----------------------------
  7404. //
  7405. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7406. // Class : CAddressTranslationInfo
  7407. // Method : Initialize
  7408. //
  7409. //
  7410. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7411. HRESULT
  7412. CAddressTranslationInfo::Initialize(
  7413. PWSTR pszDialableString,
  7414. PWSTR pszDisplayableString,
  7415. DWORD dwCurrentCountry,
  7416. DWORD dwDestCountry,
  7417. DWORD dwTranslateResults
  7418. )
  7419. {
  7420. HRESULT hr = S_OK;
  7421. LOG((TL_TRACE, "Initialize - enter" ));
  7422. Lock();
  7423. m_dwCurrentCountryCode = dwCurrentCountry;
  7424. m_dwDestinationCountryCode = dwDestCountry;
  7425. m_dwTranslationResults = dwTranslateResults;
  7426. m_szDialableString = NULL;
  7427. m_szDialableString = NULL;
  7428. // copy the Dialable String
  7429. if (pszDialableString!= NULL)
  7430. {
  7431. m_szDialableString = (PWSTR) ClientAlloc((lstrlenW(pszDialableString) + 1) * sizeof (WCHAR));
  7432. if (m_szDialableString != NULL)
  7433. {
  7434. lstrcpyW(m_szDialableString, pszDialableString);
  7435. // Now copy the Displayable String
  7436. if (pszDisplayableString!= NULL)
  7437. {
  7438. m_szDisplayableString = (PWSTR) ClientAlloc((lstrlenW(pszDisplayableString) + 1) * sizeof (WCHAR));
  7439. if (m_szDisplayableString != NULL)
  7440. {
  7441. lstrcpyW(m_szDisplayableString, pszDisplayableString);
  7442. }
  7443. else
  7444. {
  7445. LOG((TL_ERROR, "Initialize - Alloc m_szDisplayableString failed" ));
  7446. ClientFree( pszDialableString );
  7447. pszDialableString = NULL;
  7448. hr = E_OUTOFMEMORY;
  7449. }
  7450. }
  7451. }
  7452. else
  7453. {
  7454. LOG((TL_ERROR, "Initialize - Alloc m_szDialableString failed" ));
  7455. hr = E_OUTOFMEMORY;
  7456. }
  7457. }
  7458. Unlock();
  7459. LOG((TL_TRACE, hr, "Initialize - exit" ));
  7460. return hr;
  7461. }
  7462. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7463. // Class : CAddressTranslationInfo
  7464. // Method : FinalRelease
  7465. //
  7466. //
  7467. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7468. void CAddressTranslationInfo::FinalRelease()
  7469. {
  7470. LOG((TL_TRACE, "FinalRelease - enter" ));
  7471. if (m_szDialableString != NULL)
  7472. {
  7473. ClientFree( m_szDialableString);
  7474. }
  7475. if (m_szDisplayableString != NULL)
  7476. {
  7477. ClientFree( m_szDisplayableString);
  7478. }
  7479. LOG((TL_TRACE, "FinalRelease - exit" ));
  7480. }
  7481. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7482. // Class : CAddressTranslationInfo
  7483. // Interface : ITAddressTranslationInfo
  7484. // Method : get_CurrentCountryCode
  7485. //
  7486. //
  7487. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7488. STDMETHODIMP
  7489. CAddressTranslationInfo::get_CurrentCountryCode(long * CountryCode )
  7490. {
  7491. HRESULT hr = S_OK;
  7492. LOG((TL_TRACE, "get_CurrentCountryCode - enter" ));
  7493. Lock();
  7494. if ( TAPIIsBadWritePtr( CountryCode , sizeof(long) ) )
  7495. {
  7496. LOG((TL_ERROR, "get_CurrentCountryCode - Bad Pointer" ));
  7497. hr = E_POINTER;
  7498. }
  7499. else // Ok Pointer
  7500. {
  7501. *CountryCode = m_dwCurrentCountryCode;
  7502. }
  7503. Unlock();
  7504. LOG((TL_TRACE, hr, "get_CurrentCountryCode - exit" ));
  7505. return hr;
  7506. }
  7507. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7508. // Class : CAddressTranslationInfo
  7509. // Interface : ITAddressTranslationInfo
  7510. // Method : get_DestinationCountryCode
  7511. //
  7512. //
  7513. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7514. STDMETHODIMP
  7515. CAddressTranslationInfo::get_DestinationCountryCode(long * CountryCode )
  7516. {
  7517. HRESULT hr = S_OK;
  7518. LOG((TL_TRACE, "get_DestinationCountryCode - enter" ));
  7519. Lock();
  7520. if ( TAPIIsBadWritePtr( CountryCode , sizeof(long) ) )
  7521. {
  7522. LOG((TL_ERROR, "get_DestinationCountryCode - Bad Pointer" ));
  7523. hr = E_POINTER;
  7524. }
  7525. else // Ok Pointer
  7526. {
  7527. *CountryCode = m_dwDestinationCountryCode;
  7528. }
  7529. Unlock();
  7530. LOG((TL_TRACE, hr, "get_DestinationCountryCode - exit" ));
  7531. return hr;
  7532. }
  7533. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7534. // Class : CAddressTranslationInfo
  7535. // Interface : ITAddressTranslationInfo
  7536. // Method : get_TranslationResult
  7537. //
  7538. //
  7539. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7540. STDMETHODIMP
  7541. CAddressTranslationInfo::get_TranslationResults(long * Results )
  7542. {
  7543. HRESULT hr = S_OK;
  7544. LOG((TL_TRACE, "get_TranslationResults - enter" ));
  7545. Lock();
  7546. if ( TAPIIsBadWritePtr( Results , sizeof(long) ) )
  7547. {
  7548. LOG((TL_ERROR, "get_TranslationResults - Bad Pointer" ));
  7549. hr = E_POINTER;
  7550. }
  7551. else // Ok Pointer
  7552. {
  7553. *Results = m_dwTranslationResults;
  7554. }
  7555. Unlock();
  7556. LOG((TL_TRACE, hr, "get_TranslationResults - exit" ));
  7557. return hr;
  7558. }
  7559. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7560. // Class : CAddressTranslationInfo
  7561. // Interface : ITAddressTranslationInfo
  7562. // Method : get_DialableString
  7563. //
  7564. //
  7565. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7566. STDMETHODIMP
  7567. CAddressTranslationInfo::get_DialableString(BSTR * ppDialableString)
  7568. {
  7569. HRESULT hr = S_OK;
  7570. LOG((TL_TRACE, "get_DialableString - enter" ));
  7571. Lock();
  7572. if ( TAPIIsBadWritePtr( ppDialableString, sizeof(BSTR) ) )
  7573. {
  7574. LOG((TL_ERROR, "get_DialableString - Bad Pointer" ));
  7575. hr = E_POINTER;
  7576. }
  7577. else // Ok Pointer
  7578. {
  7579. *ppDialableString = SysAllocString( m_szDialableString );
  7580. if ( ( NULL == *ppDialableString ) && ( NULL != m_szDialableString ) )
  7581. {
  7582. LOG((TL_TRACE, "SysAllocString Failed" ));
  7583. hr = E_OUTOFMEMORY;
  7584. }
  7585. }
  7586. Unlock();
  7587. LOG((TL_TRACE, hr, "get_DialableString - exit" ));
  7588. return hr;
  7589. }
  7590. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7591. // Class : CAddressTranslationInfo
  7592. // Interface : ITAddressTranslationInfo
  7593. // Method : get_DisplayableString
  7594. //
  7595. //
  7596. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7597. STDMETHODIMP
  7598. CAddressTranslationInfo::get_DisplayableString(BSTR * ppDisplayableString)
  7599. {
  7600. HRESULT hr = S_OK;
  7601. LOG((TL_TRACE, "get_DisplayableString - enter" ));
  7602. Lock();
  7603. if ( TAPIIsBadWritePtr( ppDisplayableString, sizeof(BSTR) ) )
  7604. {
  7605. LOG((TL_ERROR, "get_DisplayableString - Bad Pointer" ));
  7606. hr = E_POINTER;
  7607. }
  7608. else // Ok Pointer
  7609. {
  7610. *ppDisplayableString = SysAllocString( m_szDisplayableString );
  7611. if ( ( NULL == *ppDisplayableString ) && ( NULL != m_szDisplayableString ) )
  7612. {
  7613. LOG((TL_TRACE, "SysAllocString Failed" ));
  7614. hr = E_OUTOFMEMORY;
  7615. }
  7616. }
  7617. Unlock();
  7618. LOG((TL_TRACE, hr, "get_DisplayableString - exit" ));
  7619. return hr;
  7620. }
  7621. //
  7622. // ----------------------- CCallingCard -----------------------------
  7623. //
  7624. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7625. // Class : CCallingCard
  7626. // Method : Initialize
  7627. //
  7628. //
  7629. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7630. HRESULT
  7631. CCallingCard::Initialize(
  7632. PWSTR pszCardName,
  7633. PWSTR pszSameAreaDialingRule,
  7634. PWSTR pszLongDistanceDialingRule,
  7635. PWSTR pszInternationalDialingRule,
  7636. DWORD dwPermanentCardID,
  7637. DWORD dwNumberOfDigits,
  7638. DWORD dwOptions
  7639. )
  7640. {
  7641. HRESULT hr = S_OK;
  7642. LOG((TL_TRACE, "Initialize - enter" ));
  7643. Lock();
  7644. m_dwPermanentCardID = dwPermanentCardID;
  7645. m_dwNumberOfDigits = dwNumberOfDigits;
  7646. m_dwOptions = dwOptions;
  7647. m_szCardName = NULL;
  7648. m_szSameAreaDialingRule = NULL;
  7649. m_szLongDistanceDialingRule = NULL;
  7650. m_szInternationalDialingRule = NULL;
  7651. // copy the Card Name
  7652. if (pszCardName != NULL)
  7653. {
  7654. m_szCardName = (PWSTR) ClientAlloc((lstrlenW(pszCardName) + 1) * sizeof (WCHAR));
  7655. if (m_szCardName != NULL)
  7656. {
  7657. lstrcpyW(m_szCardName, pszCardName);
  7658. }
  7659. else
  7660. {
  7661. LOG((TL_ERROR, "Initialize - Alloc m_szCardName failed" ));
  7662. hr = E_OUTOFMEMORY;
  7663. }
  7664. }
  7665. // copy the Same Area Dialing Rule
  7666. if (pszSameAreaDialingRule != NULL)
  7667. {
  7668. m_szSameAreaDialingRule = (PWSTR) ClientAlloc((lstrlenW(pszSameAreaDialingRule) + 1) * sizeof (WCHAR));
  7669. if (m_szSameAreaDialingRule != NULL)
  7670. {
  7671. lstrcpyW(m_szSameAreaDialingRule, pszSameAreaDialingRule);
  7672. }
  7673. else
  7674. {
  7675. LOG((TL_ERROR, "Initialize - Alloc m_szSameAreaDialingRule failed" ));
  7676. hr = E_OUTOFMEMORY;
  7677. }
  7678. }
  7679. // copy the Long Distance Dialing Rule
  7680. if (pszLongDistanceDialingRule != NULL)
  7681. {
  7682. m_szLongDistanceDialingRule = (PWSTR) ClientAlloc((lstrlenW(pszLongDistanceDialingRule) + 1) * sizeof (WCHAR));
  7683. if (m_szLongDistanceDialingRule != NULL)
  7684. {
  7685. lstrcpyW(m_szLongDistanceDialingRule, pszLongDistanceDialingRule);
  7686. }
  7687. else
  7688. {
  7689. LOG((TL_ERROR, "Initialize - Alloc m_szLongDistanceDialingRule failed" ));
  7690. hr = E_OUTOFMEMORY;
  7691. }
  7692. }
  7693. //copy the International Dialing Rule
  7694. if (pszInternationalDialingRule != NULL)
  7695. {
  7696. m_szInternationalDialingRule = (PWSTR) ClientAlloc((lstrlenW(pszInternationalDialingRule) + 1) * sizeof (WCHAR));
  7697. if (m_szInternationalDialingRule != NULL)
  7698. {
  7699. lstrcpyW(m_szInternationalDialingRule, pszInternationalDialingRule);
  7700. }
  7701. else
  7702. {
  7703. LOG((TL_ERROR, "Initialize - Alloc m_szInternationalDialingRule failed" ));
  7704. hr = E_OUTOFMEMORY;
  7705. }
  7706. }
  7707. Unlock();
  7708. LOG((TL_TRACE, hr, "Initialize - exit" ));
  7709. return hr;
  7710. }
  7711. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7712. // Class : CCallingCard
  7713. // Method : FinalRelease
  7714. //
  7715. //
  7716. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7717. void CCallingCard::FinalRelease()
  7718. {
  7719. LOG((TL_TRACE, "FinalRelease - enter" ));
  7720. if (m_szCardName != NULL)
  7721. {
  7722. ClientFree(m_szCardName);
  7723. }
  7724. if (m_szSameAreaDialingRule != NULL)
  7725. {
  7726. ClientFree(m_szSameAreaDialingRule);
  7727. }
  7728. if (m_szLongDistanceDialingRule != NULL)
  7729. {
  7730. ClientFree(m_szLongDistanceDialingRule);
  7731. }
  7732. if (m_szInternationalDialingRule != NULL)
  7733. {
  7734. ClientFree(m_szInternationalDialingRule);
  7735. }
  7736. LOG((TL_TRACE, "FinalRelease - exit" ));
  7737. }
  7738. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7739. // Class : CCallingCard
  7740. // Interface : ITCallingCard
  7741. // Method : get_PermanentCardID
  7742. //
  7743. //
  7744. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7745. STDMETHODIMP
  7746. CCallingCard::get_PermanentCardID(long * ulCardID)
  7747. {
  7748. HRESULT hr = S_OK;
  7749. LOG((TL_TRACE, "get_PermanentCardID - enter" ));
  7750. Lock();
  7751. if ( TAPIIsBadWritePtr( ulCardID, sizeof(long) ) )
  7752. {
  7753. LOG((TL_ERROR, "get_PermanentCardID - Bad Pointer" ));
  7754. hr = E_POINTER;
  7755. }
  7756. else // Ok Pointer
  7757. {
  7758. *ulCardID= m_dwPermanentCardID;
  7759. }
  7760. Unlock();
  7761. LOG((TL_TRACE, hr, "get_PermanentCardID - exit" ));
  7762. return hr;
  7763. }
  7764. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7765. // Class : CCallingCard
  7766. // Interface : ITCallingCard
  7767. // Method : get_NumberOfDigits
  7768. //
  7769. //
  7770. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7771. STDMETHODIMP
  7772. CCallingCard::get_NumberOfDigits(long * ulDigits)
  7773. {
  7774. HRESULT hr = S_OK;
  7775. LOG((TL_TRACE, "get_NumberOfDigits - enter" ));
  7776. Lock();
  7777. if ( TAPIIsBadWritePtr( ulDigits, sizeof(long) ) )
  7778. {
  7779. LOG((TL_ERROR, "get_NumberOfDigits - Bad Pointer" ));
  7780. hr = E_POINTER;
  7781. }
  7782. else // Ok Pointer
  7783. {
  7784. *ulDigits= m_dwNumberOfDigits;
  7785. }
  7786. Unlock();
  7787. LOG((TL_TRACE, hr, "get_NumberOfDigits - exit" ));
  7788. return hr;
  7789. }
  7790. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7791. // Class : CCallingCard
  7792. // Interface : ITCallingCard
  7793. // Method : get_Options
  7794. //
  7795. //
  7796. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7797. STDMETHODIMP
  7798. CCallingCard::get_Options(long * ulOptions)
  7799. {
  7800. HRESULT hr = S_OK;
  7801. LOG((TL_TRACE, "get_Options - enter" ));
  7802. Lock();
  7803. if ( TAPIIsBadWritePtr( ulOptions, sizeof(long) ) )
  7804. {
  7805. LOG((TL_ERROR, "get_Options - Bad Pointer" ));
  7806. hr = E_POINTER;
  7807. }
  7808. else // Ok Pointer
  7809. {
  7810. *ulOptions= m_dwOptions;
  7811. }
  7812. Unlock();
  7813. LOG((TL_TRACE, hr, "get_Options - exit" ));
  7814. return hr;
  7815. }
  7816. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7817. // Class : CCallingCard
  7818. // Interface : ITCallingCard
  7819. // Method : get_CardName
  7820. //
  7821. //
  7822. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7823. STDMETHODIMP
  7824. CCallingCard::get_CardName(BSTR * ppCardName)
  7825. {
  7826. HRESULT hr = S_OK;
  7827. LOG((TL_TRACE, "get_CardName - enter" ));
  7828. Lock();
  7829. if ( TAPIIsBadWritePtr( ppCardName, sizeof(BSTR) ) )
  7830. {
  7831. LOG((TL_ERROR, "get_CardName - Bad Pointer" ));
  7832. hr = E_POINTER;
  7833. }
  7834. else // Ok Pointer
  7835. {
  7836. *ppCardName = SysAllocString( m_szCardName );
  7837. if ( ( NULL == *ppCardName ) && ( NULL != m_szCardName ) )
  7838. {
  7839. LOG((TL_TRACE, "SysAllocString Failed" ));
  7840. hr = E_OUTOFMEMORY;
  7841. }
  7842. }
  7843. Unlock();
  7844. LOG((TL_TRACE, hr, "get_CardName - exit" ));
  7845. return hr;
  7846. }
  7847. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7848. // Class : CCallingCard
  7849. // Interface : ITCallingCard
  7850. // Method : get_SameAreaDialingRule
  7851. //
  7852. //
  7853. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7854. STDMETHODIMP
  7855. CCallingCard::get_SameAreaDialingRule(BSTR * ppRule)
  7856. {
  7857. HRESULT hr = S_OK;
  7858. LOG((TL_TRACE, "get_SameAreaDialingRule - enter" ));
  7859. Lock();
  7860. if ( TAPIIsBadWritePtr( ppRule, sizeof(BSTR) ) )
  7861. {
  7862. LOG((TL_ERROR, "get_SameAreaDialingRule - Bad Pointer" ));
  7863. hr = E_POINTER;
  7864. }
  7865. else // Ok Pointer
  7866. {
  7867. *ppRule = SysAllocString( m_szSameAreaDialingRule );
  7868. if ( ( NULL == *ppRule ) && ( NULL != m_szSameAreaDialingRule ) )
  7869. {
  7870. LOG((TL_TRACE, "SysAllocString Failed" ));
  7871. hr = E_OUTOFMEMORY;
  7872. }
  7873. }
  7874. Unlock();
  7875. LOG((TL_TRACE, hr, "get_SameAreaDialingRule - exit" ));
  7876. return hr;
  7877. }
  7878. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7879. // Class : CCallingCard
  7880. // Interface : ITCallingCard
  7881. // Method : get_LongDistanceDialingRule
  7882. //
  7883. //
  7884. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7885. STDMETHODIMP
  7886. CCallingCard::get_LongDistanceDialingRule(BSTR * ppRule)
  7887. {
  7888. HRESULT hr = S_OK;
  7889. LOG((TL_TRACE, "get_LongDistanceDialingRule - enter" ));
  7890. Lock();
  7891. if ( TAPIIsBadWritePtr( ppRule, sizeof(BSTR) ) )
  7892. {
  7893. LOG((TL_ERROR, "get_LongDistanceDialingRule - Bad Pointer" ));
  7894. hr = E_POINTER;
  7895. }
  7896. else // Ok Pointer
  7897. {
  7898. *ppRule = SysAllocString( m_szLongDistanceDialingRule );
  7899. if ( ( NULL == *ppRule ) && ( NULL != m_szLongDistanceDialingRule ) )
  7900. {
  7901. LOG((TL_TRACE, "SysAllocString Failed" ));
  7902. hr = E_OUTOFMEMORY;
  7903. }
  7904. }
  7905. Unlock();
  7906. LOG((TL_TRACE, hr, "get_LongDistanceDialingRule - exit" ));
  7907. return hr;
  7908. }
  7909. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7910. // Class : CCallingCard
  7911. // Interface : ITCallingCard
  7912. // Method : get_InternationalDialingRule
  7913. //
  7914. //
  7915. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7916. STDMETHODIMP
  7917. CCallingCard::get_InternationalDialingRule(BSTR * ppRule)
  7918. {
  7919. HRESULT hr = S_OK;
  7920. LOG((TL_TRACE, "get_InternationalDialingRule - enter" ));
  7921. Lock();
  7922. if ( TAPIIsBadWritePtr( ppRule, sizeof(BSTR) ) )
  7923. {
  7924. LOG((TL_ERROR, "get_InternationalDialingRule - Bad Pointer" ));
  7925. hr = E_POINTER;
  7926. }
  7927. else // Ok Pointer
  7928. {
  7929. *ppRule = SysAllocString( m_szInternationalDialingRule );
  7930. if ( ( NULL == *ppRule ) && ( NULL != m_szInternationalDialingRule ) )
  7931. {
  7932. LOG((TL_TRACE, "SysAllocString Failed" ));
  7933. hr = E_OUTOFMEMORY;
  7934. }
  7935. }
  7936. Unlock();
  7937. LOG((TL_TRACE, hr, "get_InternationalDialingRule - exit" ));
  7938. return hr;
  7939. }
  7940. //
  7941. // ----------------------- CLocationInfo -----------------------------
  7942. //
  7943. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7944. // Class : CLocationInfo
  7945. // Method : Initialize
  7946. //
  7947. //
  7948. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  7949. HRESULT
  7950. CLocationInfo::Initialize(
  7951. PWSTR pszLocationName,
  7952. PWSTR pszCityCode,
  7953. PWSTR pszLocalAccessCode,
  7954. PWSTR pszLongDistanceAccessCode,
  7955. PWSTR pszTollPrefixList,
  7956. PWSTR pszCancelCallWaitingCode ,
  7957. DWORD dwPermanentLocationID,
  7958. DWORD dwCountryCode,
  7959. DWORD dwPreferredCardID,
  7960. DWORD dwCountryID,
  7961. DWORD dwOptions
  7962. )
  7963. {
  7964. HRESULT hr = S_OK;
  7965. LOG((TL_TRACE, "Initialize - enter" ));
  7966. Lock();
  7967. m_dwPermanentLocationID = dwPermanentLocationID;
  7968. m_dwCountryCode = dwCountryCode;
  7969. m_dwPreferredCardID = dwPreferredCardID;
  7970. m_dwCountryID = dwCountryID;
  7971. m_dwOptions = dwOptions;
  7972. m_szLocationName = NULL;
  7973. m_szCityCode = NULL;
  7974. m_szLocalAccessCode = NULL;
  7975. m_szLongDistanceAccessCode = NULL;
  7976. m_szTollPrefixList = NULL;
  7977. m_szCancelCallWaitingCode = NULL;
  7978. // copy the Location Name
  7979. if (pszLocationName!= NULL)
  7980. {
  7981. m_szLocationName = (PWSTR) ClientAlloc((lstrlenW(pszLocationName) + 1) * sizeof (WCHAR));
  7982. if (m_szLocationName != NULL)
  7983. {
  7984. lstrcpyW(m_szLocationName, pszLocationName);
  7985. }
  7986. else
  7987. {
  7988. LOG((TL_ERROR, "Initialize - Alloc m_szDialableString failed" ));
  7989. hr = E_OUTOFMEMORY;
  7990. }
  7991. }
  7992. // copy the City Code
  7993. if (pszCityCode != NULL)
  7994. {
  7995. m_szCityCode = (PWSTR) ClientAlloc((lstrlenW(pszCityCode) + 1) * sizeof (WCHAR));
  7996. if (m_szCityCode != NULL)
  7997. {
  7998. lstrcpyW(m_szCityCode , pszCityCode);
  7999. }
  8000. else
  8001. {
  8002. LOG((TL_ERROR, "Initialize - Alloc m_szCityCode failed" ));
  8003. hr = E_OUTOFMEMORY;
  8004. }
  8005. }
  8006. // copy the Local Access Code
  8007. if (pszLocalAccessCode != NULL)
  8008. {
  8009. m_szLocalAccessCode = (PWSTR) ClientAlloc((lstrlenW(pszLocalAccessCode) + 1) * sizeof (WCHAR));
  8010. if (m_szLocalAccessCode != NULL)
  8011. {
  8012. lstrcpyW(m_szLocalAccessCode, pszLocalAccessCode);
  8013. }
  8014. else
  8015. {
  8016. LOG((TL_ERROR, "Initialize - Alloc m_szLocalAccessCode failed" ));
  8017. hr = E_OUTOFMEMORY;
  8018. }
  8019. }
  8020. // copy the Long Distance Access Code
  8021. if (pszLongDistanceAccessCode != NULL)
  8022. {
  8023. m_szLongDistanceAccessCode = (PWSTR) ClientAlloc((lstrlenW(pszLongDistanceAccessCode) + 1) * sizeof (WCHAR));
  8024. if (m_szLongDistanceAccessCode != NULL)
  8025. {
  8026. lstrcpyW(m_szLongDistanceAccessCode, pszLongDistanceAccessCode);
  8027. }
  8028. else
  8029. {
  8030. LOG((TL_ERROR, "Initialize - Alloc m_szLongDistanceAccessCode failed" ));
  8031. hr = E_OUTOFMEMORY;
  8032. }
  8033. }
  8034. // copy the Toll Prefix List
  8035. if (pszTollPrefixList != NULL)
  8036. {
  8037. m_szTollPrefixList = (PWSTR) ClientAlloc((lstrlenW(pszTollPrefixList) + 1) * sizeof (WCHAR));
  8038. if (m_szTollPrefixList != NULL)
  8039. {
  8040. lstrcpyW(m_szTollPrefixList, pszTollPrefixList);
  8041. }
  8042. else
  8043. {
  8044. LOG((TL_ERROR, "Initialize - Alloc m_szTollPrefixList failed" ));
  8045. hr = E_OUTOFMEMORY;
  8046. }
  8047. }
  8048. // copy the Cancel Call Waiting Code
  8049. if (pszCancelCallWaitingCode != NULL)
  8050. {
  8051. m_szCancelCallWaitingCode = (PWSTR) ClientAlloc((lstrlenW(pszCancelCallWaitingCode) + 1) * sizeof (WCHAR));
  8052. if (m_szCancelCallWaitingCode != NULL)
  8053. {
  8054. lstrcpyW(m_szCancelCallWaitingCode, pszCancelCallWaitingCode);
  8055. }
  8056. else
  8057. {
  8058. LOG((TL_ERROR, "Initialize - Alloc m_szCancelCallWaitingCode failed" ));
  8059. hr = E_OUTOFMEMORY;
  8060. }
  8061. }
  8062. Unlock();
  8063. LOG((TL_TRACE, hr, "Initialize - exit" ));
  8064. return hr;
  8065. }
  8066. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8067. // Class : CLocationInfo
  8068. // Method : FinalRelease
  8069. //
  8070. //
  8071. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8072. void CLocationInfo::FinalRelease()
  8073. {
  8074. LOG((TL_TRACE, "FinalRelease - enter" ));
  8075. if (m_szLocationName != NULL)
  8076. {
  8077. ClientFree( m_szLocationName);
  8078. }
  8079. if (m_szCityCode != NULL)
  8080. {
  8081. ClientFree( m_szCityCode);
  8082. }
  8083. if (m_szLocalAccessCode != NULL)
  8084. {
  8085. ClientFree( m_szLocalAccessCode);
  8086. }
  8087. if (m_szLongDistanceAccessCode != NULL)
  8088. {
  8089. ClientFree( m_szLongDistanceAccessCode);
  8090. }
  8091. if (m_szTollPrefixList != NULL)
  8092. {
  8093. ClientFree( m_szTollPrefixList);
  8094. }
  8095. if (m_szCancelCallWaitingCode != NULL)
  8096. {
  8097. ClientFree( m_szCancelCallWaitingCode);
  8098. }
  8099. LOG((TL_TRACE, "FinalRelease - exit" ));
  8100. }
  8101. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8102. // Class : CLocationInfo
  8103. // Interface : ITLocationInfo
  8104. // Method : get_PermanentLocationID
  8105. //
  8106. //
  8107. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8108. STDMETHODIMP
  8109. CLocationInfo::get_PermanentLocationID(long * ulLocationID )
  8110. {
  8111. HRESULT hr = S_OK;
  8112. LOG((TL_TRACE, "get_PermanentLocationID - enter" ));
  8113. Lock();
  8114. if ( TAPIIsBadWritePtr( ulLocationID , sizeof(long) ) )
  8115. {
  8116. LOG((TL_ERROR, "get_PermanentLocationID - Bad Pointer" ));
  8117. hr = E_POINTER;
  8118. }
  8119. else // Ok Pointer
  8120. {
  8121. *ulLocationID = m_dwPermanentLocationID;
  8122. }
  8123. Unlock();
  8124. LOG((TL_TRACE, hr, "get_PermanentLocationID - exit" ));
  8125. return hr;
  8126. }
  8127. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8128. // Class : CLocationInfo
  8129. // Interface : ITLocationInfo
  8130. // Method : get_CountryCode
  8131. //
  8132. //
  8133. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8134. STDMETHODIMP
  8135. CLocationInfo::get_CountryCode(long * ulCountryCode)
  8136. {
  8137. HRESULT hr = S_OK;
  8138. LOG((TL_TRACE, "get_CountryCode - enter" ));
  8139. Lock();
  8140. if ( TAPIIsBadWritePtr( ulCountryCode, sizeof(long) ) )
  8141. {
  8142. LOG((TL_ERROR, "get_CountryCode - Bad Pointer" ));
  8143. hr = E_POINTER;
  8144. }
  8145. else // Ok Pointer
  8146. {
  8147. *ulCountryCode= m_dwCountryCode;
  8148. }
  8149. Unlock();
  8150. LOG((TL_TRACE, hr, "get_CountryCode - exit" ));
  8151. return hr;
  8152. }
  8153. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8154. // Class : CLocationInfo
  8155. // Interface : ITLocationInfo
  8156. // Method : get_CountryID
  8157. //
  8158. //
  8159. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8160. STDMETHODIMP
  8161. CLocationInfo::get_CountryID(long * ulCountryID)
  8162. {
  8163. HRESULT hr = S_OK;
  8164. LOG((TL_TRACE, "get_CountryID - enter" ));
  8165. Lock();
  8166. if ( TAPIIsBadWritePtr( ulCountryID, sizeof(long) ) )
  8167. {
  8168. LOG((TL_ERROR, "get_CountryID - Bad Pointer" ));
  8169. hr = E_POINTER;
  8170. }
  8171. else // Ok Pointer
  8172. {
  8173. *ulCountryID= m_dwCountryID;
  8174. }
  8175. Unlock();
  8176. LOG((TL_TRACE, hr, "get_CountryID - exit" ));
  8177. return hr;
  8178. }
  8179. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8180. // Class : CLocationInfo
  8181. // Interface : ITLocationInfo
  8182. // Method : get_Options
  8183. //
  8184. //
  8185. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8186. STDMETHODIMP
  8187. CLocationInfo::get_Options(long * Options)
  8188. {
  8189. HRESULT hr = S_OK;
  8190. LOG((TL_TRACE, "get_Options - enter" ));
  8191. Lock();
  8192. if ( TAPIIsBadWritePtr( Options, sizeof(long) ) )
  8193. {
  8194. LOG((TL_ERROR, "get_Options - Bad Pointer" ));
  8195. hr = E_POINTER;
  8196. }
  8197. else // Ok Pointer
  8198. {
  8199. *Options= m_dwOptions;
  8200. }
  8201. Unlock();
  8202. LOG((TL_TRACE, hr, "get_Options - exit" ));
  8203. return hr;
  8204. }
  8205. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8206. // Class : CLocationInfo
  8207. // Interface : ITLocationInfo
  8208. // Method : get_PreferredCardID
  8209. //
  8210. //
  8211. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8212. STDMETHODIMP
  8213. CLocationInfo::get_PreferredCardID(long * ulCardID)
  8214. {
  8215. HRESULT hr = S_OK;
  8216. LOG((TL_TRACE, "get_PreferredCardID - enter" ));
  8217. Lock();
  8218. if ( TAPIIsBadWritePtr( ulCardID, sizeof(long) ) )
  8219. {
  8220. LOG((TL_ERROR, "get_PreferredCardID - Bad Pointer" ));
  8221. hr = E_POINTER;
  8222. }
  8223. else // Ok Pointer
  8224. {
  8225. *ulCardID= m_dwPreferredCardID;
  8226. }
  8227. Unlock();
  8228. LOG((TL_TRACE, hr, "get_PreferredCardID - exit" ));
  8229. return hr;
  8230. }
  8231. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8232. // Class : CLocationInfo
  8233. // Interface : ITLocationInfo
  8234. // Method : get_LocationName
  8235. //
  8236. //
  8237. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8238. STDMETHODIMP
  8239. CLocationInfo::get_LocationName(BSTR * ppLocationName)
  8240. {
  8241. HRESULT hr = S_OK;
  8242. LOG((TL_TRACE, "get_LocationName - enter" ));
  8243. Lock();
  8244. if ( TAPIIsBadWritePtr( ppLocationName, sizeof(BSTR) ) )
  8245. {
  8246. LOG((TL_ERROR, "get_LocationName - Bad Pointer" ));
  8247. hr = E_POINTER;
  8248. }
  8249. else // Ok Pointer
  8250. {
  8251. *ppLocationName = SysAllocString( m_szLocationName );
  8252. if ( ( NULL == *ppLocationName ) && ( NULL != m_szLocationName ) )
  8253. {
  8254. LOG((TL_TRACE, "SysAllocString Failed" ));
  8255. hr = E_OUTOFMEMORY;
  8256. }
  8257. }
  8258. Unlock();
  8259. LOG((TL_TRACE, hr, "gget_LocationName - exit" ));
  8260. return hr;
  8261. }
  8262. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8263. // Class : CLocationInfo
  8264. // Interface : ITLocationInfo
  8265. // Method : get_CityCode
  8266. //
  8267. //
  8268. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8269. STDMETHODIMP
  8270. CLocationInfo::get_CityCode(BSTR * ppCode)
  8271. {
  8272. HRESULT hr = S_OK;
  8273. LOG((TL_TRACE, "get_CityCode - enter" ));
  8274. Lock();
  8275. if ( TAPIIsBadWritePtr( ppCode, sizeof(BSTR) ) )
  8276. {
  8277. LOG((TL_ERROR, "get_CityCode - Bad Pointer" ));
  8278. hr = E_POINTER;
  8279. }
  8280. else // Ok Pointer
  8281. {
  8282. *ppCode = SysAllocString( m_szCityCode );
  8283. if ( ( NULL == *ppCode ) && ( NULL != m_szCityCode ) )
  8284. {
  8285. LOG((TL_TRACE, "SysAllocString Failed" ));
  8286. hr = E_OUTOFMEMORY;
  8287. }
  8288. }
  8289. Unlock();
  8290. LOG((TL_TRACE, hr, "get_CityCode - exit" ));
  8291. return hr;
  8292. }
  8293. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8294. // Class : CLocationInfo
  8295. // Interface : ITLocationInfo
  8296. // Method : get_LocalAccessCode
  8297. //
  8298. //
  8299. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8300. STDMETHODIMP
  8301. CLocationInfo::get_LocalAccessCode(BSTR * ppCode)
  8302. {
  8303. HRESULT hr = S_OK;
  8304. LOG((TL_TRACE, "get_LocalAccessCode - enter" ));
  8305. Lock();
  8306. if ( TAPIIsBadWritePtr( ppCode, sizeof(BSTR) ) )
  8307. {
  8308. LOG((TL_ERROR, "get_LocalAccessCode - Bad Pointer" ));
  8309. hr = E_POINTER;
  8310. }
  8311. else // Ok Pointer
  8312. {
  8313. *ppCode = SysAllocString( m_szLocalAccessCode );
  8314. if ( ( NULL == *ppCode ) && ( NULL != m_szLocalAccessCode ) )
  8315. {
  8316. LOG((TL_TRACE, "SysAllocString Failed" ));
  8317. hr = E_OUTOFMEMORY;
  8318. }
  8319. }
  8320. Unlock();
  8321. LOG((TL_TRACE, hr, "get_LocalAccessCode - exit" ));
  8322. return hr;
  8323. }
  8324. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8325. // Class : CLocationInfo
  8326. // Interface : ITLocationInfo
  8327. // Method : get_LongDistanceAccessCode
  8328. //
  8329. //
  8330. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8331. STDMETHODIMP
  8332. CLocationInfo::get_LongDistanceAccessCode(BSTR * ppCode )
  8333. {
  8334. HRESULT hr = S_OK;
  8335. LOG((TL_TRACE, "get_LongDistanceAccessCode - enter" ));
  8336. Lock();
  8337. if ( TAPIIsBadWritePtr( ppCode , sizeof(BSTR) ) )
  8338. {
  8339. LOG((TL_ERROR, "get_LongDistanceAccessCode - Bad Pointer" ));
  8340. hr = E_POINTER;
  8341. }
  8342. else // Ok Pointer
  8343. {
  8344. *ppCode = SysAllocString( m_szLongDistanceAccessCode);
  8345. if ( ( NULL == *ppCode ) && ( NULL != m_szLongDistanceAccessCode ) )
  8346. {
  8347. LOG((TL_TRACE, "SysAllocString Failed" ));
  8348. hr = E_OUTOFMEMORY;
  8349. }
  8350. }
  8351. Unlock();
  8352. LOG((TL_TRACE, hr, "get_LongDistanceAccessCode - exit" ));
  8353. return hr;
  8354. }
  8355. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8356. // Class : CLocationInfo
  8357. // Interface : ITLocationInfo
  8358. // Method : get_TollPrefixList
  8359. //
  8360. //
  8361. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8362. STDMETHODIMP
  8363. CLocationInfo::get_TollPrefixList(BSTR * ppTollList)
  8364. {
  8365. HRESULT hr = S_OK;
  8366. LOG((TL_TRACE, "get_TollPrefixList - enter" ));
  8367. Lock();
  8368. if ( TAPIIsBadWritePtr( ppTollList, sizeof(BSTR) ) )
  8369. {
  8370. LOG((TL_ERROR, "get_TollPrefixList - Bad Pointer" ));
  8371. hr = E_POINTER;
  8372. }
  8373. else // Ok Pointer
  8374. {
  8375. *ppTollList = SysAllocString( m_szTollPrefixList );
  8376. if ( ( NULL == *ppTollList ) && ( NULL != m_szTollPrefixList ) )
  8377. {
  8378. LOG((TL_TRACE, "SysAllocString Failed" ));
  8379. hr = E_OUTOFMEMORY;
  8380. }
  8381. }
  8382. Unlock();
  8383. LOG((TL_TRACE, hr, "get_TollPrefixList - exit" ));
  8384. return hr;
  8385. }
  8386. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8387. // Class : CLocationInfo
  8388. // Interface : ITLocationInfo
  8389. // Method : get_CancelCallWaitingCode
  8390. //
  8391. //
  8392. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8393. STDMETHODIMP
  8394. CLocationInfo::get_CancelCallWaitingCode(BSTR * ppCode)
  8395. {
  8396. HRESULT hr = S_OK;
  8397. LOG((TL_TRACE, "get_CancelCallWaitingCode - enter" ));
  8398. Lock();
  8399. if ( TAPIIsBadWritePtr( ppCode, sizeof(BSTR) ) )
  8400. {
  8401. LOG((TL_ERROR, "get_CancelCallWaitingCode - Bad Pointer" ));
  8402. hr = E_POINTER;
  8403. }
  8404. else // Ok Pointer
  8405. {
  8406. *ppCode = SysAllocString( m_szCancelCallWaitingCode );
  8407. if ( ( NULL == *ppCode ) && ( NULL != m_szCancelCallWaitingCode ) )
  8408. {
  8409. LOG((TL_TRACE, "SysAllocString Failed" ));
  8410. hr = E_OUTOFMEMORY;
  8411. }
  8412. }
  8413. Unlock();
  8414. LOG((TL_TRACE, hr, "get_CancelCallWaitingCode - exit" ));
  8415. return hr;
  8416. }
  8417. BOOL
  8418. CAddress::GetMediaMode( long lMediaType, DWORD * pdwMediaMode )
  8419. {
  8420. DWORD dwRet = (DWORD)lMediaType;
  8421. DWORD dwHold;
  8422. if (dwRet & AUDIOMEDIAMODES)
  8423. {
  8424. dwHold = m_dwMediaModesSupported & AUDIOMEDIAMODES;
  8425. if ( dwHold == AUDIOMEDIAMODES )
  8426. {
  8427. dwHold = LINEMEDIAMODE_AUTOMATEDVOICE;
  8428. }
  8429. dwRet &= ~AUDIOMEDIAMODES;
  8430. dwRet |= dwHold;
  8431. }
  8432. *pdwMediaMode = dwRet;
  8433. if ( (dwRet == 0) ||
  8434. ((dwRet & m_dwMediaModesSupported) != dwRet) )
  8435. {
  8436. return FALSE;
  8437. }
  8438. return TRUE;
  8439. }
  8440. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8441. //
  8442. // get_LineID
  8443. //
  8444. // returns the tapi 2 device ID for this line
  8445. //
  8446. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8447. STDMETHODIMP
  8448. CAddress::get_LineID(
  8449. long * plLineID
  8450. )
  8451. {
  8452. LOG((TL_TRACE, "get_LineID - enter"));
  8453. if ( TAPIIsBadWritePtr( plLineID, sizeof(long) ) )
  8454. {
  8455. LOG((TL_ERROR, "get_LineID - bad pointer"));
  8456. return E_POINTER;
  8457. }
  8458. Lock();
  8459. *plLineID = m_dwDeviceID;
  8460. Unlock();
  8461. LOG((TL_TRACE, "get_LineID - exit"));
  8462. return S_OK;
  8463. }
  8464. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8465. //
  8466. // get_AddressID
  8467. //
  8468. // returns the tapi 2 address ID of this address
  8469. //
  8470. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8471. STDMETHODIMP
  8472. CAddress::get_AddressID(
  8473. long * plAddressID
  8474. )
  8475. {
  8476. LOG((TL_TRACE, "get_AddressID - enter"));
  8477. if ( TAPIIsBadWritePtr( plAddressID, sizeof(long) ) )
  8478. {
  8479. LOG((TL_ERROR, "get_AddressID - bad pointer"));
  8480. return E_POINTER;
  8481. }
  8482. Lock();
  8483. *plAddressID = m_dwAddressID;
  8484. Unlock();
  8485. LOG((TL_TRACE, "get_AddressID - exit"));
  8486. return S_OK;
  8487. }
  8488. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8489. //
  8490. // CAddress::UpdateAddressCaps
  8491. //
  8492. // must be called in lock
  8493. //
  8494. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8495. HRESULT
  8496. CAddress::UpdateAddressCaps()
  8497. {
  8498. HRESULT hr = S_OK;
  8499. if ( NULL == m_pAddressCaps )
  8500. {
  8501. LPLINEADDRESSCAPS pTemp;
  8502. CTAPI * pCTapi;
  8503. pCTapi = GetTapi();
  8504. if( NULL == pCTapi )
  8505. {
  8506. LOG((TL_ERROR, "dynamic cast operation failed"));
  8507. hr = E_POINTER;
  8508. }
  8509. else
  8510. {
  8511. hr = pCTapi->GetBuffer( BUFFERTYPE_ADDRCAP,
  8512. (UINT_PTR)this,
  8513. (LPVOID*)&m_pAddressCaps
  8514. );
  8515. }
  8516. if ( !SUCCEEDED(hr) )
  8517. {
  8518. return hr;
  8519. }
  8520. pTemp = m_pAddressCaps;
  8521. hr = LineGetAddressCaps(
  8522. m_hLineApp,
  8523. m_dwDeviceID,
  8524. m_dwAddressID,
  8525. m_dwAPIVersion,
  8526. &m_pAddressCaps
  8527. );
  8528. if ( !SUCCEEDED(hr) )
  8529. {
  8530. return hr;
  8531. }
  8532. if ( m_pAddressCaps != pTemp )
  8533. {
  8534. pCTapi->SetBuffer( BUFFERTYPE_ADDRCAP, (UINT_PTR)this, (LPVOID)m_pAddressCaps );
  8535. }
  8536. }
  8537. return S_OK;
  8538. }
  8539. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8540. //
  8541. // CAddress::UpdateLineDevCaps
  8542. //
  8543. // must be called in lock
  8544. //
  8545. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8546. HRESULT
  8547. CAddress::UpdateLineDevCaps()
  8548. {
  8549. HRESULT hr = S_OK;
  8550. if ( NULL == m_pDevCaps )
  8551. {
  8552. LPLINEDEVCAPS pTemp;
  8553. CTAPI * pCTapi;
  8554. pCTapi = GetTapi();
  8555. if( NULL == pCTapi )
  8556. {
  8557. LOG((TL_ERROR, "dynamic cast operation failed"));
  8558. hr = E_POINTER;
  8559. }
  8560. else
  8561. {
  8562. hr = pCTapi->GetBuffer( BUFFERTYPE_LINEDEVCAP,
  8563. (UINT_PTR)this,
  8564. (LPVOID*)&m_pDevCaps
  8565. );
  8566. }
  8567. if ( !SUCCEEDED(hr) )
  8568. {
  8569. return hr;
  8570. }
  8571. pTemp = m_pDevCaps;
  8572. hr = LineGetDevCaps(
  8573. m_hLineApp,
  8574. m_dwDeviceID,
  8575. m_dwAPIVersion,
  8576. &m_pDevCaps
  8577. );
  8578. if ( !SUCCEEDED(hr) )
  8579. {
  8580. return hr;
  8581. }
  8582. if ( m_pDevCaps != pTemp )
  8583. {
  8584. pCTapi->SetBuffer( BUFFERTYPE_LINEDEVCAP, (UINT_PTR)this, (LPVOID)m_pDevCaps );
  8585. }
  8586. }
  8587. return S_OK;
  8588. }
  8589. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8590. //
  8591. // SetAddrCapBuffer
  8592. //
  8593. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8594. void
  8595. CAddress::SetAddrCapBuffer( LPVOID pBuf )
  8596. {
  8597. Lock();
  8598. m_pAddressCaps = (LPLINEADDRESSCAPS)pBuf;
  8599. Unlock();
  8600. }
  8601. void
  8602. CAddress::SetLineDevCapBuffer( LPVOID pBuf )
  8603. {
  8604. Lock();
  8605. m_pDevCaps = (LPLINEDEVCAPS)pBuf;
  8606. Unlock();
  8607. }
  8608. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8609. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8610. HRESULT
  8611. CAddress::CreateMSPCall(
  8612. MSP_HANDLE hCall,
  8613. DWORD dwReserved,
  8614. long lMediaType,
  8615. IUnknown * pOuterUnk,
  8616. IUnknown ** ppStreamControl
  8617. )
  8618. {
  8619. HRESULT hr = E_FAIL;
  8620. if ( NULL != m_pMSPAggAddress )
  8621. {
  8622. ITMSPAddress * pMSPAddress = GetMSPAddress();
  8623. __try
  8624. {
  8625. hr = pMSPAddress->CreateMSPCall(
  8626. hCall,
  8627. dwReserved,
  8628. lMediaType,
  8629. pOuterUnk,
  8630. ppStreamControl
  8631. );
  8632. }
  8633. __except(EXCEPTION_EXECUTE_HANDLER)
  8634. {
  8635. //
  8636. // catch any exceptions thrown by the msp, to protect ouirselves from
  8637. // misbehaving msps
  8638. //
  8639. LOG((TL_ERROR,
  8640. "CreateMSPCall - MSPAddress::CreateMSPCall threw an exception"));
  8641. hr = E_OUTOFMEMORY;
  8642. }
  8643. pMSPAddress->Release();
  8644. }
  8645. return hr;
  8646. }
  8647. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8648. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8649. HRESULT
  8650. CAddress::ShutdownMSPCall( IUnknown * pStreamControl )
  8651. {
  8652. HRESULT hr = S_FALSE;
  8653. if ( NULL != m_pMSPAggAddress )
  8654. {
  8655. ITMSPAddress * pMSPAddress = GetMSPAddress();
  8656. hr = pMSPAddress->ShutdownMSPCall( pStreamControl );
  8657. pMSPAddress->Release();
  8658. }
  8659. return hr;
  8660. }
  8661. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8662. //++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  8663. HRESULT
  8664. CAddress::ReceiveTSPData(
  8665. IUnknown * pMSPCall,
  8666. LPBYTE pBuffer,
  8667. DWORD dwSize
  8668. )
  8669. {
  8670. HRESULT hr = E_FAIL;
  8671. Lock();
  8672. if ( NULL != m_pMSPAggAddress )
  8673. {
  8674. ITMSPAddress * pMSPAddress = GetMSPAddress();
  8675. hr = pMSPAddress->ReceiveTSPData(
  8676. pMSPCall,
  8677. pBuffer,
  8678. dwSize
  8679. );
  8680. pMSPAddress->Release();
  8681. }
  8682. Unlock();
  8683. return hr;
  8684. }
  8685. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  8686. //
  8687. // LINE_SENDMSPMESSAGE handler
  8688. //
  8689. // give the opaque blob to the msp
  8690. //
  8691. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  8692. HRESULT HandleSendMSPDataMessage( PASYNCEVENTMSG pParams )
  8693. {
  8694. CAddress * pAddress;
  8695. BOOL bSuccess;
  8696. HRESULT hr = S_OK;
  8697. IUnknown * pMSPCall = NULL;;
  8698. LOG((TL_TRACE, "HandleSendMSPDataMessage - enter"));
  8699. //
  8700. // find the correct line
  8701. //
  8702. bSuccess = FindAddressObject(
  8703. (HLINE)(pParams->hDevice),
  8704. &pAddress
  8705. );
  8706. if (bSuccess)
  8707. {
  8708. CCall * pCall = NULL;
  8709. if ( NULL != (HCALL)(pParams->Param1) )
  8710. {
  8711. bSuccess = FindCallObject(
  8712. (HCALL)(pParams->Param1),
  8713. &pCall
  8714. );
  8715. if ( !bSuccess )
  8716. {
  8717. LOG((TL_ERROR, "HandleSendMSPDataMessage - couldn't find call %X",pParams->Param1));
  8718. //FindAddressObject addrefs the address objct
  8719. pAddress->Release();
  8720. return E_FAIL;
  8721. }
  8722. pMSPCall = pCall->GetMSPCall();
  8723. }
  8724. //
  8725. // the blob is at the end of the fixed
  8726. // structure
  8727. //
  8728. //
  8729. // get the size of the blob
  8730. //
  8731. DWORD dwSize = pParams->Param2;
  8732. BYTE *pBuffer = NULL;
  8733. //
  8734. // if the buffer's not empty, get a pointer to it
  8735. //
  8736. if (0 < dwSize)
  8737. {
  8738. pBuffer = (LPBYTE)(pParams + 1);
  8739. }
  8740. //
  8741. // call the msp
  8742. //
  8743. pAddress->ReceiveTSPData(
  8744. pMSPCall,
  8745. pBuffer,
  8746. dwSize
  8747. );
  8748. if ( pCall )
  8749. {
  8750. pCall->Release();
  8751. if ( pMSPCall )
  8752. {
  8753. pMSPCall->Release();
  8754. }
  8755. }
  8756. hr = S_OK;
  8757. //FindAddressObject addrefs the address objct
  8758. pAddress->Release();
  8759. }
  8760. else
  8761. {
  8762. LOG((TL_ERROR, "HandleSendMSPDataMessage - failed to find address Object %lx",
  8763. pParams->hDevice));
  8764. hr = E_FAIL;
  8765. }
  8766. LOG((TL_TRACE, "HandleSendMSPDataMessage - exit. hr = %lx", hr));
  8767. return hr;
  8768. }
  8769. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  8770. //
  8771. // HandleSendTSPData
  8772. //
  8773. // sends an opaque buffer from the msp to the tsp
  8774. //
  8775. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  8776. HRESULT
  8777. CAddress::HandleSendTSPData( MSP_EVENT_INFO * pEvent )
  8778. {
  8779. LOG((TL_INFO, "HandleSendTSPData - enter pEvent %p", pEvent));
  8780. //
  8781. // see if the event is bad
  8782. //
  8783. if (IsBadReadPtr(pEvent, sizeof(MSP_EVENT_INFO) ) )
  8784. {
  8785. LOG((TL_ERROR, "HandleSendTSPData - bad event structure"));
  8786. return E_POINTER;
  8787. }
  8788. HRESULT hr = S_OK;
  8789. HCALL hCall = NULL;
  8790. CCall * pCall = NULL;
  8791. AddressLineStruct * pAddressLine = NULL;
  8792. //
  8793. // if we were given msp call handle, find the corresponding call
  8794. //
  8795. if ( NULL != pEvent->hCall)
  8796. {
  8797. gpHandleHashTable->Lock();
  8798. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  8799. if ( SUCCEEDED(hr) )
  8800. {
  8801. //
  8802. // found the call, addreff the call and and release the table
  8803. //
  8804. LOG((TL_INFO, "HandleSendTSPData - Matched handle %X to Call object %p",
  8805. pEvent->hCall, pCall ));
  8806. pCall->AddRef();
  8807. gpHandleHashTable->Unlock();
  8808. //
  8809. // get the handle for this call
  8810. //
  8811. hCall = pCall->GetHCall();
  8812. //
  8813. // get call's address line, if any
  8814. //
  8815. pAddressLine = pCall->GetAddRefMyAddressLine();
  8816. LOG((TL_INFO, "HandleSendTSPData - address line[%p] hCall[%lx]",
  8817. pAddressLine, hCall ));
  8818. }
  8819. else
  8820. {
  8821. //
  8822. // there is no corresponding call in the hash table. the call no
  8823. // longer exists, or msp passed a bugus handle
  8824. //
  8825. gpHandleHashTable->Unlock();
  8826. LOG((TL_ERROR,
  8827. "HandleSendTSPData - Couldn't match handle %X to Call object. hr = %lx",
  8828. pEvent->hCall, hr));
  8829. return hr;
  8830. }
  8831. }
  8832. //
  8833. // by this point we either had msp call handle that is null or successfully
  8834. // found a matching call object and asked it for its address line
  8835. //
  8836. if (NULL == pAddressLine)
  8837. {
  8838. //
  8839. // if we don't have address line, send a message to the first address line on
  8840. // this address (?)
  8841. //
  8842. if ( m_AddressLinesPtrList.size() > 0 )
  8843. {
  8844. PtrList::iterator iter;
  8845. iter = m_AddressLinesPtrList.begin();
  8846. //
  8847. // send to the tsp
  8848. //
  8849. hr = LineReceiveMSPData(
  8850. ((AddressLineStruct *)(*iter))->t3Line.hLine,
  8851. hCall,
  8852. pEvent->MSP_TSP_DATA.pBuffer,
  8853. pEvent->MSP_TSP_DATA.dwBufferSize
  8854. );
  8855. }
  8856. else
  8857. {
  8858. LOG((TL_ERROR,
  8859. "HandleSendTSPData - no address lines on the address. E_UNEXPECTED"));
  8860. hr = E_UNEXPECTED;
  8861. }
  8862. }
  8863. else
  8864. {
  8865. //
  8866. // if we have an address line, send a message to the corresponding line
  8867. //
  8868. hr = LineReceiveMSPData(
  8869. pAddressLine->t3Line.hLine,
  8870. hCall,
  8871. pEvent->MSP_TSP_DATA.pBuffer,
  8872. pEvent->MSP_TSP_DATA.dwBufferSize
  8873. );
  8874. //
  8875. // no longer need our address line, release it. the line will be closed
  8876. // if needed
  8877. //
  8878. pCall->ReleaseAddressLine(pAddressLine);
  8879. pAddressLine = NULL;
  8880. }
  8881. //
  8882. // if we have a call, release it
  8883. //
  8884. if (NULL != pCall)
  8885. {
  8886. pCall->Release();
  8887. pCall = NULL;
  8888. }
  8889. LOG((TL_INFO, "HandleSendTSPData - finish hr = %lx", hr));
  8890. return hr;
  8891. }
  8892. //
  8893. // handle dev specific message
  8894. //
  8895. void HandleLineDevSpecificMessage( PASYNCEVENTMSG pParams )
  8896. {
  8897. LOG((TL_INFO, "HandleLineDevSpecificMessage - enter"));
  8898. //
  8899. // note:
  8900. //
  8901. // unfortunately, the message that we receive does not give us any
  8902. // indication as to whether hDevice contains a call handle or am address
  8903. // handle.
  8904. //
  8905. // to determine the kind of handle, we have to search both tables. if we
  8906. // find the corresponding address object, this is a line handle, if we find
  8907. // a corresponding call, this is a call handle. this brute-force search is
  8908. // fairly expensive.
  8909. //
  8910. // one possible optimization would be to track the number of calls and
  8911. // addresses and search the smaller table first. however, this would
  8912. // require tracking overhead (even in the cases where the optimization is
  8913. // not needed), and for this to work well, we would also need to estimate
  8914. // the likelyhood of each type of message. which makes this kind of
  8915. // optimization even more expensive.
  8916. //
  8917. // so we will simply check both tables to determine whether we have a call
  8918. // or an address.
  8919. //
  8920. CCall *pCall = NULL;
  8921. CAddress *pAddress = NULL;
  8922. //
  8923. // find the corresponding address
  8924. //
  8925. if ( !FindAddressObject( (HLINE)(pParams->hDevice), &pAddress ) )
  8926. {
  8927. LOG((TL_WARN,
  8928. "HandleLineDevSpecificMessage - FindAddressObject failed to find matching address. searching for call"));
  8929. pAddress = NULL;
  8930. //
  8931. // no address, try to find matching call
  8932. //
  8933. if ( !FindCallObject( (HCALL)(pParams->hDevice), &pCall) )
  8934. {
  8935. LOG((TL_ERROR,
  8936. "HandleLineDevSpecificMessage - FindAddressObject failed to find matching call. "));
  8937. return;
  8938. }
  8939. }
  8940. //
  8941. // if we got a call, get the corresponding address object
  8942. //
  8943. if (NULL != pCall)
  8944. {
  8945. ITAddress *pITAddress = NULL;
  8946. HRESULT hr = pCall->get_Address(&pITAddress);
  8947. if (FAILED(hr))
  8948. {
  8949. LOG((TL_ERROR,
  8950. "HandleLineDevSpecificMessage - call does not have an address. hr = %lx", hr));
  8951. pCall->Release();
  8952. pCall = NULL;
  8953. return;
  8954. }
  8955. try
  8956. {
  8957. pAddress = dynamic_cast<CAddress*>(pITAddress);
  8958. }
  8959. catch(...)
  8960. {
  8961. LOG((TL_ERROR,
  8962. "HandleLineDevSpecificMessage - exception using address. address pointer bad"));
  8963. }
  8964. //
  8965. // if the address is bad, return
  8966. //
  8967. if (NULL == pAddress)
  8968. {
  8969. LOG((TL_ERROR,
  8970. "HandleLineDevSpecificMessage - no address"));
  8971. pCall->Release();
  8972. pCall = NULL;
  8973. //
  8974. // yes, queryinterface returned us an addreffed pITAddress... but
  8975. // since it does not seem valid anyway, do not bother releasing.
  8976. //
  8977. _ASSERTE(FALSE);
  8978. return;
  8979. }
  8980. } // call is not null
  8981. //
  8982. // by this time we must have an address and maybe a call. we actually
  8983. // ensured this in the logic above. this assert is to make this condition
  8984. // explicit.
  8985. //
  8986. _ASSERTE( NULL != pAddress );
  8987. //
  8988. // fire event
  8989. //
  8990. CAddressDevSpecificEvent::FireEvent(
  8991. pAddress,
  8992. pCall,
  8993. pParams->Param1,
  8994. pParams->Param2,
  8995. pParams->Param3
  8996. );
  8997. //
  8998. // undo FindXObject's addreffs
  8999. //
  9000. if (NULL != pAddress)
  9001. {
  9002. pAddress->Release();
  9003. pAddress = NULL;
  9004. }
  9005. if (NULL != pCall)
  9006. {
  9007. pCall->Release();
  9008. pCall = NULL;
  9009. }
  9010. LOG((TL_INFO, "HandleLineDevSpecificMessage - exit. "));
  9011. }
  9012. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9013. //
  9014. // HandleMSPAddressEvent
  9015. //
  9016. // fires an addressevent to the application based on
  9017. // an event from the MSP
  9018. //
  9019. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9020. HRESULT
  9021. CAddress::HandleMSPAddressEvent( MSP_EVENT_INFO * pEvent )
  9022. {
  9023. switch ( pEvent->MSP_ADDRESS_EVENT_INFO.Type )
  9024. {
  9025. case ADDRESS_TERMINAL_AVAILABLE:
  9026. CAddressEvent::FireEvent(
  9027. this,
  9028. AE_NEWTERMINAL,
  9029. pEvent->MSP_ADDRESS_EVENT_INFO.pTerminal
  9030. );
  9031. break;
  9032. case ADDRESS_TERMINAL_UNAVAILABLE:
  9033. CAddressEvent::FireEvent(
  9034. this,
  9035. AE_REMOVETERMINAL,
  9036. pEvent->MSP_ADDRESS_EVENT_INFO.pTerminal
  9037. );
  9038. break;
  9039. default:
  9040. LOG((TL_ERROR, "HandleMSPAddressEvent - bad event"));
  9041. break;
  9042. }
  9043. return S_OK;
  9044. }
  9045. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9046. //
  9047. // HandleMSPCallEvent
  9048. //
  9049. // fires a callmediaevent to the application based on an
  9050. // event from the msp
  9051. //
  9052. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9053. HRESULT
  9054. CAddress::HandleMSPCallEvent( MSP_EVENT_INFO * pEvent )
  9055. {
  9056. CCall * pCall;
  9057. ITCallInfo * pCallInfo;
  9058. CALL_MEDIA_EVENT Event;
  9059. CALL_MEDIA_EVENT_CAUSE Cause;
  9060. ITTerminal * pTerminal = NULL;
  9061. ITStream * pStream = NULL;
  9062. HRESULT hrEvent = 0;
  9063. HRESULT hr = S_OK;
  9064. gpHandleHashTable->Lock();
  9065. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9066. if ( SUCCEEDED(hr) )
  9067. {
  9068. LOG((TL_INFO, "HandleMSPCallEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9069. pCall->AddRef();
  9070. gpHandleHashTable->Unlock();
  9071. hr = pCall->QueryInterface(
  9072. IID_ITCallInfo,
  9073. (void**) &pCallInfo
  9074. );
  9075. }
  9076. else // ( !SUCCEEDED(hr) )
  9077. {
  9078. LOG((TL_ERROR, "HandleMSPCallEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9079. gpHandleHashTable->Unlock();
  9080. return hr;
  9081. }
  9082. //
  9083. // set up the info for the event
  9084. // pStream applies to all of the currently-defined events
  9085. //
  9086. pStream = pEvent->MSP_CALL_EVENT_INFO.pStream;
  9087. //
  9088. // cause is the same, although the MSPI and API use different enumerated types.
  9089. // Note: this will have to expand to a switch if the order of enums ever gets
  9090. // out of sync!
  9091. //
  9092. Cause = (CALL_MEDIA_EVENT_CAUSE) pEvent->MSP_CALL_EVENT_INFO.Cause;
  9093. //
  9094. // Rest depend on the type...
  9095. //
  9096. switch ( pEvent->MSP_CALL_EVENT_INFO.Type )
  9097. {
  9098. case CALL_NEW_STREAM:
  9099. Event = CME_NEW_STREAM;
  9100. break;
  9101. case CALL_STREAM_FAIL:
  9102. Event = CME_STREAM_FAIL;
  9103. hrEvent = pEvent->MSP_CALL_EVENT_INFO.hrError;
  9104. break;
  9105. case CALL_TERMINAL_FAIL:
  9106. Event = CME_TERMINAL_FAIL;
  9107. pTerminal = pEvent->MSP_CALL_EVENT_INFO.pTerminal;
  9108. hrEvent = pEvent->MSP_CALL_EVENT_INFO.hrError;
  9109. break;
  9110. case CALL_STREAM_NOT_USED:
  9111. Event = CME_STREAM_NOT_USED;
  9112. break;
  9113. case CALL_STREAM_ACTIVE:
  9114. Event = CME_STREAM_ACTIVE;
  9115. break;
  9116. case CALL_STREAM_INACTIVE:
  9117. Event = CME_STREAM_INACTIVE;
  9118. break;
  9119. default:
  9120. LOG((TL_ERROR, "HandleMSPCallEvent - bad event"));
  9121. return E_INVALIDARG;
  9122. }
  9123. //
  9124. // fire the event
  9125. //
  9126. CCallMediaEvent::FireEvent(
  9127. pCallInfo,
  9128. Event,
  9129. Cause,
  9130. dynamic_cast<CTAPI *>(m_pTAPI),
  9131. pTerminal,
  9132. pStream,
  9133. hrEvent
  9134. );
  9135. //
  9136. // addref'd above
  9137. //
  9138. pCallInfo->Release();
  9139. pCall->Release();
  9140. return S_OK;
  9141. }
  9142. HRESULT
  9143. CAddress::HandleMSPTTSTerminalEvent( MSP_EVENT_INFO * pEvent )
  9144. {
  9145. if (NULL == pEvent)
  9146. {
  9147. LOG((TL_ERROR, "HandleMSPTTSTerminalEvent - pEvent is NULL"));
  9148. return E_POINTER;
  9149. }
  9150. ITAddress *pAddress = dynamic_cast<ITAddress *>(this);
  9151. if (pAddress == NULL)
  9152. {
  9153. LOG((TL_ERROR, "HandleMSPTTSTerminalEvent - can't cast the address %p to ITAddress", this));
  9154. return E_UNEXPECTED;
  9155. }
  9156. CCall * pCall = NULL;
  9157. ITCallInfo * pCallInfo = NULL;
  9158. gpHandleHashTable->Lock();
  9159. HRESULT hr = E_FAIL;
  9160. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9161. if ( SUCCEEDED(hr) )
  9162. {
  9163. LOG((TL_INFO, "HandleMSPTTSTerminalEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9164. pCall->AddRef();
  9165. gpHandleHashTable->Unlock();
  9166. hr = pCall->QueryInterface(
  9167. IID_ITCallInfo,
  9168. (void**) &pCallInfo
  9169. );
  9170. }
  9171. else // ( !SUCCEEDED(hr) )
  9172. {
  9173. LOG((TL_ERROR, "HandleMSPTTSTerminalEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9174. gpHandleHashTable->Unlock();
  9175. return hr;
  9176. }
  9177. //
  9178. // get tapi object and fire the event
  9179. //
  9180. hr = CTTSTerminalEvent::FireEvent(GetTapi(),
  9181. pCallInfo,
  9182. pEvent->MSP_TTS_TERMINAL_EVENT_INFO.pTTSTerminal,
  9183. pEvent->MSP_TTS_TERMINAL_EVENT_INFO.hrErrorCode);
  9184. if (FAILED(hr))
  9185. {
  9186. LOG((TL_ERROR, "HandleMSPTTSTerminalEvent - CFileTerminalEvent::FireEvent failed. hr = %lx", hr));
  9187. }
  9188. //
  9189. // we addref'd these above, so release now
  9190. //
  9191. pCallInfo->Release();
  9192. pCall->Release();
  9193. return hr;
  9194. }
  9195. HRESULT
  9196. CAddress::HandleMSPASRTerminalEvent( MSP_EVENT_INFO * pEvent )
  9197. {
  9198. if (NULL == pEvent)
  9199. {
  9200. LOG((TL_ERROR, "HandleMSPASRTerminalEvent - pEvent is NULL"));
  9201. return E_POINTER;
  9202. }
  9203. ITAddress *pAddress = dynamic_cast<ITAddress *>(this);
  9204. if (pAddress == NULL)
  9205. {
  9206. LOG((TL_ERROR, "HandleMSPASRTerminalEvent - can't cast the address %p to ITAddress", this));
  9207. return E_UNEXPECTED;
  9208. }
  9209. CCall * pCall = NULL;
  9210. ITCallInfo * pCallInfo = NULL;
  9211. gpHandleHashTable->Lock();
  9212. HRESULT hr = E_FAIL;
  9213. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9214. if ( SUCCEEDED(hr) )
  9215. {
  9216. LOG((TL_INFO, "HandleMSPASRTerminalEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9217. pCall->AddRef();
  9218. gpHandleHashTable->Unlock();
  9219. hr = pCall->QueryInterface(
  9220. IID_ITCallInfo,
  9221. (void**) &pCallInfo
  9222. );
  9223. }
  9224. else // ( !SUCCEEDED(hr) )
  9225. {
  9226. LOG((TL_ERROR, "HandleMSPASRTerminalEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9227. gpHandleHashTable->Unlock();
  9228. return hr;
  9229. }
  9230. //
  9231. // get tapi object and fire the event
  9232. //
  9233. hr = CASRTerminalEvent::FireEvent(GetTapi(),
  9234. pCallInfo,
  9235. pEvent->MSP_ASR_TERMINAL_EVENT_INFO.pASRTerminal,
  9236. pEvent->MSP_ASR_TERMINAL_EVENT_INFO.hrErrorCode);
  9237. if (FAILED(hr))
  9238. {
  9239. LOG((TL_ERROR, "HandleMSPASRTerminalEvent - CFileTerminalEvent::FireEvent failed. hr = %lx", hr));
  9240. }
  9241. //
  9242. // we addref'd these above, so release now
  9243. //
  9244. pCallInfo->Release();
  9245. pCall->Release();
  9246. return hr;
  9247. }
  9248. HRESULT
  9249. CAddress::HandleMSPToneTerminalEvent( MSP_EVENT_INFO * pEvent )
  9250. {
  9251. if (NULL == pEvent)
  9252. {
  9253. LOG((TL_ERROR, "HandleMSPToneTerminalEvent - pEvent is NULL"));
  9254. return E_POINTER;
  9255. }
  9256. ITAddress *pAddress = dynamic_cast<ITAddress *>(this);
  9257. if (pAddress == NULL)
  9258. {
  9259. LOG((TL_ERROR, "HandleMSPToneTerminalEvent - can't cast the address %p to ITAddress", this));
  9260. return E_UNEXPECTED;
  9261. }
  9262. CCall * pCall = NULL;
  9263. ITCallInfo * pCallInfo = NULL;
  9264. gpHandleHashTable->Lock();
  9265. HRESULT hr = E_FAIL;
  9266. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9267. if ( SUCCEEDED(hr) )
  9268. {
  9269. LOG((TL_INFO, "HandleMSPToneTerminalEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9270. pCall->AddRef();
  9271. gpHandleHashTable->Unlock();
  9272. hr = pCall->QueryInterface(
  9273. IID_ITCallInfo,
  9274. (void**) &pCallInfo
  9275. );
  9276. }
  9277. else // ( !SUCCEEDED(hr) )
  9278. {
  9279. LOG((TL_ERROR, "HandleMSPToneTerminalEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9280. gpHandleHashTable->Unlock();
  9281. return hr;
  9282. }
  9283. //
  9284. // get tapi object and fire the event
  9285. //
  9286. hr = CToneTerminalEvent::FireEvent(GetTapi(),
  9287. pCallInfo,
  9288. pEvent->MSP_TONE_TERMINAL_EVENT_INFO.pToneTerminal,
  9289. pEvent->MSP_TONE_TERMINAL_EVENT_INFO.hrErrorCode);
  9290. if (FAILED(hr))
  9291. {
  9292. LOG((TL_ERROR, "HandleMSPToneTerminalEvent - CFileTerminalEvent::FireEvent failed. hr = %lx", hr));
  9293. }
  9294. //
  9295. // we addref'd these above, so release now
  9296. //
  9297. pCallInfo->Release();
  9298. pCall->Release();
  9299. return hr;
  9300. }
  9301. HRESULT
  9302. CAddress::HandleMSPFileTerminalEvent( MSP_EVENT_INFO * pEvent )
  9303. {
  9304. if (NULL == pEvent)
  9305. {
  9306. LOG((TL_ERROR, "HandleMSPFileTerminalEvent - pEvent is NULL"));
  9307. return E_POINTER;
  9308. }
  9309. ITAddress *pAddress = dynamic_cast<ITAddress *>(this);
  9310. if (pAddress == NULL)
  9311. {
  9312. LOG((TL_ERROR, "HandleMSPFileTerminalEvent - can't cast the address %p to ITAddress", this));
  9313. return E_UNEXPECTED;
  9314. }
  9315. CCall * pCall = NULL;
  9316. ITCallInfo * pCallInfo = NULL;
  9317. gpHandleHashTable->Lock();
  9318. HRESULT hr = E_FAIL;
  9319. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9320. if ( SUCCEEDED(hr) )
  9321. {
  9322. LOG((TL_INFO, "HandleMSPFileTerminalEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9323. pCall->AddRef();
  9324. gpHandleHashTable->Unlock();
  9325. hr = pCall->QueryInterface(
  9326. IID_ITCallInfo,
  9327. (void**) &pCallInfo
  9328. );
  9329. }
  9330. else // ( !SUCCEEDED(hr) )
  9331. {
  9332. LOG((TL_ERROR, "HandleMSPFileTerminalEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9333. gpHandleHashTable->Unlock();
  9334. return hr;
  9335. }
  9336. //
  9337. // get tapi object and fire the event
  9338. //
  9339. hr = CFileTerminalEvent::FireEvent(this,
  9340. GetTapi(),
  9341. pCallInfo,
  9342. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.TerminalMediaState,
  9343. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.ftecEventCause,
  9344. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pParentFileTerminal,
  9345. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pFileTrack,
  9346. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.hrErrorCode);
  9347. if (FAILED(hr))
  9348. {
  9349. LOG((TL_ERROR, "HandleMSPFileTerminalEvent - CFileTerminalEvent::FireEvent failed. hr = %lx", hr));
  9350. }
  9351. //
  9352. // we addref'd these above, so release now
  9353. //
  9354. pCallInfo->Release();
  9355. pCall->Release();
  9356. return hr;
  9357. }
  9358. HRESULT
  9359. CAddress::HandleMSPPrivateEvent( MSP_EVENT_INFO * pEvent )
  9360. {
  9361. ITAddress *pAddress = dynamic_cast<ITAddress *>(this);
  9362. if (pAddress == NULL)
  9363. {
  9364. LOG((TL_ERROR, "HandleMSPPrivateEvent - can't cast the address %p to ITAddress", this));
  9365. return E_UNEXPECTED;
  9366. }
  9367. CCall * pCall = NULL;
  9368. ITCallInfo * pCallInfo = NULL;
  9369. HRESULT hr;
  9370. gpHandleHashTable->Lock();
  9371. hr = gpHandleHashTable->Find( (ULONG_PTR)pEvent->hCall, (ULONG_PTR *)&pCall);
  9372. if ( SUCCEEDED(hr) )
  9373. {
  9374. LOG((TL_INFO, "HandleMSPPrivateEvent - Matched handle %X to Call object %p", pEvent->hCall, pCall ));
  9375. pCall->AddRef();
  9376. gpHandleHashTable->Unlock();
  9377. hr = pCall->QueryInterface(
  9378. IID_ITCallInfo,
  9379. (void**) &pCallInfo
  9380. );
  9381. }
  9382. else // ( !SUCCEEDED(hr) )
  9383. {
  9384. LOG((TL_ERROR, "HandleMSPPrivateEvent - Couldn't match handle %X to Call object ", pEvent->hCall));
  9385. gpHandleHashTable->Unlock();
  9386. return hr;
  9387. }
  9388. CPrivateEvent::FireEvent(
  9389. GetTapi(),
  9390. pCallInfo,
  9391. pAddress,
  9392. NULL,
  9393. pEvent->MSP_PRIVATE_EVENT_INFO.pEvent,
  9394. pEvent->MSP_PRIVATE_EVENT_INFO.lEventCode
  9395. );
  9396. //
  9397. // addref'd above
  9398. //
  9399. pCallInfo->Release();
  9400. pCall->Release();
  9401. return hr;
  9402. }
  9403. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9404. //
  9405. // ReleaseEvent
  9406. //
  9407. // releases any reference counts in the event from
  9408. // the msp.
  9409. //
  9410. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9411. HRESULT
  9412. CAddress::ReleaseEvent( MSP_EVENT_INFO * pEvent )
  9413. {
  9414. LOG((TL_TRACE, "ReleaseEvent - enter"));
  9415. switch ( pEvent->Event )
  9416. {
  9417. case ME_ADDRESS_EVENT:
  9418. if (NULL != pEvent->MSP_ADDRESS_EVENT_INFO.pTerminal)
  9419. {
  9420. (pEvent->MSP_ADDRESS_EVENT_INFO.pTerminal)->Release();
  9421. }
  9422. break;
  9423. case ME_CALL_EVENT:
  9424. if (NULL != pEvent->MSP_CALL_EVENT_INFO.pTerminal)
  9425. {
  9426. (pEvent->MSP_CALL_EVENT_INFO.pTerminal)->Release();
  9427. }
  9428. if (NULL != pEvent->MSP_CALL_EVENT_INFO.pStream)
  9429. {
  9430. (pEvent->MSP_CALL_EVENT_INFO.pStream)->Release();
  9431. }
  9432. break;
  9433. case ME_TSP_DATA:
  9434. break;
  9435. case ME_PRIVATE_EVENT:
  9436. if ( NULL != pEvent->MSP_PRIVATE_EVENT_INFO.pEvent )
  9437. {
  9438. (pEvent->MSP_PRIVATE_EVENT_INFO.pEvent)->Release();
  9439. }
  9440. break;
  9441. case ME_FILE_TERMINAL_EVENT:
  9442. if( NULL != pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pParentFileTerminal)
  9443. {
  9444. (pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pParentFileTerminal)->Release();
  9445. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pParentFileTerminal = NULL;
  9446. }
  9447. if( NULL != pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pFileTrack )
  9448. {
  9449. (pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pFileTrack)->Release();
  9450. pEvent->MSP_FILE_TERMINAL_EVENT_INFO.pFileTrack = NULL;
  9451. }
  9452. break;
  9453. case ME_ASR_TERMINAL_EVENT:
  9454. if( NULL != pEvent->MSP_ASR_TERMINAL_EVENT_INFO.pASRTerminal)
  9455. {
  9456. (pEvent->MSP_ASR_TERMINAL_EVENT_INFO.pASRTerminal)->Release();
  9457. }
  9458. break;
  9459. case ME_TTS_TERMINAL_EVENT:
  9460. if( NULL != pEvent->MSP_TTS_TERMINAL_EVENT_INFO.pTTSTerminal)
  9461. {
  9462. (pEvent->MSP_TTS_TERMINAL_EVENT_INFO.pTTSTerminal)->Release();
  9463. }
  9464. break;
  9465. case ME_TONE_TERMINAL_EVENT:
  9466. if( NULL != pEvent->MSP_TONE_TERMINAL_EVENT_INFO.pToneTerminal)
  9467. {
  9468. (pEvent->MSP_TONE_TERMINAL_EVENT_INFO.pToneTerminal)->Release();
  9469. }
  9470. break;
  9471. default:
  9472. break;
  9473. }
  9474. LOG((TL_TRACE, "ReleaseEvent - finished"));
  9475. return S_OK;
  9476. }
  9477. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9478. // HRESULT
  9479. // CAddress::MSPEvent()
  9480. //
  9481. // gets an event buffer from the MSP
  9482. // and calls the relevant event handler
  9483. //
  9484. // this is _only_ called from the asynceventsthread. this is
  9485. // necessary for synchronization of events.
  9486. //
  9487. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9488. HRESULT
  9489. CAddress::MSPEvent()
  9490. {
  9491. DWORD dwSize;
  9492. MSP_EVENT_INFO * pEvent = NULL;
  9493. HRESULT hr;
  9494. //
  9495. // Allocate an MSP event buffer -- best guess as to an appropriate
  9496. // size. We need to free it before returning.
  9497. //
  9498. dwSize = sizeof (MSP_EVENT_INFO) + 500;
  9499. pEvent = (MSP_EVENT_INFO *)ClientAlloc( dwSize );
  9500. if ( NULL == pEvent )
  9501. {
  9502. LOG((TL_ERROR, "Alloc failed in MSP event"));
  9503. return E_OUTOFMEMORY;
  9504. }
  9505. pEvent->dwSize = dwSize;
  9506. //
  9507. // Get pointer to msp address object. We need to release it before returning.
  9508. // This must be done while the address object is locked.
  9509. //
  9510. Lock();
  9511. ITMSPAddress * pMSPAddress = GetMSPAddress();
  9512. Unlock();
  9513. while (TRUE)
  9514. {
  9515. //
  9516. // Get an event from the event queue. Abort and clean up if
  9517. // allocation fails.
  9518. //
  9519. do
  9520. {
  9521. hr = pMSPAddress->GetEvent(
  9522. &dwSize,
  9523. (LPBYTE)pEvent
  9524. );
  9525. if ( hr == TAPI_E_NOTENOUGHMEMORY)
  9526. {
  9527. ClientFree( pEvent );
  9528. pEvent = (MSP_EVENT_INFO *)ClientAlloc( dwSize );
  9529. if ( NULL == pEvent )
  9530. {
  9531. LOG((TL_ERROR, "Alloc failed in MSP event"));
  9532. pMSPAddress->Release();
  9533. return E_OUTOFMEMORY;
  9534. }
  9535. pEvent->dwSize = dwSize;
  9536. }
  9537. } while ( hr == TAPI_E_NOTENOUGHMEMORY );
  9538. //
  9539. // If there is nothing left in the MSP event queue, then stop the
  9540. // outer while loop.
  9541. //
  9542. if ( !SUCCEEDED(hr) )
  9543. {
  9544. break;
  9545. }
  9546. //
  9547. // Call the relevant handler, and do not hold the address lock during
  9548. // the call.
  9549. //
  9550. switch ( pEvent->Event )
  9551. {
  9552. case ME_ADDRESS_EVENT:
  9553. HandleMSPAddressEvent( pEvent );
  9554. break;
  9555. case ME_CALL_EVENT:
  9556. HandleMSPCallEvent( pEvent );
  9557. break;
  9558. case ME_TSP_DATA:
  9559. HandleSendTSPData( pEvent );
  9560. break;
  9561. case ME_PRIVATE_EVENT:
  9562. HandleMSPPrivateEvent( pEvent );
  9563. break;
  9564. case ME_FILE_TERMINAL_EVENT:
  9565. HandleMSPFileTerminalEvent( pEvent );
  9566. break;
  9567. case ME_ASR_TERMINAL_EVENT:
  9568. HandleMSPASRTerminalEvent( pEvent );
  9569. break;
  9570. case ME_TTS_TERMINAL_EVENT:
  9571. HandleMSPTTSTerminalEvent( pEvent );
  9572. break;
  9573. case ME_TONE_TERMINAL_EVENT:
  9574. HandleMSPToneTerminalEvent( pEvent );
  9575. break;
  9576. default:
  9577. break;
  9578. }
  9579. //
  9580. // release any refcounts in the
  9581. // event
  9582. //
  9583. ReleaseEvent( pEvent );
  9584. }
  9585. //
  9586. // We get here when there is nothing more to retrieve from the
  9587. // MSP event queue.
  9588. //
  9589. pMSPAddress->Release();
  9590. ClientFree( pEvent );
  9591. return hr;
  9592. }
  9593. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9594. //
  9595. //
  9596. //
  9597. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9598. ITMSPAddress * CAddress::GetMSPAddress()
  9599. {
  9600. ITMSPAddress * pMSPAddress = NULL;
  9601. if ( NULL != m_pMSPAggAddress )
  9602. {
  9603. m_pMSPAggAddress->QueryInterface(IID_ITMSPAddress, (void**)&pMSPAddress);
  9604. }
  9605. return pMSPAddress;
  9606. }
  9607. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9608. //
  9609. //
  9610. //
  9611. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9612. STDMETHODIMP
  9613. CAddress::GetID(
  9614. BSTR pDeviceClass,
  9615. DWORD * pdwSize,
  9616. BYTE ** ppDeviceID
  9617. )
  9618. {
  9619. HRESULT hr;
  9620. PtrList::iterator iter;
  9621. LPVARSTRING pVarString = NULL;
  9622. if ( IsBadStringPtrW( pDeviceClass, -1 ) )
  9623. {
  9624. LOG((TL_ERROR, "GetID - bad string"));
  9625. return E_POINTER;
  9626. }
  9627. if ( TAPIIsBadWritePtr( pdwSize, sizeof(DWORD)))
  9628. {
  9629. LOG((TL_ERROR, "GetID - bad size"));
  9630. return E_POINTER;
  9631. }
  9632. if ( TAPIIsBadWritePtr( ppDeviceID, sizeof(BYTE *) ) )
  9633. {
  9634. LOG((TL_ERROR, "GetID - bad pointer"));
  9635. return E_POINTER;
  9636. }
  9637. Lock();
  9638. if ( m_AddressLinesPtrList.size() > 0 )
  9639. {
  9640. iter = m_AddressLinesPtrList.begin();
  9641. }
  9642. else
  9643. {
  9644. Unlock();
  9645. return E_FAIL;
  9646. }
  9647. hr = LineGetID(
  9648. ((AddressLineStruct *)*iter)->t3Line.hLine,
  9649. m_dwAddressID,
  9650. NULL,
  9651. LINECALLSELECT_ADDRESS,
  9652. &pVarString,
  9653. pDeviceClass
  9654. );
  9655. Unlock();
  9656. if ( SUCCEEDED(hr) )
  9657. {
  9658. *ppDeviceID = (BYTE *)CoTaskMemAlloc( pVarString->dwUsedSize );
  9659. if ( *ppDeviceID != NULL )
  9660. {
  9661. CopyMemory(
  9662. *ppDeviceID,
  9663. ((LPBYTE)pVarString)+pVarString->dwStringOffset,
  9664. pVarString->dwStringSize
  9665. );
  9666. *pdwSize = pVarString->dwStringSize;
  9667. }
  9668. else
  9669. {
  9670. hr = E_OUTOFMEMORY;
  9671. }
  9672. }
  9673. if ( NULL != pVarString )
  9674. {
  9675. ClientFree( pVarString );
  9676. }
  9677. return hr;
  9678. }
  9679. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9680. // Class : CAddress
  9681. // Interface : ITLegacyAddressMediaControl
  9682. // Method : GetDevConfig
  9683. //
  9684. // Get Device Config
  9685. //
  9686. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9687. STDMETHODIMP
  9688. CAddress::GetDevConfig(
  9689. BSTR pDeviceClass,
  9690. DWORD * pdwSize,
  9691. BYTE ** ppDeviceConfig
  9692. )
  9693. {
  9694. HRESULT hr;
  9695. LPVARSTRING pVarString = NULL;
  9696. LOG((TL_TRACE, "GetDevConfig - enter"));
  9697. if ( IsBadStringPtrW( pDeviceClass, -1 ) )
  9698. {
  9699. LOG((TL_ERROR, "GetDevConfig - bad DeviceClass string"));
  9700. return E_POINTER;
  9701. }
  9702. if ( TAPIIsBadWritePtr( pdwSize, sizeof(DWORD)))
  9703. {
  9704. LOG((TL_ERROR, "GetDevConfig - bad size"));
  9705. return E_POINTER;
  9706. }
  9707. if ( TAPIIsBadWritePtr( ppDeviceConfig, sizeof(BYTE*) ) )
  9708. {
  9709. LOG((TL_ERROR, "GetDevConfig - bad buffer pointer"));
  9710. return E_POINTER;
  9711. }
  9712. Lock();
  9713. hr = LineGetDevConfig(m_dwDeviceID,
  9714. &pVarString,
  9715. pDeviceClass
  9716. );
  9717. Unlock();
  9718. if ( SUCCEEDED(hr) )
  9719. {
  9720. *ppDeviceConfig = (BYTE *)CoTaskMemAlloc( pVarString->dwUsedSize );
  9721. if(*ppDeviceConfig != NULL)
  9722. {
  9723. CopyMemory(
  9724. *ppDeviceConfig,
  9725. ((LPBYTE)pVarString)+pVarString->dwStringOffset,
  9726. pVarString->dwStringSize
  9727. );
  9728. *pdwSize = pVarString->dwStringSize;
  9729. }
  9730. else
  9731. {
  9732. hr = E_OUTOFMEMORY;
  9733. }
  9734. }
  9735. if ( NULL != pVarString )
  9736. {
  9737. ClientFree( pVarString );
  9738. }
  9739. LOG((TL_TRACE, hr, "GetDevConfig - exit" ));
  9740. return hr;
  9741. }
  9742. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9743. // Class : CAddress
  9744. // Interface : ITLegacyAddressMediaControl
  9745. // Method : SetDevConfig
  9746. //
  9747. // Set Device Config
  9748. //
  9749. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9750. STDMETHODIMP
  9751. CAddress::SetDevConfig(
  9752. BSTR pDeviceClass,
  9753. DWORD dwSize,
  9754. BYTE * pDeviceConfig
  9755. )
  9756. {
  9757. HRESULT hr = S_OK;
  9758. LOG((TL_TRACE, "SetDevConfig - enter"));
  9759. if ( IsBadStringPtrW( pDeviceClass, -1) )
  9760. {
  9761. LOG((TL_ERROR, "SetDevConfig - bad string pointer"));
  9762. return E_POINTER;
  9763. }
  9764. if (dwSize == 0)
  9765. {
  9766. LOG((TL_ERROR, "SetDevConfig - dwSize = 0"));
  9767. return E_INVALIDARG;
  9768. }
  9769. if (IsBadReadPtr( pDeviceConfig, dwSize) )
  9770. {
  9771. LOG((TL_ERROR, "SetDevConfig - bad pointer"));
  9772. return E_POINTER;
  9773. }
  9774. Lock();
  9775. hr = lineSetDevConfigW(m_dwDeviceID,
  9776. pDeviceConfig,
  9777. dwSize,
  9778. pDeviceClass
  9779. );
  9780. Unlock();
  9781. LOG((TL_TRACE, hr, "SetDevConfig - exit"));
  9782. return hr;
  9783. }
  9784. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9785. // Class : CAddress
  9786. // Interface : ITLegacyAddressMediaControl2
  9787. // Method : ConfigDialog
  9788. //
  9789. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9790. STDMETHODIMP
  9791. CAddress::ConfigDialog(
  9792. HWND hwndOwner,
  9793. BSTR pDeviceClass
  9794. )
  9795. {
  9796. HRESULT hr = S_OK;
  9797. LOG((TL_TRACE, "ConfigDialog - enter"));
  9798. if ( (pDeviceClass != NULL) && IsBadStringPtrW( pDeviceClass, -1) )
  9799. {
  9800. LOG((TL_ERROR, "ConfigDialog - bad string pointer"));
  9801. return E_POINTER;
  9802. }
  9803. Lock();
  9804. hr = LineConfigDialogW(
  9805. m_dwDeviceID,
  9806. hwndOwner,
  9807. pDeviceClass
  9808. );
  9809. Unlock();
  9810. LOG((TL_TRACE, hr, "ConfigDialog - exit"));
  9811. return hr;
  9812. }
  9813. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9814. // Class : CAddress
  9815. // Interface : ITLegacyAddressMediaControl2
  9816. // Method : ConfigDialogEdit
  9817. //
  9818. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  9819. STDMETHODIMP
  9820. CAddress::ConfigDialogEdit(
  9821. HWND hwndOwner,
  9822. BSTR pDeviceClass,
  9823. DWORD dwSizeIn,
  9824. BYTE * pDeviceConfigIn,
  9825. DWORD * pdwSizeOut,
  9826. BYTE ** ppDeviceConfigOut
  9827. )
  9828. {
  9829. HRESULT hr = S_OK;
  9830. LPVARSTRING pVarString = NULL;
  9831. LOG((TL_TRACE, "ConfigDialogEdit - enter"));
  9832. if ( (pDeviceClass != NULL) && IsBadStringPtrW( pDeviceClass, -1) )
  9833. {
  9834. LOG((TL_ERROR, "ConfigDialogEdit - bad string pointer"));
  9835. return E_POINTER;
  9836. }
  9837. if (dwSizeIn == 0)
  9838. {
  9839. LOG((TL_ERROR, "ConfigDialogEdit - dwSize = 0"));
  9840. return E_INVALIDARG;
  9841. }
  9842. if (IsBadReadPtr( pDeviceConfigIn, dwSizeIn) )
  9843. {
  9844. LOG((TL_ERROR, "ConfigDialogEdit - bad pointer"));
  9845. return E_POINTER;
  9846. }
  9847. if ( TAPIIsBadWritePtr( pdwSizeOut, sizeof(DWORD)))
  9848. {
  9849. LOG((TL_ERROR, "ConfigDialogEdit - bad size"));
  9850. return E_POINTER;
  9851. }
  9852. if ( TAPIIsBadWritePtr( ppDeviceConfigOut, sizeof(BYTE*) ) )
  9853. {
  9854. LOG((TL_ERROR, "ConfigDialogEdit - bad buffer pointer"));
  9855. return E_POINTER;
  9856. }
  9857. Lock();
  9858. hr = LineConfigDialogEditW(
  9859. m_dwDeviceID,
  9860. hwndOwner,
  9861. pDeviceClass,
  9862. pDeviceConfigIn,
  9863. dwSizeIn,
  9864. &pVarString
  9865. );
  9866. Unlock();
  9867. if ( SUCCEEDED(hr) )
  9868. {
  9869. *ppDeviceConfigOut = (BYTE *)CoTaskMemAlloc( pVarString->dwUsedSize );
  9870. if(*ppDeviceConfigOut != NULL)
  9871. {
  9872. CopyMemory(
  9873. *ppDeviceConfigOut,
  9874. ((LPBYTE)pVarString)+pVarString->dwStringOffset,
  9875. pVarString->dwStringSize
  9876. );
  9877. *pdwSizeOut = pVarString->dwStringSize;
  9878. }
  9879. else
  9880. {
  9881. hr = E_OUTOFMEMORY;
  9882. }
  9883. }
  9884. if ( NULL != pVarString )
  9885. {
  9886. ClientFree( pVarString );
  9887. }
  9888. LOG((TL_TRACE, hr, "ConfigDialogEdit - exit"));
  9889. return hr;
  9890. }
  9891. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9892. //
  9893. //
  9894. //
  9895. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9896. HRESULT HandleLineCloseMessage( PASYNCEVENTMSG pParams )
  9897. {
  9898. CAddress * pAddress;
  9899. AddressLineStruct * pAddressLine;
  9900. CTAPI * pTapi;
  9901. ITAddress * pITAddress;
  9902. LOG((TL_TRACE, "HandleLineCloseMessage - enter"));
  9903. if ( !FindAddressObject(
  9904. (HLINE)(pParams->hDevice),
  9905. &pAddress
  9906. ) )
  9907. {
  9908. LOG((TL_TRACE, "HandleLineCloseMessage - FindAddressObject failed. exiting... "));
  9909. return S_OK;
  9910. }
  9911. pTapi = pAddress->GetTapi();
  9912. pAddress->QueryInterface(
  9913. IID_ITAddress,
  9914. (void **)&pITAddress
  9915. );
  9916. //
  9917. // get the lock so that the address line does not go away meanwhile
  9918. //
  9919. pAddress->Lock();
  9920. //
  9921. // covert the 32bit address line handle (contained in pParams->OpenContext)
  9922. // into a pointer value
  9923. //
  9924. pAddressLine = (AddressLineStruct *)GetHandleTableEntry(pParams->OpenContext);
  9925. //
  9926. // is this a good line anyway?
  9927. //
  9928. BOOL bValidLine = pAddress->IsValidAddressLine(pAddressLine);
  9929. long lCallBackInstance = 0;
  9930. //
  9931. // if seems to be a good line, attempt to get callback instance from it.
  9932. //
  9933. if (bValidLine)
  9934. {
  9935. try
  9936. {
  9937. lCallBackInstance = pAddressLine->lCallbackInstance;
  9938. }
  9939. catch(...)
  9940. {
  9941. LOG((TL_ERROR,
  9942. "HandleLineCloseMessage - exception getting callback instance from line struc"));
  9943. _ASSERTE(FALSE);
  9944. bValidLine = FALSE;
  9945. }
  9946. }
  9947. pAddress->Unlock();
  9948. //
  9949. // if good and known line, attempt to fire an event
  9950. //
  9951. if ( bValidLine && pTapi->FindRegistration( (PVOID)pAddressLine ) )
  9952. {
  9953. LOG((TL_TRACE, "HandleLineCloseMessage - found registration, firing event"));
  9954. CTapiObjectEvent::FireEvent(
  9955. pTapi,
  9956. TE_ADDRESSCLOSE,
  9957. pITAddress,
  9958. lCallBackInstance,
  9959. NULL
  9960. );
  9961. }
  9962. else
  9963. {
  9964. LOG((TL_TRACE,
  9965. "HandleLineCloseMessage AddressLine %p not found. calling maybeclosealine",
  9966. pAddressLine ));
  9967. pAddress->MaybeCloseALine(&pAddressLine);
  9968. }
  9969. pITAddress->Release();
  9970. //FindAddressObject addrefs the address objct
  9971. pAddress->Release();
  9972. LOG((TL_TRACE, "HandleLineCloseMessage - exit"));
  9973. return S_OK;
  9974. }
  9975. /////////////////////////////////////////////////////////////////////////////
  9976. // IDispatch implementation
  9977. //
  9978. typedef IDispatchImpl<ITAddress2Vtbl<CAddress>, &IID_ITAddress2, &LIBID_TAPI3Lib> AddressType;
  9979. typedef IDispatchImpl<ITAddressCapabilitiesVtbl<CAddress>, &IID_ITAddressCapabilities, &LIBID_TAPI3Lib> AddressCapabilitiesType;
  9980. typedef IDispatchImpl<ITMediaSupportVtbl<CAddress>, &IID_ITMediaSupport, &LIBID_TAPI3Lib> MediaSupportType;
  9981. typedef IDispatchImpl<ITAddressTranslationVtbl<CAddress>, &IID_ITAddressTranslation, &LIBID_TAPI3Lib> AddressTranslationType;
  9982. typedef IDispatchImpl<ITLegacyAddressMediaControl2Vtbl<CAddress>, &IID_ITLegacyAddressMediaControl2, &LIBID_TAPI3Lib> LegacyAddressMediaControlType;
  9983. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9984. //
  9985. // CAddress::GetIDsOfNames
  9986. //
  9987. // Overide if IDispatch method
  9988. //
  9989. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  9990. STDMETHODIMP CAddress::GetIDsOfNames(REFIID riid,
  9991. LPOLESTR* rgszNames,
  9992. UINT cNames,
  9993. LCID lcid,
  9994. DISPID* rgdispid
  9995. )
  9996. {
  9997. HRESULT hr = DISP_E_UNKNOWNNAME;
  9998. // See if the requsted method belongs to the default interface
  9999. hr = AddressType::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10000. if (SUCCEEDED(hr))
  10001. {
  10002. LOG((TL_INFO, "GetIDsOfNames - found %S on ITAddress", *rgszNames));
  10003. rgdispid[0] |= IDISPADDRESS;
  10004. return hr;
  10005. }
  10006. // If not, then try the Address Capabilities interface
  10007. hr = AddressCapabilitiesType::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10008. if (SUCCEEDED(hr))
  10009. {
  10010. LOG((TL_INFO, "GetIDsOfNames - found %S on ITAddressCapabilities", *rgszNames));
  10011. rgdispid[0] |= IDISPADDRESSCAPABILITIES;
  10012. return hr;
  10013. }
  10014. // If not, then try the Media Support interface
  10015. hr = MediaSupportType::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10016. if (SUCCEEDED(hr))
  10017. {
  10018. LOG((TL_INFO, "GetIDsOfNames - found %S on ITMediaSupport", *rgszNames));
  10019. rgdispid[0] |= IDISPMEDIASUPPORT;
  10020. return hr;
  10021. }
  10022. // If not, then try the Address Translation interface
  10023. hr = AddressTranslationType::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10024. if (SUCCEEDED(hr))
  10025. {
  10026. LOG((TL_INFO, "GetIDsOfNames - found %S on ITAddressTranslation", *rgszNames));
  10027. rgdispid[0] |= IDISPADDRESSTRANSLATION;
  10028. return hr;
  10029. }
  10030. // If not, then try the Legacy Address Media Control interface
  10031. hr = LegacyAddressMediaControlType::GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10032. if (SUCCEEDED(hr))
  10033. {
  10034. LOG((TL_INFO, "GetIDsOfNames - found %S on ITLegacyAddressMediaControl", *rgszNames));
  10035. rgdispid[0] |= IDISPLEGACYADDRESSMEDIACONTROL;
  10036. return hr;
  10037. }
  10038. // If not, then try the aggregated MSP Address object
  10039. if (m_pMSPAggAddress != NULL)
  10040. {
  10041. IDispatch *pIDispatchMSPAggAddress;
  10042. m_pMSPAggAddress->QueryInterface(IID_IDispatch, (void**)&pIDispatchMSPAggAddress);
  10043. hr = pIDispatchMSPAggAddress->GetIDsOfNames(riid, rgszNames, cNames, lcid, rgdispid);
  10044. if (SUCCEEDED(hr))
  10045. {
  10046. pIDispatchMSPAggAddress->Release();
  10047. LOG((TL_INFO, "GetIDsOfNames - found %S on our aggregated MSP Address", *rgszNames));
  10048. rgdispid[0] |= IDISPAGGREGATEDMSPADDRESSOBJ;
  10049. return hr;
  10050. }
  10051. pIDispatchMSPAggAddress->Release();
  10052. }
  10053. LOG((TL_INFO, "GetIDsOfNames - Didn't find %S on our iterfaces", *rgszNames));
  10054. return hr;
  10055. }
  10056. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  10057. //
  10058. // CAddress::Invoke
  10059. //
  10060. // Overide if IDispatch method
  10061. //
  10062. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  10063. STDMETHODIMP CAddress::Invoke(DISPID dispidMember,
  10064. REFIID riid,
  10065. LCID lcid,
  10066. WORD wFlags,
  10067. DISPPARAMS* pdispparams,
  10068. VARIANT* pvarResult,
  10069. EXCEPINFO* pexcepinfo,
  10070. UINT* puArgErr
  10071. )
  10072. {
  10073. HRESULT hr = DISP_E_MEMBERNOTFOUND;
  10074. DWORD dwInterface = (dispidMember & INTERFACEMASK);
  10075. LOG((TL_TRACE, "Invoke - dispidMember %X", dispidMember));
  10076. // Call invoke for the required interface
  10077. switch (dwInterface)
  10078. {
  10079. case IDISPADDRESS:
  10080. {
  10081. hr = AddressType::Invoke(dispidMember,
  10082. riid,
  10083. lcid,
  10084. wFlags,
  10085. pdispparams,
  10086. pvarResult,
  10087. pexcepinfo,
  10088. puArgErr
  10089. );
  10090. break;
  10091. }
  10092. case IDISPADDRESSCAPABILITIES:
  10093. {
  10094. hr = AddressCapabilitiesType::Invoke(dispidMember,
  10095. riid,
  10096. lcid,
  10097. wFlags,
  10098. pdispparams,
  10099. pvarResult,
  10100. pexcepinfo,
  10101. puArgErr
  10102. );
  10103. break;
  10104. }
  10105. case IDISPMEDIASUPPORT:
  10106. {
  10107. hr = MediaSupportType::Invoke(dispidMember,
  10108. riid,
  10109. lcid,
  10110. wFlags,
  10111. pdispparams,
  10112. pvarResult,
  10113. pexcepinfo,
  10114. puArgErr
  10115. );
  10116. break;
  10117. }
  10118. case IDISPADDRESSTRANSLATION:
  10119. {
  10120. hr = AddressTranslationType::Invoke(dispidMember,
  10121. riid,
  10122. lcid,
  10123. wFlags,
  10124. pdispparams,
  10125. pvarResult,
  10126. pexcepinfo,
  10127. puArgErr
  10128. );
  10129. break;
  10130. }
  10131. case IDISPLEGACYADDRESSMEDIACONTROL:
  10132. {
  10133. hr = LegacyAddressMediaControlType::Invoke(dispidMember,
  10134. riid,
  10135. lcid,
  10136. wFlags,
  10137. pdispparams,
  10138. pvarResult,
  10139. pexcepinfo,
  10140. puArgErr
  10141. );
  10142. break;
  10143. }
  10144. case IDISPAGGREGATEDMSPADDRESSOBJ:
  10145. {
  10146. IDispatch *pIDispatchMSPAggAddress = NULL;
  10147. if (m_pMSPAggAddress != NULL)
  10148. {
  10149. m_pMSPAggAddress->QueryInterface(IID_IDispatch, (void**)&pIDispatchMSPAggAddress);
  10150. hr = pIDispatchMSPAggAddress->Invoke(dispidMember,
  10151. riid,
  10152. lcid,
  10153. wFlags,
  10154. pdispparams,
  10155. pvarResult,
  10156. pexcepinfo,
  10157. puArgErr
  10158. );
  10159. pIDispatchMSPAggAddress->Release();
  10160. }
  10161. break;
  10162. }
  10163. } // end switch (dwInterface)
  10164. LOG((TL_TRACE, hr, "Invoke - exit" ));
  10165. return hr;
  10166. }
  10167. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  10168. //
  10169. // CAddress::IsValidAddressLine
  10170. //
  10171. // returns TRUE if the address line passed in is in the address line list
  10172. //
  10173. // This method is not thread safe -- all calls must be protected, unless
  10174. // bAddRef argument is TRUE
  10175. //
  10176. //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++=
  10177. BOOL CAddress::IsValidAddressLine(AddressLineStruct *pAddressLine, BOOL bAddref)
  10178. {
  10179. if (IsBadReadPtr(pAddressLine, sizeof(AddressLineStruct) ) )
  10180. {
  10181. LOG((TL_WARN,
  10182. "IsValidAddressLine - unreadeable memory at [%p]",
  10183. pAddressLine));
  10184. return FALSE;
  10185. }
  10186. if (bAddref)
  10187. {
  10188. Lock();
  10189. }
  10190. PtrList::iterator it = m_AddressLinesPtrList.begin();
  10191. PtrList::iterator end = m_AddressLinesPtrList.end();
  10192. // iterate through address line list until we find the line
  10193. for ( ; it != end; it++ )
  10194. {
  10195. if (pAddressLine == (AddressLineStruct *)(*it))
  10196. {
  10197. if (bAddref)
  10198. {
  10199. //
  10200. // addref the line and release the lock
  10201. //
  10202. try
  10203. {
  10204. pAddressLine->AddRef();
  10205. }
  10206. catch(...)
  10207. {
  10208. //
  10209. // this is a bug. debug.
  10210. //
  10211. LOG((TL_INFO, "IsValidAddressLine -- exception while addreffing the line"));
  10212. _ASSERTE(FALSE);
  10213. Unlock();
  10214. return FALSE;
  10215. }
  10216. Unlock();
  10217. }
  10218. LOG((TL_INFO, "IsValidAddressLine returning TRUE"));
  10219. return TRUE;
  10220. }
  10221. }
  10222. if (bAddref)
  10223. {
  10224. Unlock();
  10225. }
  10226. LOG((TL_INFO, "IsValidAddressLine returning FALSE"));
  10227. return FALSE;
  10228. }
  10229. /////////////////////////////////////////////////////////////
  10230. //
  10231. // implementing method from CObjectSafeImpl
  10232. //
  10233. // check the aggregated objects to see if they support the interface requested.
  10234. // if they do, return the non-delegating IUnknown of the object that supports
  10235. // the interface.
  10236. //
  10237. HRESULT CAddress::QIOnAggregates(REFIID riid, IUnknown **ppNonDelegatingUnknown)
  10238. {
  10239. //
  10240. // argument check
  10241. //
  10242. if ( TAPIIsBadWritePtr(ppNonDelegatingUnknown, sizeof(IUnknown*)) )
  10243. {
  10244. return E_POINTER;
  10245. }
  10246. //
  10247. // if we fail, at least return consistent values
  10248. //
  10249. *ppNonDelegatingUnknown = NULL;
  10250. //
  10251. // see if m_pMSPAggAddress or private support the interface riid
  10252. //
  10253. HRESULT hr = E_FAIL;
  10254. Lock();
  10255. if (m_pMSPAggAddress)
  10256. {
  10257. //
  10258. // does m_pMSPAggAddress expose this interface?
  10259. //
  10260. IUnknown *pUnk = NULL;
  10261. hr = m_pMSPAggAddress->QueryInterface(riid, (void**)&pUnk);
  10262. if (SUCCEEDED(hr))
  10263. {
  10264. pUnk->Release();
  10265. pUnk = NULL;
  10266. //
  10267. // return the mspcall's non-delegating unknown
  10268. //
  10269. *ppNonDelegatingUnknown = m_pMSPAggAddress;
  10270. (*ppNonDelegatingUnknown)->AddRef();
  10271. }
  10272. }
  10273. if ( FAILED(hr) && m_pPrivate )
  10274. {
  10275. //
  10276. // bad luck with m_pMSPAggAddress? still have a chance with private
  10277. //
  10278. IUnknown *pUnk = NULL;
  10279. hr = m_pPrivate->QueryInterface(riid, (void**)&pUnk);
  10280. if (SUCCEEDED(hr))
  10281. {
  10282. pUnk->Release();
  10283. pUnk = NULL;
  10284. *ppNonDelegatingUnknown = m_pPrivate;
  10285. (*ppNonDelegatingUnknown)->AddRef();
  10286. }
  10287. }
  10288. Unlock();
  10289. return hr;
  10290. }
  10291. //
  10292. // Event filtering methods
  10293. //
  10294. /*++
  10295. SetEventFilterMask
  10296. Is called by TAPI object to set the event filter mask to each address
  10297. The caller method is CTAPI::SetEventFilterToAddresses()
  10298. dwEventFilterMask is the TAPI event filter mask
  10299. --*/
  10300. HRESULT CAddress::SetEventFilterMask(
  10301. DWORD dwEventFilterMask
  10302. )
  10303. {
  10304. LOG((TL_TRACE, "SetEventFilterMask - enter. dwEventFilterMask[%lx]", dwEventFilterMask ));
  10305. //
  10306. // Set the event mask for all TAPI events
  10307. //
  10308. // TE_ADDRESS
  10309. SetSubEventFlag(
  10310. TE_ADDRESS,
  10311. CEventMasks::EM_ALLSUBEVENTS,
  10312. (dwEventFilterMask & TE_ADDRESS) ? TRUE : FALSE);
  10313. // TE_CALLHUB
  10314. SetSubEventFlag(
  10315. TE_CALLHUB,
  10316. CEventMasks::EM_ALLSUBEVENTS,
  10317. (dwEventFilterMask & TE_CALLHUB) ? TRUE : FALSE);
  10318. // TE_CALLINFOCHANGE
  10319. SetSubEventFlag(
  10320. TE_CALLINFOCHANGE,
  10321. CEventMasks::EM_ALLSUBEVENTS,
  10322. (dwEventFilterMask & TE_CALLINFOCHANGE) ? TRUE : FALSE);
  10323. // TE_CALLMEDIA
  10324. SetSubEventFlag(
  10325. TE_CALLMEDIA,
  10326. CEventMasks::EM_ALLSUBEVENTS,
  10327. (dwEventFilterMask & TE_CALLMEDIA) ? TRUE : FALSE);
  10328. // TE_CALLNOTIFICATION
  10329. SetSubEventFlag(
  10330. TE_CALLNOTIFICATION,
  10331. CEventMasks::EM_ALLSUBEVENTS,
  10332. (dwEventFilterMask & TE_CALLNOTIFICATION) ? TRUE : FALSE);
  10333. // TE_CALLSTATE
  10334. SetSubEventFlag(
  10335. TE_CALLSTATE,
  10336. CEventMasks::EM_ALLSUBEVENTS,
  10337. (dwEventFilterMask & TE_CALLSTATE) ? TRUE : FALSE);
  10338. // TE_FILETERMINAL
  10339. SetSubEventFlag(
  10340. TE_FILETERMINAL,
  10341. CEventMasks::EM_ALLSUBEVENTS,
  10342. (dwEventFilterMask & TE_FILETERMINAL) ? TRUE : FALSE);
  10343. // TE_PRIVATE
  10344. SetSubEventFlag(
  10345. TE_PRIVATE,
  10346. CEventMasks::EM_ALLSUBEVENTS,
  10347. (dwEventFilterMask & TE_PRIVATE) ? TRUE : FALSE);
  10348. // TE_QOSEVENT
  10349. SetSubEventFlag(
  10350. TE_QOSEVENT,
  10351. CEventMasks::EM_ALLSUBEVENTS,
  10352. (dwEventFilterMask & TE_QOSEVENT) ? TRUE : FALSE);
  10353. // TE_ADDRESSDEVSPECIFIC
  10354. SetSubEventFlag(
  10355. TE_ADDRESSDEVSPECIFIC,
  10356. CEventMasks::EM_ALLSUBEVENTS,
  10357. (dwEventFilterMask & TE_ADDRESSDEVSPECIFIC) ? TRUE : FALSE);
  10358. // TE_PHONEDEVSPECIFIC
  10359. SetSubEventFlag(
  10360. TE_PHONEDEVSPECIFIC,
  10361. CEventMasks::EM_ALLSUBEVENTS,
  10362. (dwEventFilterMask & TE_PHONEDEVSPECIFIC) ? TRUE : FALSE);
  10363. LOG((TL_TRACE, "SetEventFilterMask exit S_OK"));
  10364. return S_OK;
  10365. }
  10366. /*++
  10367. SetSubEventFlag
  10368. Sets in m_EventFilterMasks array the bitflag for an subevent
  10369. Is called by CAddress::SetEventFilterMask
  10370. --*/
  10371. HRESULT CAddress::SetSubEventFlag(
  10372. TAPI_EVENT TapiEvent,
  10373. DWORD dwSubEvent,
  10374. BOOL bEnable
  10375. )
  10376. {
  10377. LOG((TL_TRACE,
  10378. "SetSubEventFlag - enter. event [%lx] subevent[%lx] enable?[%d]",
  10379. TapiEvent, dwSubEvent, bEnable ));
  10380. HRESULT hr = S_OK;
  10381. //
  10382. // Set the mask for the event
  10383. //
  10384. hr = m_EventMasks.SetSubEventFlag(
  10385. TapiEvent,
  10386. dwSubEvent,
  10387. bEnable);
  10388. if( SUCCEEDED(hr) )
  10389. {
  10390. hr = SetSubEventFlagToCalls(
  10391. TapiEvent,
  10392. dwSubEvent,
  10393. bEnable
  10394. );
  10395. }
  10396. LOG((TL_TRACE, "SetSubEventFlag exit 0x%08x", hr));
  10397. return hr;
  10398. }
  10399. /*++
  10400. GetSubEventFlag
  10401. It is calle by get_EventFilter() method
  10402. --*/
  10403. HRESULT CAddress::GetSubEventFlag(
  10404. TAPI_EVENT TapiEvent,
  10405. DWORD dwSubEvent,
  10406. BOOL* pEnable
  10407. )
  10408. {
  10409. LOG((TL_TRACE, "GetSubEventFlag enter" ));
  10410. HRESULT hr = E_FAIL;
  10411. //
  10412. // Get the subevent falg
  10413. //
  10414. hr = m_EventMasks.GetSubEventFlag(
  10415. TapiEvent,
  10416. dwSubEvent,
  10417. pEnable
  10418. );
  10419. LOG((TL_TRACE, "GetSubEventFlag exit 0x%08x", hr));
  10420. return hr;
  10421. }
  10422. /*++
  10423. SetSubEventFlagToCalls
  10424. Sets the flags to all calls
  10425. Is called by SetSubEventFlag() method
  10426. --*/
  10427. HRESULT CAddress::SetSubEventFlagToCalls(
  10428. TAPI_EVENT TapiEvent,
  10429. DWORD dwSubEvent,
  10430. BOOL bEnable
  10431. )
  10432. {
  10433. LOG((TL_TRACE, "SetSubEventFlagToCalls enter" ));
  10434. HRESULT hr = S_OK;
  10435. //
  10436. // Apply the sub event filter mask to all the calls on this address
  10437. //
  10438. for (int nCall = 0; nCall < m_CallArray.GetSize() ; nCall++ )
  10439. {
  10440. CCall * pCall = NULL;
  10441. pCall = dynamic_cast<CCall *>(m_CallArray[nCall]);
  10442. if ( NULL != pCall )
  10443. {
  10444. hr = pCall->SetSubEventFlag(
  10445. TapiEvent,
  10446. dwSubEvent,
  10447. bEnable
  10448. );
  10449. if( FAILED(hr) )
  10450. {
  10451. break;
  10452. }
  10453. }
  10454. }
  10455. LOG((TL_TRACE, "SetSubEventFlagToCalls exit 0x%08x", hr));
  10456. return hr;
  10457. }
  10458. /*++
  10459. GetSubEventsMask
  10460. Is called by SetSubEventFlag to get the sub events mask for
  10461. a specific TAPI_EVENT
  10462. Assumed is called into a Lock statement
  10463. --*/
  10464. DWORD CAddress::GetSubEventsMask(
  10465. IN TAPI_EVENT TapiEvent
  10466. )
  10467. {
  10468. LOG((TL_TRACE, "GetSubEventsMask - enter"));
  10469. DWORD dwSubEventFlag = m_EventMasks.GetSubEventMask( TapiEvent );
  10470. LOG((TL_TRACE, "GetSubEventsMask - exit %ld", dwSubEventFlag));
  10471. return dwSubEventFlag;
  10472. }
  10473. /*++
  10474. GetEventMasks
  10475. It is called by CCall::Initialize()
  10476. --*/
  10477. HRESULT CAddress::GetEventMasks(
  10478. OUT CEventMasks* pEventMasks
  10479. )
  10480. {
  10481. LOG((TL_TRACE, "GetEventMasks - enter"));
  10482. m_EventMasks.CopyEventMasks( pEventMasks );
  10483. LOG((TL_TRACE, "GetEventMasks - exit S_OK"));
  10484. return S_OK;
  10485. }
  10486. ////////////////////////////////////////////
  10487. //
  10488. // RegisterNotificationCookie
  10489. //
  10490. // adds the specified cookie to the list of cookies on this address
  10491. //
  10492. HRESULT CAddress::RegisterNotificationCookie(long lCookie)
  10493. {
  10494. HRESULT hr = S_OK;
  10495. LOG((TL_INFO,
  10496. "RegisterNotificationCookie - adding cookie %lx to m_NotificationCookies list",
  10497. lCookie ));
  10498. Lock();
  10499. try
  10500. {
  10501. m_NotificationCookies.push_back(lCookie);
  10502. }
  10503. catch(...)
  10504. {
  10505. LOG((TL_ERROR,
  10506. "RegisterNotificationCookie - failed to add a cookie to m_NotificationCookies list - alloc failure" ));
  10507. hr = E_OUTOFMEMORY;
  10508. }
  10509. Unlock();
  10510. return hr;
  10511. }
  10512. ////////////////////////////////////////////
  10513. //
  10514. // RemoveNotificationCookie
  10515. //
  10516. // removes the specified cookie from this address's cookies
  10517. //
  10518. HRESULT CAddress::RemoveNotificationCookie(long lCookie)
  10519. {
  10520. HRESULT hr = S_OK;
  10521. LOG((TL_INFO,
  10522. "RemoveNotificationCookie - removing cookie %lx from m_NotificationCookies list",
  10523. lCookie ));
  10524. Lock();
  10525. m_NotificationCookies.remove(lCookie);
  10526. Unlock();
  10527. return hr;
  10528. }
  10529. ////////////////////////////////////////////
  10530. //
  10531. // UnregisterAllCookies
  10532. //
  10533. // removes all cookies from this address's cookie list
  10534. //
  10535. // for each valid cookie, call RemoveCallNotification
  10536. //
  10537. void CAddress::UnregisterAllCookies()
  10538. {
  10539. LOG((TL_TRACE, "UnregisterAllCookies entering. this[%p]", this));
  10540. Lock();
  10541. LongList::iterator it = m_NotificationCookies.begin();
  10542. LongList::iterator end = m_NotificationCookies.end();
  10543. //
  10544. // if there are any cookies, remove the corresponding registeritems
  10545. // from the handlehashtable
  10546. //
  10547. // also, unregister call notification for these cookies, to make sure
  10548. // everything is cleaned up (and in particular that lineCloseMSPInstance
  10549. // gets called if needed so the tsp gets notified that the msp is going
  10550. // away
  10551. //
  10552. for ( ; it != end; it++ )
  10553. {
  10554. long lCookie = (long)(*it);
  10555. LOG((TL_INFO, "UnregisterAllCookies removing handle %lx", lCookie));
  10556. //
  10557. // get register item, so we can unregister call notifications with it.
  10558. //
  10559. REGISTERITEM *pRegisterItem = (REGISTERITEM*) GetHandleTableEntry(lCookie);
  10560. //
  10561. // remove the entry from the handle table
  10562. //
  10563. RemoveHandleFromHashTable(lCookie);
  10564. //
  10565. // if we did not get a good registeritem that corresponds to this cookie,
  10566. // go on to the next cookie
  10567. //
  10568. if ( (NULL == pRegisterItem) ||
  10569. IsBadReadPtr(pRegisterItem, sizeof(REGISTERITEM)) )
  10570. {
  10571. LOG((TL_INFO,
  10572. "UnregisterAllCookies - no corresponfing registeritem for cookie 0x%lx",
  10573. lCookie));
  10574. continue;
  10575. }
  10576. //
  10577. // if the register item is not RA_ADDRESS, ignore it and go on to
  10578. // the next cookie.
  10579. //
  10580. if (RA_ADDRESS != pRegisterItem->dwType)
  10581. {
  10582. LOG((TL_INFO,
  10583. "UnregisterAllCookies - cookie 0x%lx is of type 0x%lx, not RA_ADDRESS",
  10584. lCookie,
  10585. pRegisterItem->dwType));
  10586. continue;
  10587. }
  10588. //
  10589. // remove call notification for this cookie, since the app did not
  10590. // do this. best effort -- ignore the error code.
  10591. //
  10592. LOG((TL_INFO,
  10593. "UnregisterAllCookies - removing call notification for cookie 0x%lx",
  10594. lCookie));
  10595. RemoveCallNotification(pRegisterItem->pRegister);
  10596. }
  10597. //
  10598. // clear cookie list
  10599. //
  10600. m_NotificationCookies.clear();
  10601. Unlock();
  10602. LOG((TL_TRACE, "UnregisterAllCookies exiting"));
  10603. }
  10604. ///////////////////////////////////////////////////////////////////////////////
  10605. //
  10606. // CAddress::AddressOnTapiShutdown
  10607. //
  10608. // this function is called by the owner tapi object when it is going away.
  10609. //
  10610. // we then propagate tapi shutdown notification to each call, so the calls
  10611. // can do their clean up
  10612. //
  10613. void CAddress::AddressOnTapiShutdown()
  10614. {
  10615. LOG((TL_TRACE, "AddressOnTapiShutdown - enter"));
  10616. //
  10617. // protect access to m_CallArray
  10618. //
  10619. Lock();
  10620. //
  10621. // tell each call to clean up
  10622. //
  10623. int nCalls = m_CallArray.GetSize();
  10624. for (int i = 0; i < nCalls; i++)
  10625. {
  10626. ITCallInfo *pCallInfo = m_CallArray[i];
  10627. try
  10628. {
  10629. //
  10630. // get a pointer to the call object
  10631. //
  10632. CCall *pCallObject = dynamic_cast<CCall *>(pCallInfo);
  10633. if (NULL == pCallObject)
  10634. {
  10635. //
  10636. // the pointer is not pointing to a call object. this is odd
  10637. // and is worth debugging
  10638. //
  10639. LOG((TL_ERROR,
  10640. "AddressOnTapiShutdown - invalid call pointer[%p] in the call array",
  10641. pCallInfo));
  10642. _ASSERTE(FALSE);
  10643. }
  10644. else
  10645. {
  10646. //
  10647. // tell the call that it's time to go
  10648. //
  10649. pCallObject->CallOnTapiShutdown();
  10650. }
  10651. }
  10652. catch(...)
  10653. {
  10654. //
  10655. // not only is the pointer not pointing to a call object, but it is
  10656. // also pointing to memory that is not readable. how did this happen?
  10657. //
  10658. LOG((TL_ERROR,
  10659. "AddressOnTapiShutdown - unreadable call pointer[%p] in the call array",
  10660. pCallInfo));
  10661. _ASSERTE(FALSE);
  10662. }
  10663. }
  10664. //
  10665. // set the flag so no new calls are created following this
  10666. //
  10667. m_bTapiSignaledShutdown = TRUE;
  10668. //
  10669. // unregister the msp wait event in the thread pool. we don't want to get
  10670. // any callbacks after tapi has shut down
  10671. //
  10672. if ( NULL != m_hWaitEvent )
  10673. {
  10674. LOG((TL_TRACE, "AddressOnTapiShutdown - unregistering the MSPEventCallback callback"));
  10675. UnregisterWaitEx( m_hWaitEvent, INVALID_HANDLE_VALUE);
  10676. m_hWaitEvent = NULL;
  10677. }
  10678. Unlock();
  10679. LOG((TL_TRACE, "AddressOnTapiShutdown - finish"));
  10680. }