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.

758 lines
21 KiB

  1. /*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2. Copyright (c) 1996-2000 Microsoft Corporation
  3. Module Name :
  4. attack.c
  5. Abstract :
  6. This file contains the ndr correlation check for denial of attacks.
  7. Author :
  8. Ryszard K. Kott (ryszardk) Sep 1997
  9. Revision History :
  10. ---------------------------------------------------------------------*/
  11. #include "ndrp.h"
  12. #include "hndl.h"
  13. #include "ndrole.h"
  14. #include "attack.h"
  15. #include "interp.h"
  16. #include "interp2.h"
  17. #include "mulsyntx.h"
  18. #include "asyncu.h"
  19. extern "C" {
  20. extern const GUID CLSID_RpcHelper;
  21. }
  22. inline
  23. PFORMAT_STRING
  24. GetConformanceDescriptor(
  25. PFORMAT_STRING pFormat )
  26. {
  27. static uchar
  28. ConformanceDescIncrements[] =
  29. {
  30. 4, // Conformant array.
  31. 4, // Conformant varying array.
  32. 0, 0, // Fixed arrays - unused.
  33. 0, 0, // Varying arrays - unused.
  34. 4, // Complex array.
  35. 2, // Conformant char string.
  36. 2, // Conformant byte string.
  37. 4, // Conformant stringable struct.
  38. 2, // Conformant wide char string.
  39. 0, 0, 0, 0, // Non-conformant strings - unused.
  40. 0, // Encapsulated union - unused.
  41. 2, // Non-encapsulated union.
  42. 2, // Byte count pointer.
  43. 0, 0, // Xmit/Rep as - unused.
  44. 2 // Interface pointer.
  45. };
  46. ASSERT( FC_CARRAY <= *pFormat && *pFormat < FC_END);
  47. return pFormat + ConformanceDescIncrements[ *pFormat - FC_CARRAY ];
  48. }
  49. inline
  50. PFORMAT_STRING
  51. GetVarianceDescriptor(
  52. PFORMAT_STRING pFormat )
  53. {
  54. // The array gives offset according to the size of the new correlation descriptors.
  55. static uchar
  56. VarianceDescIncrements[] =
  57. { 8 + NDR_CORR_EXTENSION_SIZE, // Conformant varying array.
  58. 0, 0, // Fixed arrays - unsed.
  59. 8, 12, // Varying array.
  60. 8 + NDR_CORR_EXTENSION_SIZE, // Complex array.
  61. };
  62. ASSERT( FC_CVARRAY <= *pFormat && *pFormat <= FC_BOGUS_ARRAY );
  63. return pFormat + VarianceDescIncrements[ *pFormat - FC_CVARRAY ];
  64. }
  65. void
  66. NdrpCheckCorrelation(
  67. PMIDL_STUB_MESSAGE pStubMsg,
  68. LONG_PTR Value,
  69. PFORMAT_STRING pFormat,
  70. int CheckKind )
  71. /*
  72. Checks if a correlation check can be performed or needs postponing.
  73. For early correlations it performs the check.
  74. For late correlations it adds an entry in the late correlatrion data base.
  75. Parameters
  76. Value - conformant value related to the descriptor
  77. pFormat - array, string etc object with the descriptor
  78. pStubMsg->pCorrMemory - current memory context like struct
  79. */
  80. {
  81. // pStubMsg->pCorrelationInfo->pCorrFstr - descriptor for the current object
  82. // ASSERT( pStubMsg->pCorrInfo );
  83. // TBD performance: index through 2 dim table could be faster.
  84. if ( ( CheckKind & ~NDR_RESET_VALUE ) == NDR_CHECK_CONFORMANCE )
  85. pFormat = GetConformanceDescriptor( pFormat );
  86. else
  87. pFormat = GetVarianceDescriptor( pFormat );
  88. // See if we can actually check it out or whether we have to postpone it
  89. // till the correlation pass.
  90. NDR_FCDEF_CORRELATION * pConf = (NDR_FCDEF_CORRELATION *)pFormat;
  91. if ( pConf->CorrFlags.DontCheck )
  92. return;
  93. unsigned char * pMemory = pStubMsg->pCorrMemory;
  94. if ( pConf->CorrFlags.Early )
  95. {
  96. ULONG_PTR MaxCountSave = pStubMsg->MaxCount;
  97. ulong OffsetSave = pStubMsg->Offset;
  98. pStubMsg->Offset = 0;
  99. // this call overwrites pStubMsg->MaxCount
  100. NdrpValidateCorrelatedValue( pStubMsg, pMemory, pFormat, Value, CheckKind );
  101. pStubMsg->MaxCount = MaxCountSave;
  102. pStubMsg->Offset = OffsetSave;
  103. }
  104. else
  105. {
  106. // Create correlation data base entry for the correlation pass.
  107. NdrpAddCorrelationData( pStubMsg, pMemory, pFormat, Value, CheckKind );
  108. }
  109. }
  110. //
  111. void
  112. NdrpValidateCorrelatedValue (
  113. PMIDL_STUB_MESSAGE pStubMsg,
  114. uchar * pMemory,
  115. PFORMAT_STRING pFormat,
  116. LONG_PTR Value,
  117. int CheckKind
  118. )
  119. /*++
  120. Routine Description :
  121. This routine computes the conformant size for an array or the switch_is
  122. value for a union.
  123. Arguments :
  124. pStubMsg - Pointer to the stub message.
  125. pMemory - Pointer to the embedding entity: a struct or a stack top.
  126. pFormat - Format string description of the correlation (at old).
  127. It indicates the correlated argument value.
  128. Value - value from the array etc, i.e. from the buffer, to be checked against
  129. Return :
  130. The array or string size or the union switch_is.
  131. --*/
  132. {
  133. void * pCount = 0;
  134. LONG_PTR Count;
  135. unsigned char FormatCopy[4];
  136. BOOL fAsyncSplit = FALSE;
  137. BOOL fResetValue;
  138. fResetValue = CheckKind & NDR_RESET_VALUE;
  139. CheckKind = CheckKind & ~NDR_RESET_VALUE;
  140. // Ignore top level checks for -Os stubs.
  141. if ( !pMemory )
  142. return;
  143. PNDR_FCDEF_CORRELATION pFormatCorr = (PNDR_FCDEF_CORRELATION) pFormat;
  144. //
  145. // First check if this is a callback to an expression evaluation routine.
  146. //
  147. if ( pFormatCorr->Operation == FC_CALLBACK )
  148. {
  149. uchar * pOldStackTop;
  150. ushort Index;
  151. // Index into expression callback routines table.
  152. Index = (ushort) pFormatCorr->Offset;
  153. NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval != 0,
  154. "NdrpComputeConformance : no expr eval routines");
  155. NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval[Index] != 0,
  156. "NdrpComputeConformance : bad expr eval routine index");
  157. pOldStackTop = pStubMsg->StackTop;
  158. //
  159. // The callback routine uses the StackTop field of the stub message
  160. // to base it's offsets from. So if this is a complex attribute for
  161. // an embedded field of a structure then set StackTop equal to the
  162. // pointer to the structure.
  163. //
  164. if ( (*pFormat & 0xf0) != FC_TOP_LEVEL_CONFORMANCE )
  165. {
  166. pStubMsg->StackTop = pMemory;
  167. }
  168. //
  169. // This call puts the result in pStubMsg->MaxCount.
  170. //
  171. (*pStubMsg->StubDesc->apfnExprEval[Index])( pStubMsg );
  172. pStubMsg->StackTop = pOldStackTop;
  173. if ( CheckKind == NDR_CHECK_OFFSET )
  174. pStubMsg->MaxCount = pStubMsg->Offset;
  175. goto ValidateValue;
  176. }
  177. if ( CheckKind == NDR_CHECK_OFFSET )
  178. {
  179. // Checking offset without a call to expr eval routine -
  180. // this means that the offset should be zero.
  181. pStubMsg->MaxCount = 0;
  182. goto ValidateValue;
  183. }
  184. if ( (*pFormat & 0xf0) == FC_NORMAL_CONFORMANCE )
  185. {
  186. // Get the address where the conformance variable is in the struct.
  187. pCount = pMemory + pFormatCorr->Offset;
  188. goto ComputeConformantGetCount;
  189. }
  190. // See if this is an async split
  191. if ( pFormat[1] & 0x20 )
  192. {
  193. fAsyncSplit = TRUE;
  194. RpcpMemoryCopy( & FormatCopy[0], pFormat, 4 );
  195. pFormat = (PFORMAT_STRING) & FormatCopy[0];
  196. // Remove the async marker
  197. FormatCopy[1] = pFormat[1] & 0xdf; // ~0x20
  198. }
  199. //
  200. // Get a pointer to the conformance describing variable.
  201. //
  202. if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_CONFORMANCE )
  203. {
  204. //
  205. // Top level conformance. For /Os stubs, the stubs put the max
  206. // count in the stub message. For /Oi stubs, we get the max count
  207. // via an offset from the stack top.
  208. //
  209. if ( pStubMsg->StackTop )
  210. {
  211. if ( fAsyncSplit )
  212. {
  213. PNDR_DCOM_ASYNC_MESSAGE pAsyncMsg;
  214. pAsyncMsg = (PNDR_DCOM_ASYNC_MESSAGE) pStubMsg->pAsyncMsg;
  215. pCount = pAsyncMsg->BeginStack + (ushort)pFormatCorr->Offset;
  216. }
  217. else
  218. pCount = pStubMsg->StackTop + (ushort)pFormatCorr->Offset;
  219. goto ComputeConformantGetCount;
  220. }
  221. else
  222. {
  223. // Top level conformance with -Os - not supported yet.
  224. //
  225. // If this is top level conformance with /Os then
  226. // a) For early correlation, the compiler should generate the code to
  227. // assign appropriate value to pStubMsg->MaxCount.
  228. // goto ValideValue
  229. // b) For late correlation, we should have a registration call generated,
  230. // so there would be nothing to do.
  231. //
  232. return;
  233. }
  234. }
  235. //
  236. // If we're computing the size of an embedded sized pointer then we
  237. // use the memory pointer in the stub message, which points to the
  238. // beginning of the embedding structure.
  239. //
  240. if ( (*pFormat & 0xf0) == FC_POINTER_CONFORMANCE )
  241. {
  242. pCount = pMemory + pFormatCorr->Offset;
  243. goto ComputeConformantGetCount;
  244. }
  245. //
  246. // Check for constant size/switch.
  247. //
  248. if ( (*pFormat & 0xf0) == FC_CONSTANT_CONFORMANCE )
  249. {
  250. //
  251. // The size/switch is contained in the lower three bytes of the
  252. // long currently pointed to by pFormat.
  253. //
  254. Count = (ULONG_PTR)pFormat[1] << 16;
  255. Count |= (ULONG_PTR) *((ushort *)(pFormat + 2));
  256. goto ComputeConformanceEnd;
  257. }
  258. //
  259. // Check for conformance of a multidimensional array element in
  260. // a -Os stub.
  261. //
  262. if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_MULTID_CONFORMANCE )
  263. {
  264. long Dimension;
  265. if ( fAsyncSplit )
  266. RpcRaiseException( RPC_X_WRONG_STUB_VERSION );
  267. //
  268. // If pArrayInfo is non-null than we have a multi-D array. If it
  269. // is null then we have multi-leveled sized pointers.
  270. //
  271. if ( pStubMsg->pArrayInfo )
  272. {
  273. Dimension = pStubMsg->pArrayInfo->Dimension;
  274. pStubMsg->MaxCount = pStubMsg->pArrayInfo->MaxCountArray[Dimension];
  275. }
  276. else
  277. {
  278. Dimension = *((ushort *)(pFormat + 2));
  279. pStubMsg->MaxCount = pStubMsg->SizePtrCountArray[Dimension];
  280. }
  281. goto ValidateValue;
  282. }
  283. NDR_ASSERT(0, "NdrpValidateCorrelatedValue:, Invalid Correlation type");
  284. RpcRaiseException( RPC_S_INTERNAL_ERROR );
  285. return;
  286. ComputeConformantGetCount:
  287. //
  288. // Must check now if there is a dereference op.
  289. //
  290. if ( pFormatCorr->Operation == FC_DEREFERENCE )
  291. {
  292. pCount = *(void **)pCount;
  293. }
  294. //
  295. // If we're supposed to whack the value instead of checking it do it
  296. // and quit
  297. //
  298. if ( fResetValue )
  299. {
  300. // hypers are not legal types for cs_char size/length_is expressions
  301. if ( FC_HYPER == pFormatCorr->Type )
  302. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  303. CHECK_BOUND( (long)Value, pFormatCorr->Type );
  304. switch ( pFormatCorr->Type )
  305. {
  306. case FC_ULONG:
  307. * (ulong *) pCount = (ulong) Value;
  308. return;
  309. case FC_LONG:
  310. * (ulong *) pCount = (long) Value;
  311. return;
  312. case FC_ENUM16 :
  313. case FC_USHORT :
  314. * (ushort *) pCount = (ushort) Value;
  315. return;
  316. case FC_SHORT :
  317. * (short *) pCount = (short) Value;
  318. return;
  319. case FC_USMALL :
  320. * (uchar *) pCount = (uchar) Value;
  321. return;
  322. case FC_SMALL :
  323. * (char *) pCount = (char) Value;
  324. return;
  325. default :
  326. NDR_ASSERT(0,"NdrpValidateCorrelatedValue : bad reset type");
  327. RpcRaiseException( RPC_S_INTERNAL_ERROR );
  328. }
  329. }
  330. //
  331. // Now get the conformance count.
  332. //
  333. switch ( pFormatCorr->Type )
  334. {
  335. case FC_HYPER :
  336. // iid_is on 64b platforms only.
  337. Count = *((LONG_PTR *)pCount);
  338. break;
  339. case FC_ULONG :
  340. Count = (LONG_PTR)*((ulong *)pCount);
  341. break;
  342. case FC_LONG :
  343. Count = *((long *)pCount);
  344. break;
  345. case FC_ENUM16:
  346. case FC_USHORT :
  347. Count = (long) *((ushort *)pCount);
  348. break;
  349. case FC_SHORT :
  350. Count = (long) *((short *)pCount);
  351. break;
  352. case FC_USMALL :
  353. Count = (long) *((uchar *)pCount);
  354. break;
  355. case FC_SMALL :
  356. Count = (long) *((char *)pCount);
  357. break;
  358. default :
  359. NDR_ASSERT(0,"NdrpValidateCorrelatedValue : bad count type");
  360. RpcRaiseException( RPC_S_INTERNAL_ERROR );
  361. }
  362. //
  363. // Check the operator.
  364. //
  365. switch ( pFormatCorr->Operation )
  366. {
  367. case FC_DIV_2 :
  368. Count /= 2;
  369. break;
  370. case FC_MULT_2 :
  371. Count *= 2;
  372. break;
  373. case FC_SUB_1 :
  374. Count -= 1;
  375. break;
  376. case FC_ADD_1 :
  377. Count += 1;
  378. break;
  379. default :
  380. // OK
  381. break;
  382. }
  383. ComputeConformanceEnd:
  384. pStubMsg->MaxCount = (ulong) Count;
  385. ValidateValue:
  386. // Compare pStubMsg->MaxCount with the value
  387. BOOL fValueOK = FALSE;
  388. LONG_PTR ArgValue = (LONG_PTR) pStubMsg->MaxCount;
  389. unsigned char FcType = (uchar)pFormatCorr->Type;
  390. if ( (*pFormat & 0xf0) == FC_CONSTANT_CONFORMANCE ||
  391. pFormatCorr->Operation == FC_CALLBACK )
  392. FcType = FC_ULONG;
  393. switch ( FcType )
  394. {
  395. case FC_HYPER :
  396. fValueOK = ArgValue == (LONG_PTR)Value;
  397. break;
  398. case FC_ULONG :
  399. fValueOK = ArgValue == (LONG_PTR)(ulong)Value;
  400. break;
  401. case FC_LONG :
  402. fValueOK = ArgValue == (LONG_PTR)(long)Value;
  403. break;
  404. case FC_ENUM16:
  405. case FC_USHORT :
  406. fValueOK = ArgValue == (LONG_PTR)(ushort)Value;
  407. break;
  408. case FC_SHORT :
  409. fValueOK = ArgValue == (LONG_PTR)(short)Value;
  410. break;
  411. case FC_USMALL :
  412. fValueOK = ArgValue == (LONG_PTR)(uchar)Value;
  413. break;
  414. case FC_SMALL :
  415. fValueOK = ArgValue == (LONG_PTR)(char)Value;
  416. break;
  417. default :
  418. NDR_ASSERT(0,"NdrpValidateCorrelatedValue : bad count type");
  419. RpcRaiseException( RPC_S_INTERNAL_ERROR );
  420. }
  421. if ( ! fValueOK )
  422. {
  423. if ( ! pFormatCorr->CorrFlags.IsIidIs )
  424. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  425. else
  426. {
  427. // For interface pointers the check is more complicated.
  428. // Value is in this case a pointer to the value, actually.
  429. IID * piidValue = (IID *)Value;
  430. IID * piidArg = (IID *)ArgValue;
  431. if ( 0 != memcmp( piidValue, piidArg, sizeof( IID )))
  432. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  433. }
  434. }
  435. return;
  436. }
  437. RPCRTAPI
  438. void
  439. RPC_ENTRY
  440. NdrCorrelationInitialize(
  441. PMIDL_STUB_MESSAGE pStubMsg,
  442. void * pCache,
  443. unsigned long CacheSize,
  444. unsigned long Flags
  445. )
  446. /*
  447. Initializes the correlation package for -Os stubs.
  448. */
  449. {
  450. PNDR_CORRELATION_INFO pCorrInfo = (PNDR_CORRELATION_INFO)pCache;
  451. if ( CacheSize == 0xffffffff )
  452. CacheSize = NDR_DEFAULT_CORR_CACHE_SIZE;
  453. pCorrInfo->Header.pCache = (NDR_CORRELATION_INFO *)pCache;
  454. pCorrInfo->Header.pInfo = (NDR_CORRELATION_INFO *)pCache;
  455. pCorrInfo->Header.DataLen = 0;
  456. pCorrInfo->Header.DataSize = (CacheSize - sizeof(NDR_CORRELATION_INFO))
  457. / sizeof(NDR_CORRELATION_INFO_DATA);
  458. pStubMsg->pCorrMemory = pStubMsg->StackTop;
  459. pStubMsg->pCorrInfo = (NDR_CORRELATION_INFO *)pCache;
  460. pStubMsg->fHasExtensions = 1;
  461. pStubMsg->fHasNewCorrDesc = 1;
  462. }
  463. void
  464. NdrpAddCorrelationData(
  465. PMIDL_STUB_MESSAGE pStubMsg,
  466. uchar * pMemory,
  467. PFORMAT_STRING pFormat,
  468. LONG_PTR Value,
  469. int CheckKind
  470. )
  471. /*
  472. Adds a check data to the correlation data base for a later evaluation.
  473. */
  474. {
  475. ASSERT( pStubMsg->pCorrInfo );
  476. PNDR_CORRELATION_INFO pCorrInfo = (PNDR_CORRELATION_INFO)pStubMsg->pCorrInfo;
  477. if ( pCorrInfo->Header.DataSize <= pCorrInfo->Header.DataLen )
  478. {
  479. // Not enough space, need to realloc.
  480. unsigned long NewDataSize;
  481. PNDR_CORRELATION_INFO pCorrInfoNew;
  482. NewDataSize = 2 * pCorrInfo->Header.DataSize;
  483. pCorrInfoNew = (PNDR_CORRELATION_INFO) I_RpcAllocate(
  484. sizeof(NDR_CORRELATION_INFO_HEADER) +
  485. NewDataSize * sizeof(NDR_CORRELATION_INFO_DATA) );
  486. if ( ! pCorrInfoNew )
  487. RpcRaiseException( RPC_S_OUT_OF_MEMORY );
  488. RpcpMemoryCopy( pCorrInfoNew,
  489. pCorrInfo,
  490. sizeof(NDR_CORRELATION_INFO_HEADER) +
  491. pCorrInfo->Header.DataSize * sizeof(NDR_CORRELATION_INFO_DATA) );
  492. pCorrInfoNew->Header.pInfo = pCorrInfoNew;
  493. pCorrInfoNew->Header.DataSize = NewDataSize;
  494. pStubMsg->pCorrInfo = pCorrInfoNew;
  495. if ( pCorrInfo->Header.pInfo != pCorrInfo->Header.pCache )
  496. I_RpcFree( pCorrInfo->Header.pInfo );
  497. pCorrInfo = pCorrInfoNew;
  498. }
  499. pCorrInfo->Data[ pCorrInfo->Header.DataLen ].pMemoryObject = pMemory;
  500. pCorrInfo->Data[ pCorrInfo->Header.DataLen ].Value = Value;
  501. pCorrInfo->Data[ pCorrInfo->Header.DataLen ].pCorrDesc = pFormat;
  502. pCorrInfo->Data[ pCorrInfo->Header.DataLen ].CheckKind = CheckKind;
  503. pCorrInfo->Header.DataLen++;
  504. }
  505. RPCRTAPI
  506. void
  507. RPC_ENTRY
  508. NdrCorrelationPass(
  509. PMIDL_STUB_MESSAGE pStubMsg
  510. )
  511. /*
  512. Walks the data base to check all the correlated values that could not be checked
  513. on fly.
  514. */
  515. {
  516. ASSERT( pStubMsg->pCorrInfo );
  517. if ( pStubMsg->pCorrInfo->Header.DataLen == 0 )
  518. return;
  519. PNDR_CORRELATION_INFO pCorrInfo = (PNDR_CORRELATION_INFO)pStubMsg->pCorrInfo;
  520. for ( int i = 0; i < pCorrInfo->Header.DataLen; i++ )
  521. {
  522. NdrpValidateCorrelatedValue( pStubMsg,
  523. pCorrInfo->Data[ i ].pMemoryObject,
  524. pCorrInfo->Data[ i ].pCorrDesc,
  525. pCorrInfo->Data[ i ].Value,
  526. pCorrInfo->Data[ i ].CheckKind );
  527. }
  528. }
  529. RPCRTAPI
  530. void
  531. RPC_ENTRY
  532. NdrCorrelationFree(
  533. PMIDL_STUB_MESSAGE pStubMsg
  534. )
  535. /*
  536. Releases the correlation data structures.
  537. In /Os stub, NdrCorrelationInitialize is called after fullpointer initialization
  538. and NdrCorrelationFree is called without checking pStubMsg->pCorrInfo. changing
  539. ndr might be better in this case.
  540. */
  541. {
  542. // ASSERT( pStubMsg->pCorrInfo );
  543. PNDR_CORRELATION_INFO pCorrInfo = (PNDR_CORRELATION_INFO)pStubMsg->pCorrInfo;
  544. if ( NULL == pCorrInfo )
  545. return;
  546. if ( pCorrInfo->Header.pInfo != pCorrInfo->Header.pCache )
  547. {
  548. I_RpcFree( pCorrInfo->Header.pInfo );
  549. }
  550. }
  551. HRESULT
  552. NdrpGetRpcHelper(
  553. IRpcHelper ** ppRpcHelper )
  554. /*++
  555. This routine attempts to get hold of an IRpcHelper interface pointer.
  556. We cache the one IP on each process and never release it. This is an inproc server
  557. and interface will be cleanup when process exit.
  558. --*/
  559. {
  560. HRESULT hr;
  561. static IRpcHelper *pRpcHelper = NULL;
  562. *ppRpcHelper = NULL;
  563. if (pRpcHelper)
  564. {
  565. *ppRpcHelper = pRpcHelper;
  566. hr = S_OK;
  567. }
  568. else
  569. {
  570. hr = NdrLoadOleRoutines();
  571. if (SUCCEEDED(hr))
  572. hr = (*pfnCoCreateInstance)( CLSID_RpcHelper,
  573. NULL,
  574. CLSCTX_INPROC_SERVER,
  575. IID_IRpcHelper,
  576. (void **)&pRpcHelper );
  577. if (SUCCEEDED(hr))
  578. *ppRpcHelper = pRpcHelper;
  579. }
  580. return hr;
  581. }
  582. void
  583. NdrpGetIIDFromBuffer(
  584. PMIDL_STUB_MESSAGE pStubMsg,
  585. IID ** ppIID
  586. )
  587. /*
  588. The routine recovers an interface pointer IID from the marshaling buffer.
  589. The IID stays in the buffer, an IID* is returned by the routine.
  590. */
  591. {
  592. IRpcHelper * pRpcHelper = 0;
  593. HRESULT hr;
  594. *ppIID = 0;
  595. hr = NdrpGetRpcHelper( & pRpcHelper );
  596. if(FAILED(hr))
  597. RpcRaiseException(hr);
  598. hr = pRpcHelper->GetIIDFromOBJREF( pStubMsg->Buffer,
  599. ppIID );
  600. if (FAILED(hr))
  601. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  602. // The IRpcHelper is cached in NdrpGetRpcHelper so we don't need to release it.
  603. // pRpcHelper->Release();
  604. }