Leaked source code of windows server 2003
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.

1464 lines
49 KiB

  1. /*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2. Copyright (c) 1999-2000 Microsoft Corporation
  3. Module Name :
  4. mulsyntx.cxx
  5. Abstract :
  6. This file contains multiple transfer syntaxes negotiation related code
  7. Author :
  8. Yong Qu yongqu September 1999.
  9. Revision History :
  10. ---------------------------------------------------------------------*/
  11. #include "ndrp.h"
  12. #define CINTERFACE
  13. #include "ndrole.h"
  14. #include "rpcproxy.h"
  15. #include "mulsyntx.h"
  16. #include "hndl.h"
  17. #include "auxilary.h"
  18. #include "pipendr.h"
  19. #include "ndr64tkn.h"
  20. const extern PMARSHALL_ROUTINE MarshallRoutinesTable[];
  21. const extern PUNMARSHALL_ROUTINE UnmarshallRoutinesTable[];
  22. const extern PSIZE_ROUTINE SizeRoutinesTable[];
  23. const extern PMEM_SIZE_ROUTINE MemSizeRoutinesTable[];
  24. const extern PFREE_ROUTINE FreeRoutinesTable[];
  25. //const extern PWALKIP_ROUTINE WalkIPRoutinesTable[];
  26. // TODO: move this to ndrpall.h after Preview.
  27. #define MIDL_VERSION_6_0_322 ((6UL << 24) | (0UL << 16) | 322UL)
  28. void RPC_ENTRY
  29. NdrpClientInit(MIDL_STUB_MESSAGE * pStubMsg,
  30. void * pReturnValue )
  31. {
  32. PFORMAT_STRING pFormatParam;
  33. ulong ProcNum;
  34. BOOL fRaiseExcFlag;
  35. ulong n;
  36. uchar * pArg;
  37. INTERPRETER_FLAGS InterpreterFlags;
  38. PPARAM_DESCRIPTION Params;
  39. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *) pStubMsg->pContext;
  40. INTERPRETER_OPT_FLAGS OptFlags = pContext->NdrInfo.pProcDesc->Oi2Flags;
  41. PFORMAT_STRING pTypeFormat;
  42. // When this routine is called from ndrclientcall2, we don't have MIDL_SYNTAX_INFO,
  43. // so we need to read it from MIDL_STUB_DESC;
  44. // Note: we need to conenct StubDesc to pStubMsg before calling into here.
  45. if ( NULL == pContext->pSyntaxInfo )
  46. pContext->DceTypeFormatString = pStubMsg->StubDesc->pFormatTypes;
  47. else
  48. pContext->DceTypeFormatString = pContext->pSyntaxInfo->TypeString;
  49. InterpreterFlags = pContext->NdrInfo.InterpreterFlags;
  50. Params = ( PPARAM_DESCRIPTION )pContext->Params;
  51. pTypeFormat = pContext->DceTypeFormatString;
  52. if ( InterpreterFlags.FullPtrUsed )
  53. pStubMsg->FullPtrXlatTables = NdrFullPointerXlatInit( 0, XLAT_CLIENT );
  54. else
  55. pStubMsg->FullPtrXlatTables = 0;
  56. if ( InterpreterFlags.RpcSsAllocUsed )
  57. NdrRpcSmSetClientToOsf( pStubMsg );
  58. // Set Rpc flags after the call to client initialize.
  59. pStubMsg->RpcMsg->RpcFlags = pContext->RpcFlags;
  60. if ( OptFlags.HasPipes )
  61. NdrpPipesInitialize32( pStubMsg,
  62. &pContext->AllocateContext,
  63. (PFORMAT_STRING) pContext->Params,
  64. ( char * )pContext->StartofStack,
  65. pContext->NumberParams );
  66. // Must do this before the sizing pass!
  67. pStubMsg->StackTop = pContext->StartofStack;
  68. if ( OptFlags.HasExtensions )
  69. {
  70. pStubMsg->fHasExtensions = 1;
  71. pStubMsg->fHasNewCorrDesc = pContext->NdrInfo.pProcDesc->NdrExts.Flags2.HasNewCorrDesc;
  72. if ( pContext->NdrInfo.pProcDesc->NdrExts.Flags2.ClientCorrCheck )
  73. {
  74. ulong *pCache =
  75. (ulong*)NdrpAlloca(&pContext->AllocateContext,
  76. NDR_DEFAULT_CORR_CACHE_SIZE );
  77. NdrCorrelationInitialize( pStubMsg,
  78. pCache,
  79. NDR_DEFAULT_CORR_CACHE_SIZE,
  80. 0 /* flags */ );
  81. }
  82. }
  83. //
  84. // Get the compile time computed buffer size.
  85. //
  86. pStubMsg->BufferLength = pContext->NdrInfo.pProcDesc->ClientBufferSize;
  87. //
  88. // Check ref pointers and do object proc [out] zeroing.
  89. //
  90. fRaiseExcFlag = FALSE;
  91. for ( n = 0; n < pContext->NumberParams; n++ )
  92. {
  93. if ( Params[n].ParamAttr.IsReturn )
  94. pArg = (uchar *) &pReturnValue;
  95. else
  96. pArg = pContext->StartofStack + Params[n].StackOffset;
  97. if ( Params[n].ParamAttr.IsSimpleRef && !Params[n].ParamAttr.IsReturn )
  98. {
  99. // We cannot raise the exception here,
  100. // as some out args may not be zeroed out yet.
  101. if ( ! *((uchar **)pArg) )
  102. {
  103. fRaiseExcFlag = TRUE;
  104. continue;
  105. }
  106. }
  107. // if top level point is FC_RP and the arg is NULL, we'll catch this
  108. // before the call goes to server.
  109. // We wouldn't catch all the null ref pointer here, but we should be able
  110. // to catch the most obvious ones.
  111. // This code is called from sync & raw async code; in async dcom,
  112. // we only go through outinit in finish routine so we can't do anything anyhow.
  113. if ( Params[n].ParamAttr.IsOut && ! Params[n].ParamAttr.IsBasetype )
  114. {
  115. pFormatParam = pTypeFormat + Params[n].TypeOffset;
  116. if ( * pFormatParam == FC_RP && !*((uchar **)pArg) )
  117. {
  118. fRaiseExcFlag = TRUE;
  119. continue;
  120. }
  121. }
  122. //
  123. // In object procs and complex return types we have to zero
  124. // out all [out] parameters. We do the basetype check to
  125. // cover the [out] simple ref to basetype case.
  126. //
  127. if ( ( InterpreterFlags.ObjectProc &&
  128. ! pContext->IsAsync &&
  129. ( Params[n].ParamAttr.IsPartialIgnore ||
  130. ( ! Params[n].ParamAttr.IsIn &&
  131. ! Params[n].ParamAttr.IsReturn &&
  132. ! Params[n].ParamAttr.IsPipe ) ) ) ||
  133. ( pContext->HasComplexReturn &&
  134. Params[n].ParamAttr.IsReturn ) )
  135. {
  136. if ( Params[n].ParamAttr.IsBasetype )
  137. {
  138. // [out] only arg can only be ref, we checked that above.
  139. MIDL_memset( *(uchar **)pArg,
  140. 0,
  141. (size_t)SIMPLE_TYPE_MEMSIZE( Params[n].SimpleType.Type ));
  142. }
  143. else
  144. {
  145. pFormatParam = pTypeFormat +
  146. Params[n].TypeOffset;
  147. NdrClientZeroOut(
  148. pStubMsg,
  149. pFormatParam,
  150. *(uchar **)pArg );
  151. }
  152. }
  153. }
  154. if ( fRaiseExcFlag )
  155. RpcRaiseException( RPC_X_NULL_REF_POINTER );
  156. if ( !OptFlags.ClientMustSize )
  157. pContext->pfnSizing = (PFNSIZING)NdrpNoopSizing;
  158. }
  159. void RPC_ENTRY
  160. NdrpNoopSizing( MIDL_STUB_MESSAGE * pStubMsg,
  161. BOOL IsClient )
  162. {
  163. return;
  164. }
  165. void RPC_ENTRY
  166. NdrpSizing( MIDL_STUB_MESSAGE * pStubMsg,
  167. BOOL IsClient )
  168. {
  169. PFORMAT_STRING pFormatParam;
  170. ulong n;
  171. uchar * pArg;
  172. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  173. PPARAM_DESCRIPTION Params = ( PPARAM_DESCRIPTION )pContext->Params;
  174. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  175. //
  176. // Skip buffer size pass if possible.
  177. //
  178. for ( n = 0; n < pContext->NumberParams; n++ )
  179. {
  180. if ( !SAMEDIRECTION(IsClient, Params[n]) ||
  181. ! Params[n].ParamAttr.MustSize )
  182. continue;
  183. if ( IsClient &&
  184. Params[n].ParamAttr.IsPartialIgnore )
  185. {
  186. LENGTH_ALIGN( pStubMsg->BufferLength, 0x3 );
  187. pStubMsg->BufferLength += PTR_WIRE_SIZE;
  188. continue;
  189. }
  190. //
  191. // Note : Basetypes will always be factored into the
  192. // constant buffer size emitted by in the format strings.
  193. //
  194. pFormatParam = pTypeFormat +
  195. Params[n].TypeOffset;
  196. pArg = pContext->StartofStack + Params[n].StackOffset;
  197. if ( ! Params[n].ParamAttr.IsByValue )
  198. pArg = *((uchar **)pArg);
  199. (*SizeRoutinesTable[ROUTINE_INDEX(*pFormatParam)])
  200. ( pStubMsg,
  201. pArg,
  202. pFormatParam );
  203. }
  204. }
  205. void RPC_ENTRY
  206. NdrpClientMarshal( MIDL_STUB_MESSAGE * pStubMsg,
  207. BOOL IsObject )
  208. {
  209. ulong n;
  210. uchar * pArg;
  211. PFORMAT_STRING pFormatParam;
  212. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  213. PPARAM_DESCRIPTION Params = (PPARAM_DESCRIPTION ) pContext->Params;
  214. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  215. //
  216. // ----------------------------------------------------------
  217. // Marshall Pass.
  218. // ----------------------------------------------------------
  219. //
  220. for ( n = 0; n < pContext->NumberParams; n++ )
  221. {
  222. if (!Params[n].ParamAttr.IsIn ||
  223. Params[n].ParamAttr.IsPipe )
  224. continue;
  225. if ( Params[n].ParamAttr.IsPartialIgnore )
  226. {
  227. pArg = pContext->StartofStack + Params[n].StackOffset;
  228. ALIGN( pStubMsg->Buffer, 0x3 );
  229. *(ulong*)pStubMsg->Buffer = *(void**)pArg ? 1 : 0;
  230. pStubMsg->Buffer += PTR_WIRE_SIZE;
  231. continue;
  232. }
  233. pArg = pContext->StartofStack + Params[n].StackOffset;
  234. if ( Params[n].ParamAttr.IsBasetype )
  235. {
  236. //
  237. // Check for pointer to basetype.
  238. //
  239. if ( Params[n].ParamAttr.IsSimpleRef )
  240. pArg = *((uchar **)pArg);
  241. else
  242. {
  243. #if defined(_IA64_) || defined(_AMD64_)
  244. if ( !IsObject &&
  245. Params[n].SimpleType.Type == FC_FLOAT )
  246. {
  247. // Due to the fact that NdrClientCall2 is called with the
  248. // parameters in ... arguments, floats get promoted to doubles.
  249. // This is not true for DCOM since an assembly langauge wrapper
  250. // is used that saves the floats as floats.
  251. //
  252. // We do not handle this case properly.
  253. *((float *) pArg) = (float) *((double *)pArg);
  254. }
  255. #endif
  256. }
  257. if ( Params[n].SimpleType.Type == FC_ENUM16 )
  258. {
  259. if ( *((int *)pArg) & ~((int)0x7fff) )
  260. RpcRaiseException(RPC_X_ENUM_VALUE_OUT_OF_RANGE);
  261. }
  262. // a fastcall lookup should be faster than possible 3 missing caches, and we can zero out the gap.
  263. SimpleTypeMarshallRoutinesTable[ROUTINE_INDEX( Params[n].SimpleType.Type)](pStubMsg, pArg );
  264. continue;
  265. }
  266. pFormatParam = pTypeFormat +
  267. Params[n].TypeOffset;
  268. if ( ! Params[n].ParamAttr.IsByValue )
  269. pArg = *((uchar **)pArg);
  270. (* MarshallRoutinesTable[ROUTINE_INDEX(*pFormatParam)] )
  271. ( pStubMsg,
  272. pArg,
  273. pFormatParam );
  274. }
  275. if ( pStubMsg->RpcMsg->BufferLength <
  276. (uint)(pStubMsg->Buffer - (uchar *)pStubMsg->RpcMsg->Buffer) )
  277. {
  278. NDR_ASSERT( 0, "NdrpClientMarshal marshal: buffer overflow!" );
  279. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  280. }
  281. }
  282. void RPC_ENTRY
  283. NdrpServerMarshal( MIDL_STUB_MESSAGE * pStubMsg,
  284. BOOL IsObject )
  285. {
  286. ulong n;
  287. uchar * pArg;
  288. PFORMAT_STRING pFormatParam;
  289. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  290. PPARAM_DESCRIPTION Params = (PPARAM_DESCRIPTION ) pContext->Params;
  291. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  292. //
  293. // ----------------------------------------------------------
  294. // Marshall Pass.
  295. // ----------------------------------------------------------
  296. //
  297. for ( n = 0; n < pContext->NumberParams; n++ )
  298. {
  299. if (!Params[n].ParamAttr.IsOut ||
  300. Params[n].ParamAttr.IsPipe )
  301. continue;
  302. pArg = pContext->StartofStack + Params[n].StackOffset;
  303. if ( Params[n].ParamAttr.IsBasetype )
  304. {
  305. //
  306. // Check for pointer to basetype.
  307. //
  308. if ( Params[n].ParamAttr.IsSimpleRef )
  309. pArg = *((uchar **)pArg);
  310. if ( Params[n].SimpleType.Type == FC_ENUM16 )
  311. {
  312. if ( *((int *)pArg) & ~((int)0x7fff) )
  313. RpcRaiseException(RPC_X_ENUM_VALUE_OUT_OF_RANGE);
  314. }
  315. // a fastcall lookup should be faster than possible 3 missing caches, and we can zero out the gap.
  316. SimpleTypeMarshallRoutinesTable[ROUTINE_INDEX( Params[n].SimpleType.Type)](pStubMsg, pArg );
  317. continue;
  318. }
  319. pFormatParam = pTypeFormat +
  320. Params[n].TypeOffset;
  321. if ( ! Params[n].ParamAttr.IsByValue )
  322. pArg = *((uchar **)pArg);
  323. (* MarshallRoutinesTable[ROUTINE_INDEX(*pFormatParam)] )
  324. ( pStubMsg,
  325. pArg,
  326. pFormatParam );
  327. }
  328. if ( pStubMsg->RpcMsg->BufferLength <
  329. (uint)(pStubMsg->Buffer - (uchar *)pStubMsg->RpcMsg->Buffer) )
  330. {
  331. NDR_ASSERT( 0, "NdrpServerMarshal marshal: buffer overflow!" );
  332. RpcRaiseException( RPC_X_BAD_STUB_DATA );
  333. }
  334. }
  335. /*
  336. From: To: Marshall: User Exception: Cleanup:Rundown: context handle:
  337. ---------- ---------------------------------- --------------------------------
  338. Any NULL Y N/A No No INVALID_HANDLE from the server
  339. !NULL Same value Any Any No No Same as before
  340. !NULL Different Any Any No No New context on the server
  341. !NULL Any N/A Y No No No new context handle is created
  342. NULL ANY N/A Y Yes No INVALID_HANDLE from the server
  343. NULL !NULL Y N Yes Yes INVALID_HANDLE from the server
  344. NULL !NULL N N Yes Yes INVALID_HANDLE from the server
  345. !NULL NULL N N Yes No No new context handle is created
  346. NULL NULL N N Yes No No new context handle is created
  347. In the OUT only context handle case:
  348. To: Marshall: User Exception: Cleanup:Rundown: context handle:
  349. --------------------------------------------------------------------------------------
  350. Any N/A Y N N No new context handle is created
  351. NULL N N N N No new context handle is created
  352. NULL Y N N N No new context handle is created
  353. !NULL N N N Y No new context handle is created
  354. !NULL Y N Y Y No new context handle is created
  355. */
  356. void
  357. NdrpEmergencyContextCleanup(
  358. MIDL_STUB_MESSAGE * pStubMsg,
  359. PNDR_CONTEXT_HANDLE_ARG_DESC pCtxtDesc,
  360. void * pArg,
  361. BOOL fManagerRoutineException
  362. )
  363. {
  364. int RtnIndex = pCtxtDesc->RundownRtnIndex;
  365. NDR_RUNDOWN pfnRundown = pStubMsg->StubDesc->apfnNdrRundownRoutines[ RtnIndex ];
  366. NDR_SCONTEXT SContext = pStubMsg->SavedContextHandles[ pCtxtDesc->ParamOrdinal ];
  367. void * NewHandle = pArg;
  368. // if runtime failes during marshalling context handle, we shouldn't call into
  369. // cleanup routine since it's already cleanup by runtime.
  370. if ( SContext == (NDR_SCONTEXT )CONTEXT_HANDLE_BEFORE_MARSHAL_MARKER )
  371. return;
  372. if ( fManagerRoutineException )
  373. {
  374. // NDR_ASSERT( SContext != NULL || pCtxtDesc->Flags.IsReturn ,
  375. // "only return context handle can have null scontext in exception" );
  376. // if we failed somewhere during unmarshalling, or this is a return context handle,
  377. // we don't need to cleanup
  378. if ( SContext == NULL )
  379. return;
  380. // in NdrServerOutInit, we initialize the scontext for regular [out] parameters,
  381. // and runtime has already allocated some memory. But for return context handle,
  382. // we initialize the scontext during marshalling and saved context is NULL till
  383. // just before marshalling.
  384. }
  385. else
  386. {
  387. // haven't unmarshalled yet.
  388. if ( SContext == NULL )
  389. {
  390. if ( NULL == NewHandle )
  391. return;
  392. else
  393. {
  394. // note : what if the context handle is both return and viaptr?
  395. NDR_ASSERT( pCtxtDesc->Flags.IsReturn, "has to be return context handle" );
  396. }
  397. }
  398. else
  399. if ( SContext == (NDR_SCONTEXT )CONTEXT_HANDLE_AFTER_MARSHAL_MARKER )
  400. {
  401. // this particular context handle has been marshalled; the exception happens
  402. // during marshalling other parameters after this one.
  403. // After marshalling is done, the runtime will release the user context if new context
  404. // handle is NULL, so we can't reference to the user context at this point.
  405. NewHandle = NULL;
  406. }
  407. else
  408. {
  409. if ( pCtxtDesc->Flags.IsViaPtr )
  410. NewHandle = *((void * UNALIGNED *)pArg);
  411. }
  412. // if this is a regular [in,out] or [out] context handle, and it hasn't been marshalled
  413. // yet, we need to call into runtime to cleanup.
  414. }
  415. // Kamen volunteer to process the logic of calling runtime or not. In NDR we just
  416. // don't call into the routine when it isn't supposed to.
  417. NDRSContextEmergencyCleanup( pStubMsg->RpcMsg->Handle,
  418. SContext,
  419. pfnRundown,
  420. NewHandle,
  421. fManagerRoutineException );
  422. }
  423. void
  424. NdrpCleanupServerContextHandles(
  425. MIDL_STUB_MESSAGE * pStubMsg,
  426. uchar * pStartOfStack,
  427. BOOL fManagerRoutineException )
  428. {
  429. ulong n;
  430. uchar * pArg;
  431. PFORMAT_STRING pFormatParam;
  432. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  433. PPARAM_DESCRIPTION Params = (PPARAM_DESCRIPTION ) pContext->Params;
  434. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  435. //
  436. // ------------------------------------------------------------------------
  437. // Context handle loop: clean up out context handles to prevent leaks.
  438. //
  439. // Note, this routine handles only the handles that may have been dropped
  440. // due to the NDR engine raising exception between a clean return from
  441. // the manager routine and end of marshaling back of the parameters.
  442. // This addresses a situation where handles get dropped by NDR without being
  443. // registered with the runtime and so the server leaks because the rundown
  444. // routine is never called on the dropped handles.
  445. // ------------------------------------------------------------------------
  446. //
  447. for ( n = 0; n < pContext->NumberParams; n++ )
  448. {
  449. if (!Params[n].ParamAttr.IsOut ||
  450. Params[n].ParamAttr.IsPipe )
  451. continue;
  452. pArg = pContext->StartofStack + Params[n].StackOffset;
  453. if ( Params[n].ParamAttr.IsBasetype )
  454. {
  455. //
  456. // Check for pointer to basetype.
  457. //
  458. continue;
  459. }
  460. pFormatParam = pTypeFormat +
  461. Params[n].TypeOffset;
  462. if ( ! Params[n].ParamAttr.IsByValue )
  463. pArg = *((uchar **)pArg);
  464. // Context handle have their own "via pointer" flag to mark dereference.
  465. if ( *pFormatParam == FC_BIND_CONTEXT )
  466. {
  467. NdrpEmergencyContextCleanup( pStubMsg,
  468. (PNDR_CONTEXT_HANDLE_ARG_DESC) pFormatParam,
  469. pArg,
  470. fManagerRoutineException
  471. );
  472. }
  473. } // parameter loop
  474. }
  475. void RPC_ENTRY
  476. NdrpClientUnMarshal ( MIDL_STUB_MESSAGE * pStubMsg,
  477. void * pReturnValue )
  478. {
  479. ulong n;
  480. uchar * pArg;
  481. uchar ** ppArg;
  482. PFORMAT_STRING pFormatParam, pFormat;
  483. NDR_PROC_CONTEXT * pContext = ( NDR_PROC_CONTEXT *) pStubMsg->pContext;
  484. PPARAM_DESCRIPTION Params = (PPARAM_DESCRIPTION)pContext->Params;
  485. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  486. // we only need to do conversion in NDR32 now: we cut off endian
  487. // conversion in NDR64.
  488. // Do endian/floating point conversions if necessary.
  489. //
  490. if ( (pStubMsg->RpcMsg->DataRepresentation & 0X0000FFFFUL) !=
  491. NDR_LOCAL_DATA_REPRESENTATION )
  492. {
  493. NdrConvert2( pStubMsg,
  494. (PFORMAT_STRING) Params,
  495. pContext->NumberParams );
  496. }
  497. #ifdef TEST_CONVERT
  498. NdrConvert2( pStubMsg,
  499. (PFORMAT_STRING) Params,
  500. pContext->NumberParams );
  501. #endif
  502. //
  503. // ----------------------------------------------------------
  504. // Unmarshall Pass.
  505. // ----------------------------------------------------------
  506. //
  507. for ( n = 0; n < pContext->NumberParams; n++ )
  508. {
  509. if ( Params[n].ParamAttr.IsPipe )
  510. continue;
  511. if ( !Params[n].ParamAttr.IsOut )
  512. {
  513. if ( !Params[n].ParamAttr.IsIn
  514. && !Params[n].ParamAttr.IsReturn )
  515. {
  516. // If a param is not [in], [out], or a return value,
  517. // then it is a "hidden" client-side only status
  518. // paramater. It will get set below if an exception
  519. // happens. If everything is ok we need to zero it
  520. // out here.
  521. NDR_ASSERT( Params[n].ParamAttr.IsSimpleRef
  522. && Params[n].ParamAttr.IsBasetype
  523. && FC_ERROR_STATUS_T ==
  524. Params[n].SimpleType.Type,
  525. "Apparently not a hidden status param" );
  526. pArg = pContext->StartofStack + Params[n].StackOffset;
  527. ** (error_status_t **) pArg = RPC_S_OK;
  528. }
  529. continue;
  530. }
  531. if ( Params[n].ParamAttr.IsReturn )
  532. pArg = (uchar *) pReturnValue;
  533. else
  534. pArg = pContext->StartofStack + Params[n].StackOffset;
  535. //
  536. // This is for returned basetypes and for pointers to
  537. // basetypes.
  538. //
  539. if ( Params[n].ParamAttr.IsBasetype )
  540. {
  541. //
  542. // Check for a pointer to a basetype.
  543. //
  544. if ( Params[n].ParamAttr.IsSimpleRef )
  545. pArg = *((uchar **)pArg);
  546. ALIGN( pStubMsg->Buffer,
  547. SIMPLE_TYPE_ALIGNMENT(Params[n].SimpleType.Type) );
  548. #if defined(__RPC_WIN64__)
  549. // Special case for int3264.
  550. if ( Params[n].SimpleType.Type == FC_INT3264 ||
  551. Params[n].SimpleType.Type == FC_UINT3264 )
  552. {
  553. if ( Params[n].SimpleType.Type == FC_INT3264 )
  554. *((INT64 *)pArg) = *((long * &)pStubMsg->Buffer)++;
  555. else
  556. *((UINT64 *)pArg) = *((ulong * &)pStubMsg->Buffer)++;
  557. continue;
  558. }
  559. #endif
  560. if ( Params[n].SimpleType.Type == FC_ENUM16 )
  561. {
  562. *((int *)(pArg)) = *((int *)pArg) & ((int)0x7fff) ;
  563. }
  564. RpcpMemoryCopy(
  565. pArg,
  566. pStubMsg->Buffer,
  567. (uint)SIMPLE_TYPE_BUFSIZE(Params[n].SimpleType.Type) );
  568. pStubMsg->Buffer +=
  569. SIMPLE_TYPE_BUFSIZE( Params[n].SimpleType.Type );
  570. continue;
  571. }
  572. ppArg = Params[n].ParamAttr.IsByValue ? &pArg : (uchar **)pArg;
  573. pFormatParam = pTypeFormat +
  574. Params[n].TypeOffset;
  575. //
  576. // Transmit/Represent as can be passed as [out] only, thus
  577. // the IsByValue check.
  578. //
  579. (* UnmarshallRoutinesTable[ROUTINE_INDEX(*pFormatParam)] )
  580. ( pStubMsg,
  581. ppArg,
  582. pFormatParam,
  583. FALSE );
  584. }
  585. if ( pStubMsg->pCorrInfo )
  586. NdrCorrelationPass( pStubMsg );
  587. return ;
  588. }
  589. void RPC_ENTRY
  590. NdrpServerUnMarshal ( MIDL_STUB_MESSAGE * pStubMsg )
  591. {
  592. ulong n;
  593. uchar * pArg;
  594. uchar ** ppArg;
  595. PFORMAT_STRING pFormatParam, pFormat;
  596. NDR_PROC_CONTEXT * pContext = ( NDR_PROC_CONTEXT *) pStubMsg->pContext;
  597. PPARAM_DESCRIPTION Params = (PPARAM_DESCRIPTION)pContext->Params;
  598. PFORMAT_STRING pTypeFormat = pContext->DceTypeFormatString;
  599. // we only need to do conversion in NDR32 now: we cut off endian
  600. // conversion in NDR64.
  601. // Do endian/floating point conversions if necessary.
  602. //
  603. if ( ( pStubMsg->RpcMsg->DataRepresentation & 0X0000FFFFUL) !=
  604. NDR_LOCAL_DATA_REPRESENTATION )
  605. {
  606. NdrConvert2( pStubMsg,
  607. (PFORMAT_STRING) Params,
  608. (long) pContext->NumberParams );
  609. }
  610. #ifdef TEST_CONVERT
  611. NdrConvert2( pStubMsg,
  612. (PFORMAT_STRING) Params,
  613. pContext->NumberParams );
  614. #endif
  615. // --------------------------------
  616. // Unmarshall all of our parameters.
  617. // --------------------------------
  618. for ( n = 0; n < pContext->NumberParams; n++ )
  619. {
  620. if ( ! Params[n].ParamAttr.IsIn ||
  621. Params[n].ParamAttr.IsPipe )
  622. continue;
  623. if ( Params[n].ParamAttr.IsPartialIgnore )
  624. {
  625. pArg = pContext->StartofStack + Params[n].StackOffset;
  626. ALIGN( pStubMsg->Buffer, 0x3 );
  627. *(void**)pArg = *(ulong*)pStubMsg->Buffer ? (void*)1 : (void*)0;
  628. pStubMsg->Buffer += PTR_WIRE_SIZE;
  629. continue;
  630. }
  631. pArg = pContext->StartofStack + Params[n].StackOffset;
  632. if ( Params[n].ParamAttr.IsBasetype )
  633. {
  634. //
  635. // Check for a pointer to a basetype. Set the arg pointer
  636. // at the correct buffer location and you're done.
  637. // Except darn int3264
  638. if ( Params[n].ParamAttr.IsSimpleRef )
  639. {
  640. ALIGN( pStubMsg->Buffer,
  641. SIMPLE_TYPE_ALIGNMENT( Params[n].SimpleType.Type ) );
  642. #if defined(__RPC_WIN64__)
  643. // Special case for a ref pointer to int3264.
  644. if ( Params[n].SimpleType.Type == FC_INT3264 ||
  645. Params[n].SimpleType.Type == FC_UINT3264 )
  646. {
  647. *((void **)pArg) = NdrpAlloca( &pContext->AllocateContext, 8 );
  648. if ( Params[n].SimpleType.Type == FC_INT3264 )
  649. *(*(INT64**)pArg) = *((long * &)pStubMsg->Buffer)++;
  650. else
  651. *(*(UINT64**)pArg)= *((ulong * &)pStubMsg->Buffer)++;
  652. continue;
  653. }
  654. #endif
  655. *((uchar **)pArg) = pStubMsg->Buffer;
  656. pStubMsg->Buffer +=
  657. SIMPLE_TYPE_BUFSIZE( Params[n].SimpleType.Type );
  658. }
  659. else
  660. {
  661. NdrUnmarshallBasetypeInline(
  662. pStubMsg,
  663. pArg,
  664. Params[n].SimpleType.Type );
  665. }
  666. continue;
  667. } // IsBasetype
  668. //
  669. // This is an initialization of [in] and [in,out] ref pointers
  670. // to pointers. These can not be initialized to point into the
  671. // rpc buffer and we want to avoid doing a malloc of 4 bytes!
  672. // 32b: a ref pointer to any pointer, we allocate the pointee pointer.
  673. //
  674. if ( Params[n].ParamAttr.ServerAllocSize != 0 )
  675. {
  676. *((void **)pArg) = NdrpAlloca(& pContext->AllocateContext, PTR_MEM_SIZE );
  677. // Triple indirection - cool!
  678. **((void ***)pArg) = 0;
  679. }
  680. pStubMsg->ReuseBuffer = Params[n].ParamAttr.IsForceAllocate;
  681. ppArg = Params[n].ParamAttr.IsByValue ? &pArg : (uchar **)pArg;
  682. pFormatParam = pTypeFormat +
  683. Params[n].TypeOffset;
  684. (*UnmarshallRoutinesTable[ROUTINE_INDEX(*pFormatParam)])
  685. ( pStubMsg,
  686. ppArg,
  687. pFormatParam,
  688. Params[n].ParamAttr.IsForceAllocate &&
  689. !Params[n].ParamAttr.IsByValue );
  690. pStubMsg->ReuseBuffer = FALSE;
  691. }
  692. if ( pStubMsg->pCorrInfo )
  693. NdrCorrelationPass( pStubMsg );
  694. if (CheckVerificationTrailer(pStubMsg->Buffer, pStubMsg->BufferEnd, pStubMsg->RpcMsg) == FALSE)
  695. RpcRaiseException( RPC_S_ACCESS_DENIED );
  696. return ;
  697. }
  698. void RPC_ENTRY
  699. NdrpDcomClientExceptionHandling( MIDL_STUB_MESSAGE * pStubMsg,
  700. ulong ProcNum,
  701. RPC_STATUS ExceptionCode,
  702. CLIENT_CALL_RETURN * pReturnValue )
  703. {
  704. ulong NumberParams;
  705. PPARAM_DESCRIPTION Params;
  706. PFORMAT_STRING pTypeFormat;
  707. ulong n;
  708. uchar * pArg;
  709. PFORMAT_STRING pFormatParam;
  710. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  711. pReturnValue->Simple = NdrProxyErrorHandler(ExceptionCode);
  712. if( pStubMsg->dwStubPhase != PROXY_UNMARSHAL)
  713. return;
  714. NumberParams = pContext->NumberParams;
  715. Params = ( PPARAM_DESCRIPTION )pContext->Params;
  716. // alert: this can't be directly called from ndrclientcall2: we don't have pSyntaxInfo.
  717. pTypeFormat = pContext->DceTypeFormatString;
  718. //
  719. // In OLE, since they don't know about error_status_t and wanted to
  720. // reinvent the wheel, check to see if we need to map the exception.
  721. // In either case, set the return value and then try to free the
  722. // [out] params, if required.
  723. //
  724. pStubMsg->BufferStart = 0;
  725. pStubMsg->BufferEnd = 0;
  726. for ( n = 0; n < pContext->NumberParams; n++ )
  727. {
  728. //
  729. // Skip everything but [out] only parameters. We make
  730. // the basetype check to cover [out] simple ref pointers
  731. // to basetypes.
  732. //
  733. if ( !Params[n].ParamAttr.IsPartialIgnore )
  734. {
  735. if ( Params[n].ParamAttr.IsIn ||
  736. Params[n].ParamAttr.IsReturn ||
  737. Params[n].ParamAttr.IsBasetype ||
  738. Params[n].ParamAttr.IsPipe )
  739. continue;
  740. }
  741. pArg = pContext->StartofStack + Params[n].StackOffset;
  742. pFormatParam = pTypeFormat +
  743. Params[n].TypeOffset;
  744. NdrClearOutParameters( pStubMsg,
  745. pFormatParam,
  746. *((uchar **)pArg) );
  747. }
  748. return ;
  749. }
  750. void RPC_ENTRY
  751. NdrpClientExceptionHandling( MIDL_STUB_MESSAGE * pStubMsg,
  752. ulong ProcNum,
  753. RPC_STATUS ExceptionCode,
  754. CLIENT_CALL_RETURN * pReturnValue )
  755. {
  756. NDR_PROC_CONTEXT * pContext = ( NDR_PROC_CONTEXT * ) pStubMsg->pContext;
  757. NDR_ASSERT( pContext->NdrInfo.InterpreterFlags.HasCommOrFault,
  758. " must have comm or fault to catch" );
  759. NdrClientMapCommFault( pStubMsg,
  760. ProcNum,
  761. ExceptionCode,
  762. (ULONG_PTR*)&pReturnValue->Simple );
  763. return ;
  764. }
  765. void RPC_ENTRY
  766. NdrpAsyncClientExceptionHandling( MIDL_STUB_MESSAGE * pStubMsg,
  767. ulong ProcNum,
  768. RPC_STATUS ExceptionCode,
  769. CLIENT_CALL_RETURN * pReturnValue )
  770. {
  771. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  772. if ( pContext->NdrInfo.InterpreterFlags.HasCommOrFault )
  773. {
  774. NdrClientMapCommFault( pStubMsg,
  775. ProcNum,
  776. ExceptionCode,
  777. (ULONG_PTR*)&pReturnValue->Simple );
  778. if ( ExceptionCode == RPC_S_ASYNC_CALL_PENDING )
  779. {
  780. // If the call is just pending, force the pending error code
  781. // to show up in the return value of RpcAsyncCallComplete.
  782. pReturnValue->Simple = RPC_S_ASYNC_CALL_PENDING;
  783. }
  784. }
  785. else
  786. {
  787. RpcRaiseException(ExceptionCode);
  788. }
  789. return ;
  790. }
  791. void RPC_ENTRY
  792. NdrpClientFinally( PMIDL_STUB_MESSAGE pStubMsg,
  793. void * pThis )
  794. {
  795. NDR_PROC_CONTEXT * pContext = ( NDR_PROC_CONTEXT * ) pStubMsg->pContext;
  796. NdrFullPointerXlatFree(pStubMsg->FullPtrXlatTables);
  797. NdrCorrelationFree( pStubMsg );
  798. //
  799. // Free the RPC buffer.
  800. //
  801. if ( pThis )
  802. {
  803. NdrProxyFreeBuffer( pThis, pStubMsg );
  804. }
  805. else
  806. {
  807. NdrFreeBuffer( pStubMsg );
  808. //
  809. // Unbind if generic handle used. We do this last so that if the
  810. // the user's unbind routine faults, then all of our internal stuff
  811. // will already have been freed.
  812. //
  813. if ( pContext->SavedGenericHandle )
  814. GenericHandleUnbind( pStubMsg->StubDesc,
  815. pContext->StartofStack,
  816. pContext->pHandleFormatSave,
  817. (pContext->HandleType) ? IMPLICIT_MASK : 0,
  818. &pContext->SavedGenericHandle );
  819. }
  820. NdrpAllocaDestroy( & pContext->AllocateContext );
  821. }
  822. void
  823. NdrpServerInit( PMIDL_STUB_MESSAGE pStubMsg,
  824. RPC_MESSAGE * pRpcMsg,
  825. PMIDL_STUB_DESC pStubDesc,
  826. void * pThis,
  827. IRpcChannelBuffer * pChannel,
  828. PNDR_ASYNC_MESSAGE pAsyncMsg )
  829. {
  830. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *) pStubMsg->pContext;
  831. uchar * pArg;
  832. uchar * pArgBuffer = pContext->StartofStack;
  833. if ( pContext->pSyntaxInfo == NULL )
  834. pContext->DceTypeFormatString = pStubDesc->pFormatTypes;
  835. else
  836. pContext->DceTypeFormatString = pContext->pSyntaxInfo->TypeString;
  837. if ( ! pContext->HandleType )
  838. {
  839. //
  840. // For a handle_t parameter we must pass the handle field of
  841. // the RPC message to the server manager.
  842. //
  843. if ( *pContext->pHandleFormatSave == FC_BIND_PRIMITIVE )
  844. {
  845. pArg = pArgBuffer + *((ushort *)&pContext->pHandleFormatSave[2]);
  846. if ( IS_HANDLE_PTR(pContext->pHandleFormatSave[1]) )
  847. pArg = *((uchar **)pArg);
  848. *((handle_t *)pArg) = pRpcMsg->Handle;
  849. }
  850. }
  851. //
  852. // If OLE, put pThis in first dword of stack.
  853. //
  854. if ( pThis )
  855. {
  856. *((void **)pArgBuffer) =
  857. (void *)((CStdStubBuffer *)pThis)->pvServerObject;
  858. }
  859. //
  860. // Initialize the Stub message.
  861. //
  862. if ( ! pChannel )
  863. {
  864. if ( ! pContext->NdrInfo.pProcDesc->Oi2Flags.HasPipes )
  865. {
  866. NdrServerInitializeNew( pRpcMsg,
  867. pStubMsg,
  868. pStubDesc );
  869. }
  870. else
  871. NdrServerInitializePartial( pRpcMsg,
  872. pStubMsg,
  873. pStubDesc,
  874. pContext->NdrInfo.pProcDesc->ClientBufferSize );
  875. }
  876. else
  877. {
  878. // pipe is not supported in obj interface.
  879. NDR_ASSERT( ! pContext->HasPipe, "Pipe is not supported in dcom" );
  880. NdrStubInitialize( pRpcMsg,
  881. pStubMsg,
  882. pStubDesc,
  883. pChannel );
  884. }
  885. if ( pAsyncMsg )
  886. {
  887. pStubMsg->pAsyncMsg = pAsyncMsg;
  888. pRpcMsg->RpcFlags |= RPC_BUFFER_ASYNC;
  889. }
  890. if ( pContext->NdrInfo.InterpreterFlags.FullPtrUsed )
  891. pStubMsg->FullPtrXlatTables = NdrFullPointerXlatInit( 0, XLAT_SERVER );
  892. else
  893. pStubMsg->FullPtrXlatTables = NULL;
  894. //
  895. // Set StackTop AFTER the initialize call, since it zeros the field
  896. // out.
  897. //
  898. pStubMsg->StackTop = pArgBuffer;
  899. if ( pContext->NdrInfo.pProcDesc->Oi2Flags.HasPipes )
  900. {
  901. NdrpPipesInitialize32( pStubMsg,
  902. &pContext->AllocateContext,
  903. (PFORMAT_STRING) pContext->Params,
  904. (char*)pArgBuffer,
  905. pContext->NumberParams );
  906. }
  907. //
  908. // We must make this check AFTER the call to ServerInitialize,
  909. // since that routine puts the stub descriptor alloc/dealloc routines
  910. // into the stub message.
  911. //
  912. if ( pContext->NdrInfo.InterpreterFlags.RpcSsAllocUsed )
  913. NdrRpcSsEnableAllocate( pStubMsg );
  914. if ( pContext->NdrInfo.pProcDesc->Oi2Flags.HasExtensions )
  915. {
  916. pStubMsg->fHasExtensions = 1;
  917. pStubMsg->fHasNewCorrDesc = pContext->NdrInfo.pProcDesc->NdrExts.Flags2.HasNewCorrDesc;
  918. if ( pContext->NdrInfo.pProcDesc->NdrExts.Flags2.ServerCorrCheck )
  919. {
  920. void * pCorr = NdrpAlloca( &pContext->AllocateContext, NDR_DEFAULT_CORR_CACHE_SIZE );
  921. NdrCorrelationInitialize( pStubMsg,
  922. pCorr,
  923. NDR_DEFAULT_CORR_CACHE_SIZE,
  924. 0 /* flags */ );
  925. }
  926. }
  927. }
  928. void NdrpServerOutInit( PMIDL_STUB_MESSAGE pStubMsg )
  929. {
  930. ulong n;
  931. NDR_PROC_CONTEXT * pContext = (NDR_PROC_CONTEXT *)pStubMsg->pContext;
  932. PPARAM_DESCRIPTION Params = ( PPARAM_DESCRIPTION ) pContext->Params;
  933. PFORMAT_STRING pFormatTypes = pContext->DceTypeFormatString;
  934. uchar * pArgBuffer = pContext->StartofStack;
  935. uchar * pArg;
  936. PFORMAT_STRING pFormatParam;
  937. for ( n = 0; n < pContext->NumberParams; n++ )
  938. {
  939. if ( !Params[n].ParamAttr.IsPartialIgnore )
  940. {
  941. if ( Params[n].ParamAttr.IsIn ||
  942. ( Params[n].ParamAttr.IsReturn && !pContext->HasComplexReturn ) ||
  943. Params[n].ParamAttr.IsPipe )
  944. continue;
  945. pArg = pArgBuffer + Params[n].StackOffset;
  946. }
  947. else
  948. {
  949. pArg = pArgBuffer + Params[n].StackOffset;
  950. if ( !*(void**)pArg )
  951. continue;
  952. }
  953. //
  954. // Check if we can initialize this parameter using some of our
  955. // stack.
  956. //
  957. if ( Params[n].ParamAttr.ServerAllocSize != 0 )
  958. {
  959. *((void **)pArg) = NdrpAlloca( &pContext->AllocateContext,
  960. Params[n].ParamAttr.ServerAllocSize * 8);
  961. MIDL_memset( *((void **)pArg),
  962. 0,
  963. Params[n].ParamAttr.ServerAllocSize * 8 );
  964. continue;
  965. }
  966. else if ( Params[n].ParamAttr.IsBasetype )
  967. {
  968. *((void **)pArg) = NdrpAlloca( &pContext->AllocateContext, 8 );
  969. MIDL_memset( *((void **)pArg), 0, 8 );
  970. continue;
  971. };
  972. pFormatParam = pFormatTypes + Params[n].TypeOffset;
  973. NdrOutInit( pStubMsg,
  974. pFormatParam,
  975. (uchar **)pArg );
  976. }
  977. }
  978. #if defined( BUILD_NDR64 )
  979. BOOL IsServerSupportNDR64( MIDL_SERVER_INFO *pServerInfo )
  980. {
  981. if ( ( pServerInfo->pStubDesc->Version > NDR_VERSION ) ||
  982. ( pServerInfo->pStubDesc->Version < NDR_VERSION_6_0 ) ||
  983. ( pServerInfo->pStubDesc->MIDLVersion < MIDL_VERSION_6_0_322 ) ||
  984. ! ( pServerInfo->pStubDesc->mFlags & RPCFLG_HAS_MULTI_SYNTAXES ) )
  985. return FALSE;
  986. return TRUE;
  987. }
  988. #endif
  989. RPC_STATUS RPC_ENTRY
  990. NdrClientGetSupportedSyntaxes(
  991. IN RPC_CLIENT_INTERFACE * pInf,
  992. OUT ulong * pCount,
  993. MIDL_SYNTAX_INFO ** pArr )
  994. {
  995. MIDL_SYNTAX_INFO *pSyntaxInfo;
  996. NDR_ASSERT( pInf->Flags & RPCFLG_HAS_MULTI_SYNTAXES, "invalid clientif" );
  997. if ( pInf->Flags & RPCFLG_HAS_CALLBACK )
  998. {
  999. // interpreter info is MIDL_SERVER_INFO
  1000. MIDL_SERVER_INFO * pServerInfo = ( MIDL_SERVER_INFO *) pInf->InterpreterInfo;
  1001. *pCount = ( ulong ) pServerInfo->nCount ;
  1002. *pArr = pServerInfo->pSyntaxInfo;
  1003. }
  1004. else
  1005. {
  1006. MIDL_STUBLESS_PROXY_INFO * pProxyInfo = ( MIDL_STUBLESS_PROXY_INFO *) pInf->InterpreterInfo;
  1007. *pCount = ( ulong ) pProxyInfo->nCount ;
  1008. *pArr = pProxyInfo->pSyntaxInfo;
  1009. }
  1010. return RPC_S_OK;
  1011. }
  1012. RPC_STATUS RPC_ENTRY
  1013. NdrServerGetSupportedSyntaxes(
  1014. IN RPC_SERVER_INTERFACE * pInf,
  1015. OUT ulong * pCount,
  1016. MIDL_SYNTAX_INFO ** pArr,
  1017. OUT ulong * pPrefer)
  1018. {
  1019. NDR_ASSERT( pInf->Flags & RPCFLG_HAS_MULTI_SYNTAXES,"invalid serverif" );
  1020. MIDL_SERVER_INFO *pServerInfo = ( MIDL_SERVER_INFO *) pInf->InterpreterInfo;
  1021. *pCount = ( ulong ) pServerInfo->nCount;
  1022. *pArr = pServerInfo->pSyntaxInfo;
  1023. NdrpGetPreferredSyntax( ( ulong )pServerInfo->nCount, pServerInfo->pSyntaxInfo, pPrefer );
  1024. return RPC_S_OK;
  1025. }
  1026. RPC_STATUS RPC_ENTRY
  1027. NdrCreateServerInterfaceFromStub(
  1028. IN IRpcStubBuffer* pStub,
  1029. IN OUT RPC_SERVER_INTERFACE *pServerIf )
  1030. {
  1031. #if !defined( BUILD_NDR64 )
  1032. return S_OK;
  1033. #else
  1034. CInterfaceStubVtbl * pStubVTable;
  1035. PMIDL_SERVER_INFO pServerInfo;
  1036. IRpcStubBuffer * pv = NULL;
  1037. RpcTryExcept
  1038. {
  1039. // filter out non-ndr stub first.
  1040. if ( S_OK != pStub->lpVtbl->QueryInterface(pStub, IID_IPrivStubBuffer, (void **)& pv ) )
  1041. return S_OK;
  1042. pv->lpVtbl->Release(pv);
  1043. pStubVTable = (CInterfaceStubVtbl *)
  1044. (*((uchar **)pStub) - sizeof(CInterfaceStubHeader));
  1045. pServerInfo = (PMIDL_SERVER_INFO) pStubVTable->header.pServerInfo;
  1046. // In /Os mode, we don't have pServerInfo.
  1047. if ( pServerInfo &&
  1048. IsServerSupportNDR64( pServerInfo ) )
  1049. {
  1050. memcpy ( &pServerIf->TransferSyntax, pServerInfo->pTransferSyntax, sizeof( RPC_SYNTAX_IDENTIFIER ) );
  1051. pServerIf->Flags |= RPCFLG_HAS_MULTI_SYNTAXES;
  1052. pServerIf->InterpreterInfo = pServerInfo;
  1053. }
  1054. }
  1055. RpcExcept( 1 )
  1056. {
  1057. }
  1058. RpcEndExcept
  1059. #endif
  1060. return S_OK;
  1061. }
  1062. /*++
  1063. Routine Description :
  1064. Read the proc header for different transfer syntax.
  1065. We need to return proc number in dce because for stubs compiled with
  1066. DCE only, proc header is the only place to get the procnum.
  1067. This rountine being called from both client and server. The difference
  1068. is that client side we are reading from the default one; server side we
  1069. are reading from the selected one.
  1070. Arguments :
  1071. Return :
  1072. none. Raise exception if something goes wrong. We can't recovered
  1073. from here because we don't have enough information about how the
  1074. stub looks like if we don't have valid proc header.
  1075. --*/
  1076. ulong RPC_ENTRY
  1077. MulNdrpInitializeContextFromProc (
  1078. SYNTAX_TYPE SyntaxType,
  1079. PFORMAT_STRING pFormat,
  1080. NDR_PROC_CONTEXT * pContext,
  1081. uchar * StartofStack,
  1082. BOOLEAN IsReset )
  1083. {
  1084. ulong RpcFlags;
  1085. ulong ProcNum = 0;
  1086. if ( !IsReset )
  1087. NdrpInitializeProcContext( pContext );
  1088. pContext->pProcFormat = pFormat;
  1089. pContext->StartofStack = StartofStack;
  1090. if ( SyntaxType == XFER_SYNTAX_DCE )
  1091. {
  1092. PPARAM_DESCRIPTION Params;
  1093. INTERPRETER_FLAGS InterpreterFlags;
  1094. PFORMAT_STRING pNewProcDescr;
  1095. INTERPRETER_OPT_FLAGS OptFlags;
  1096. pContext->CurrentSyntaxType = XFER_SYNTAX_DCE;
  1097. pContext->HandleType = *pFormat++ ;
  1098. pContext->UseLocator = (FC_AUTO_HANDLE == pContext->HandleType);
  1099. pContext->NdrInfo.InterpreterFlags = *((PINTERPRETER_FLAGS)pFormat++);
  1100. InterpreterFlags = pContext->NdrInfo.InterpreterFlags;
  1101. if ( InterpreterFlags.HasRpcFlags )
  1102. RpcFlags = *(( UNALIGNED ulong* &)pFormat)++;
  1103. else
  1104. RpcFlags = 0;
  1105. ProcNum = *(ushort *)pFormat; pFormat += 2;
  1106. pContext->StackSize = *(ushort *)pFormat; pFormat += 2;
  1107. pContext->pHandleFormatSave = pFormat;
  1108. pNewProcDescr = pFormat;
  1109. if ( ! pContext->HandleType )
  1110. {
  1111. // explicit handle
  1112. pNewProcDescr += ((*pFormat == FC_BIND_PRIMITIVE) ? 4 : 6);
  1113. }
  1114. pContext->NdrInfo.pProcDesc = (NDR_PROC_DESC *)pNewProcDescr;
  1115. OptFlags = ( (NDR_PROC_DESC *)pNewProcDescr )->Oi2Flags;
  1116. pContext->NumberParams = pContext->NdrInfo.pProcDesc->NumberParams;
  1117. //
  1118. // Parameter descriptions are nicely spit out by MIDL.
  1119. // If there is no extension, Params is in the position of extensions.
  1120. //
  1121. Params = (PPARAM_DESCRIPTION) &( pContext->NdrInfo.pProcDesc->NdrExts );
  1122. // Proc header extentions, from NDR ver. 5.2.
  1123. // Params must be set correctly here because of exceptions.
  1124. if ( OptFlags.HasExtensions )
  1125. {
  1126. pContext->HasComplexReturn = pContext->NdrInfo.pProcDesc->NdrExts.Flags2.HasComplexReturn;
  1127. Params = (PPARAM_DESCRIPTION)((uchar*)Params + pContext->NdrInfo.pProcDesc->NdrExts.Size);
  1128. #if defined(_WIN64)
  1129. PNDR_PROC_HEADER_EXTS64 pExts = (PNDR_PROC_HEADER_EXTS64 )&pContext->NdrInfo.pProcDesc->NdrExts;
  1130. pContext->FloatDoubleMask = pExts->FloatArgMask;
  1131. #endif
  1132. }
  1133. pContext->Params = Params;
  1134. pContext->IsAsync = OptFlags.HasAsyncUuid ;
  1135. pContext->IsObject = InterpreterFlags.ObjectProc;
  1136. pContext->HasPipe = OptFlags.HasPipes;
  1137. pContext->ExceptionFlag = ! ( InterpreterFlags.IgnoreObjectException ) &&
  1138. ( pContext->IsObject || InterpreterFlags.HasCommOrFault );
  1139. } // XFER_SYNTAX_DCE
  1140. #if defined(BUILD_NDR64)
  1141. else if ( SyntaxType == XFER_SYNTAX_NDR64 )
  1142. {
  1143. NDR64_PROC_FLAGS * pProcFlags;
  1144. pContext->CurrentSyntaxType = XFER_SYNTAX_NDR64;
  1145. pContext->Ndr64Header = (NDR64_PROC_FORMAT *)pFormat;
  1146. pContext->HandleType =
  1147. NDR64MAPHANDLETYPE( NDR64GETHANDLETYPE( &pContext->Ndr64Header->Flags ) );
  1148. pContext->UseLocator = (FC64_AUTO_HANDLE == pContext->HandleType);
  1149. RpcFlags = pContext->Ndr64Header->RpcFlags;
  1150. #if defined(_WIN64)
  1151. pContext->FloatDoubleMask = pContext->Ndr64Header->FloatDoubleMask;
  1152. #endif
  1153. pContext->NumberParams = pContext->Ndr64Header->NumberOfParams;
  1154. pContext->Params = (NDR64_PROC_FORMAT *)( (char *) pFormat + sizeof( NDR64_PROC_FORMAT ) + pContext->Ndr64Header->ExtensionSize );
  1155. pContext->StackSize = pContext->Ndr64Header->StackSize;
  1156. pProcFlags = (NDR64_PROC_FLAGS *) &pContext->Ndr64Header->Flags;
  1157. pContext->HasComplexReturn = pProcFlags->HasComplexReturn;
  1158. pContext->IsAsync = pProcFlags->IsAsync;
  1159. pContext->IsObject = pProcFlags->IsObject;
  1160. pContext->HasPipe = pProcFlags->UsesPipes;
  1161. pContext->ExceptionFlag = pContext->IsObject || pProcFlags->HandlesExceptions;
  1162. } // XFER_SYNTAX_NDR64
  1163. #endif
  1164. else
  1165. {
  1166. RpcRaiseException( RPC_S_INTERNAL_ERROR );
  1167. NDR_ASSERT( 0, "Invalid transfer syntax.");
  1168. }
  1169. // setup the pipe flag before negotiation.
  1170. if ( pContext->HasPipe )
  1171. {
  1172. RpcFlags &= ~RPC_BUFFER_COMPLETE;
  1173. RpcFlags |= RPC_BUFFER_PARTIAL;
  1174. }
  1175. pContext->RpcFlags = RpcFlags;
  1176. // We need to cleanup the resend flag during initialization in preparation
  1177. // for retry later.
  1178. pContext->NeedsResend = FALSE;
  1179. return ProcNum;
  1180. }