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.

365 lines
10 KiB

  1. /*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
  2. Copyright (c) 1989-2000 Microsoft Corporation
  3. Module Name:
  4. ndrcls.cxx
  5. Abstract:
  6. Routines for the ndrcls code generation class.
  7. Notes:
  8. History:
  9. Aug-31-1993 VibhasC Created.
  10. ----------------------------------------------------------------------------*/
  11. /****************************************************************************
  12. * include files
  13. ***************************************************************************/
  14. #include "becls.hxx"
  15. #pragma hdrstop
  16. /****************************************************************************
  17. * local definitions
  18. ***************************************************************************/
  19. /****************************************************************************
  20. * local data
  21. ***************************************************************************/
  22. /****************************************************************************
  23. * externs
  24. ***************************************************************************/
  25. /****************************************************************************/
  26. expr_node *
  27. CG_NDR::PresentedLengthExpression( CCB* )
  28. {
  29. expr_sizeof * pExpr = new expr_sizeof( GetType() );
  30. return pExpr;
  31. }
  32. expr_node *
  33. CG_NDR::PresentedSizeExpression( CCB* )
  34. {
  35. expr_sizeof * pExpr = new expr_sizeof( GetType() );
  36. return pExpr;
  37. }
  38. expr_node *
  39. CG_NDR::PresentedFirstExpression( CCB* )
  40. {
  41. return new expr_constant(0L);
  42. }
  43. /****************************************************************************
  44. utility functions
  45. ****************************************************************************/
  46. CG_STATUS
  47. CG_NDR::SizeAnalysis( ANALYSIS_INFO * pAna )
  48. {
  49. pAna;
  50. return CG_OK;
  51. }
  52. CG_STATUS
  53. CG_NDR::MarshallAnalysis( ANALYSIS_INFO * pAna )
  54. {
  55. pAna;
  56. return CG_OK;
  57. }
  58. node_skl *
  59. CG_NDR::GetBasicType()
  60. {
  61. node_skl * pT = GetType();
  62. switch (pT->NodeKind())
  63. {
  64. case NODE_ID:
  65. case NODE_FIELD:
  66. case NODE_PARAM:
  67. case NODE_DEF:
  68. return pT->GetBasicType();
  69. }
  70. return pT;
  71. }
  72. CG_STATUS
  73. CG_NDR::RefCheckAnalysis(
  74. ANALYSIS_INFO * pAna )
  75. {
  76. CG_NDR * pC = (CG_NDR *)GetChild();
  77. if( pC )
  78. pC->RefCheckAnalysis( pAna );
  79. return CG_OK;
  80. }
  81. CG_STATUS
  82. CG_NDR::InLocalAnalysis(
  83. ANALYSIS_INFO * pAna )
  84. {
  85. CG_NDR * pC = (CG_NDR *)GetChild();
  86. if( pC )
  87. pC->InLocalAnalysis( pAna );
  88. return CG_OK;
  89. }
  90. CG_STATUS
  91. CG_NDR::S_GenInitInLocals(
  92. CCB * pCCB )
  93. {
  94. CG_NDR * pC = (CG_NDR *)GetChild();
  95. if( pC )
  96. pC->S_GenInitInLocals( pCCB );
  97. return CG_OK;
  98. }
  99. CG_STATUS
  100. CG_NDR::GenRefChecks(
  101. CCB * pCCB )
  102. {
  103. CG_NDR * pC = (CG_NDR *)GetChild();
  104. if( pC )
  105. pC->GenRefChecks( pCCB );
  106. return CG_OK;
  107. }
  108. extern CMD_ARG * pCommand;
  109. void
  110. CG_NDR::GetNdrParamAttributes( CCB *pCCB, PARAM_ATTRIBUTES *attributes)
  111. {
  112. CG_PARAM * pParam;
  113. CG_NDR * pChild;
  114. CG_PROC * pProc;
  115. pChild = (CG_NDR *) GetChild();
  116. if ( pChild && (pChild->GetCGID() == ID_CG_GENERIC_HDL) )
  117. pChild = (CG_NDR *) pChild->GetChild();
  118. pParam = (CG_PARAM *) pCCB->GetLastPlaceholderClass();
  119. pProc = (CG_PROC *) pCCB->GetCGNodeContext();
  120. if ( IsPipeOrPipeReference() )
  121. {
  122. // Pipe parameters are never sized.
  123. attributes->MustSize = 0;
  124. }
  125. else
  126. attributes->MustSize = (unsigned short) pParam->GetInterpreterMustSize();
  127. attributes->IsIn = (unsigned short) pParam->IsParamIn();
  128. attributes->IsOut = (unsigned short) pParam->IsParamOut();
  129. attributes->IsReturn = ( pParam->GetCGID() == ID_CG_RETURN )
  130. || ( pProc->HasComplexReturnType()
  131. && NULL == pParam->GetSibling() );
  132. attributes->IsPartialIgnore = pParam->IsParamPartialIgnore();
  133. attributes->IsForceAllocate = pParam->IsForceAllocate();
  134. attributes->IsBasetype = 0;
  135. // SAFEARRAY(FOO) is being generated as ID_CG_SAFEARRAY. It's in fact user
  136. // marshal on the wire.
  137. attributes->IsByValue =
  138. IsStruct() ||
  139. IsUnion() ||
  140. ( GetCGID() == ID_CG_TRANSMIT_AS ) ||
  141. ( GetCGID() == ID_CG_REPRESENT_AS ) ||
  142. ( GetCGID() == ID_CG_USER_MARSHAL ) ||
  143. ( GetCGID() == ID_CG_SAFEARRAY ) ||
  144. ( GetCGID() == ID_CG_CS_TAG ) ;
  145. attributes->IsSimpleRef =
  146. (pParam->GetCGID() != ID_CG_RETURN) &&
  147. IsPointer() &&
  148. ((CG_POINTER *)this)->IsBasicRefPointer();
  149. attributes->IsDontCallFreeInst = (unsigned short) pParam->GetDontCallFreeInst();
  150. attributes->IsPipe = (unsigned short) IsPipeOrPipeReference();
  151. attributes->SaveForAsyncFinish = (unsigned short) pParam->IsSaveForAsyncFinish();
  152. if ( (attributes->IsPipe) || (GetCGID() == ID_CG_CONTEXT_HDL) ||
  153. (IsPointer() && pChild && (pChild->GetCGID() == ID_CG_CONTEXT_HDL)))
  154. attributes->MustFree = 0;
  155. else
  156. attributes->MustFree = 1;
  157. attributes->ServerAllocSize = 0;
  158. if ( GetCGID() == ID_CG_PTR )
  159. {
  160. long Size = 0;
  161. (void)
  162. ((CG_POINTER *)this)->InterpreterAllocatesOnStack( pCCB,
  163. pParam,
  164. &Size );
  165. attributes->ServerAllocSize = unsigned short(Size / 8);
  166. }
  167. //
  168. // Now make a final check for a simple ref pointer to a basetype and for
  169. // a ref pointer to pointer.
  170. //
  171. // We mark pointers to basetypes as being both a basetype and a simple
  172. // ref pointer. Kind of an anomoly, but it works out well in the
  173. // interpreter.
  174. //
  175. // For both of these cases the pointer must have no allocate attributes
  176. // and can not be a return value.
  177. //
  178. if (
  179. IsPointer() &&
  180. (((CG_POINTER *)this)->GetPtrType() == PTR_REF) &&
  181. ! IS_ALLOCATE( ((CG_POINTER *)this)->GetAllocateDetails(),
  182. ALLOCATE_ALL_NODES ) &&
  183. ! IS_ALLOCATE( ((CG_POINTER *)this)->GetAllocateDetails(),
  184. ALLOCATE_DONT_FREE ) &&
  185. (pParam->GetCGID() != ID_CG_RETURN)
  186. )
  187. {
  188. //
  189. // Now make sure it's a pointer to basetype and that it is either
  190. // [in] or [in,out] (in which case we use the buffer to hold it) or
  191. // that it is [out] and there is room for it on the interpreter's
  192. // stack. We don't mark enum16s as SimpleRef-Basetype because of
  193. // their special handling (i.e. wire size != memory size).
  194. //
  195. if ( ((CG_POINTER *)this)->IsPointerToBaseType() &&
  196. ( pParam->IsParamIn() ||
  197. (((CG_POINTER *)this)->GetFormatAttr() & FC_ALLOCED_ON_STACK) ) )
  198. {
  199. if ( ((CG_BASETYPE *)pChild)->GetFormatChar() != FC_ENUM16 )
  200. {
  201. attributes->IsBasetype = 1;
  202. attributes->IsSimpleRef = 1;
  203. }
  204. attributes->MustFree = 0;
  205. }
  206. }
  207. }
  208. void
  209. CG_NDR::GenNdrParamDescription( CCB * pCCB )
  210. {
  211. FORMAT_STRING * pProcFormatString;
  212. PARAM_ATTRIBUTES Attributes;
  213. long FormatStringOffset;
  214. CG_PARAM * pParam;
  215. CG_NDR * pChild;
  216. pChild = (CG_NDR *) GetChild();
  217. if ( pChild && (pChild->GetCGID() == ID_CG_GENERIC_HDL) )
  218. pChild = (CG_NDR *) pChild->GetChild();
  219. pParam = (CG_PARAM *) pCCB->GetLastPlaceholderClass();
  220. pProcFormatString = pCCB->GetProcFormatString();
  221. // Attributes.
  222. GetNdrParamAttributes( pCCB, &Attributes );
  223. pProcFormatString->PushParamFlagsShort( *((short *)&Attributes) );
  224. // Stack offset as number of ints.
  225. pProcFormatString->PushUShortStackOffsetOrSize(
  226. pParam->GetStackOffset( pCCB, I386_STACK_SIZING ) );
  227. //
  228. // If we found a pointer to a basetype, make sure and emit the basetype's
  229. // param format.
  230. //
  231. if ( Attributes.IsSimpleRef && Attributes.IsBasetype )
  232. {
  233. pProcFormatString->PushFormatChar(
  234. ((CG_BASETYPE *)pChild)->GetFormatChar() );
  235. pProcFormatString->PushByte( 0 );
  236. return;
  237. }
  238. if ( Attributes.IsSimpleRef )
  239. {
  240. CG_POINTER * pPointer;
  241. pPointer = (CG_POINTER *) this;
  242. switch ( pPointer->GetCGID() )
  243. {
  244. case ID_CG_STRING_PTR :
  245. if ( ((CG_STRING_POINTER *)pPointer)->IsStringableStruct() )
  246. FormatStringOffset = GetFormatStringOffset();
  247. else
  248. FormatStringOffset = pPointer->GetFormatStringOffset() + 2;
  249. break;
  250. case ID_CG_SIZE_STRING_PTR :
  251. if ( ((CG_STRING_POINTER *)pPointer)->IsStringableStruct() )
  252. FormatStringOffset = GetFormatStringOffset();
  253. else
  254. FormatStringOffset = pPointer->GetPointeeFormatStringOffset();
  255. break;
  256. case ID_CG_STRUCT_STRING_PTR :
  257. FormatStringOffset = GetFormatStringOffset();
  258. break;
  259. default :
  260. FormatStringOffset = pPointer->GetPointeeFormatStringOffset();
  261. break;
  262. }
  263. }
  264. else
  265. {
  266. FormatStringOffset = GetFormatStringOffset();
  267. }
  268. //
  269. // Push the offset in the type format string to the param's description.
  270. //
  271. pProcFormatString->PushShortTypeOffset( FormatStringOffset );
  272. }
  273. void
  274. CG_NDR::GenNdrParamDescriptionOld( CCB * pCCB )
  275. {
  276. FORMAT_STRING * pProcFormatString;
  277. CG_PARAM * pParam;
  278. long StackSize;
  279. long StackElem;
  280. pProcFormatString = pCCB->GetProcFormatString();
  281. pParam = (CG_PARAM *) pCCB->GetLastPlaceholderClass();
  282. if ( pCommand->Is64BitEnv() )
  283. StackElem = 8;
  284. else
  285. StackElem = 4;
  286. StackSize = pParam->GetStackSize();
  287. StackSize = (StackSize + StackElem - 1) & ~ (StackElem - 1);
  288. pProcFormatString->PushSmallStackSize( (char) (StackSize / StackElem) );
  289. //
  290. // Push the offset in the type format string to the param's description.
  291. //
  292. pProcFormatString->PushShortTypeOffset( GetFormatStringOffset() );
  293. }