windows-nt/Source/XPSP1/NT/com/rpc/midl/codegen/ndr64.cxx
2020-09-26 16:20:57 +08:00

3592 lines
114 KiB
C++

/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Copyright (c) 1999-2000 Microsoft Corporation
Module Name:
ndr64.cxx
Abstract:
Routines for the ndr64 transfer syntax.
Notes:
History:
----------------------------------------------------------------------------*/
#include "becls.hxx"
char * _SimpleTypeName[] = {
"",
"NDR64_FORMAT_UINT8",
"NDR64_FORMAT_UINT16",
"NDR64_FORMAT_UINT32",
"NDR64_FORMAT_UINT64"
};
// define the name table for the NDR64 format characters
#define NDR64_BEGIN_TABLE \
const char *pNDR64FormatCharNames[] = {
#define NDR64_TABLE_END \
};
#define NDR64_ZERO_ENTRY \
"FC64_ZERO"
#define NDR64_TABLE_ENTRY( number, tokenname, marshal, embeddedmarshall, unmarshall, embeddedunmarshal, buffersize, embeddedbuffersize, memsize, embeddedmemsize, free, embeddedfree, typeflags ) \
, #tokenname
#define NDR64_SIMPLE_TYPE_TABLE_ENTRY( number, tokenname, simpletypebuffersize, simpletypememorysize ) \
, #tokenname
#define NDR64_UNUSED_TABLE_ENTRY( number, tokenname ) \
, #tokenname
#define NDR64_UNUSED_TABLE_ENTRY_NOSYM( number ) \
, NULL
#include "tokntbl.h"
extern const pNDR64FormatCharNamesSize = (sizeof(pNDR64FormatCharNames) / sizeof(*pNDR64FormatCharNames));
C_ASSERT( (sizeof(pNDR64FormatCharNames) / sizeof(*pNDR64FormatCharNames)) == 256 );
#undef NDR64_BEGIN_TABLE
#undef NDR64_TABLE_END
#undef NDR64_ZERO_ENTRY
#undef NDR64_TABLE_ENTRY
#undef NDR64_SIMPLE_TYPE_TABLE_ENTRY
#undef NDR64_UNUSED_TABLE_ENTRY
#undef NDR64_UNUSED_TABLE_ENTRY_NOSYM
FormatFragment * GenExprFormatString(CCB *pCCB,
expr_node *pSizeExpr,
CompositeFormatFragment *FragmentList,
BOOL * IsEarly,
ulong * pExprLength );
//+--------------------------------------------------------------------------
//
// Method: FormatFragment::OutputDescription
//
// Synopsis: Output a description of this fragment
//
// Notes: The output is in the form of a C style comment.
// The decription is the name of the type that this fragment
// represents. If this fragment represents more than on type
// because of optimization then all the types will be output.
//
//---------------------------------------------------------------------------
void FormatFragment::OutputDescription( ISTREAM *stream )
{
FormatFragment *frag = this;
bool first = true;
stream->WriteOnNewLine( "/* " );
do
{
if ( !first )
stream->Write(", ");
CG_CLASS *pClass = frag->GetCGNode();
const char *pName = NULL;
while ( NULL != pClass && pClass->IsPointer() )
{
stream->Write( "*" );
pClass = pClass->GetChild();
}
if ( NULL != pClass && NULL != pClass->GetType() )
{
pName = pClass->GetName();
if ( pName != NULL && pName[0] == '\0' )
pName = NULL;
}
if ( NULL == pName )
pName = frag->GetTypeName();
stream->Write( pName );
frag = frag->pNextOptimized;
first = false;
}
while ( NULL != frag );
stream->Write( " */" );
}
//+--------------------------------------------------------------------------
//
// Method: CompositeFormatFragment::IsEqualTo
//
// Synopsis: Compare two composite fragments
//
// Notes: This method compares two composite fragments. Two composite
// fragments are equal if and only if the elements of the composite
// are equal.
//
//---------------------------------------------------------------------------
bool CompositeFormatFragment::IsEqualTo( FormatFragment *candidate )
{
CompositeFormatFragment *pOther = (CompositeFormatFragment *)candidate;
FormatFragment *pCurrent = pHead;
FormatFragment *pCurrentOther = pOther->pHead;
while ( (pCurrent != NULL) &&
(pCurrentOther != NULL) )
{
const type_info &frag_type = typeid( *pCurrent );
const type_info &other_type = typeid( *pCurrentOther );
if ( frag_type != other_type )
return false;
if ( !pCurrent->IsEqualTo( pCurrentOther ) )
return false;
pCurrent = pCurrent->Next;
pCurrentOther = pCurrentOther->Next;
}
// Both lists must have the same length.
bool bResult = ( NULL == pCurrent ) && ( NULL == pCurrentOther );
return bResult;
}
//+--------------------------------------------------------------------------
//
// Class: CompositeFormatFragment::LookupFragment
//
// Synopsis: Find the fragment corresponding to a class
//
// Parameters: [pClass] -- The class
//
// Returns: NULL if the class has no fragment yet
//
//---------------------------------------------------------------------------
FormatFragment * CompositeFormatFragment::LookupFragment( CG_CLASS *pClass )
{
FormatFragment *frag;
// Generic handles are a special case. They don't actually have any
// representation in the format info.
if ( ID_CG_GENERIC_HDL == pClass->GetCGID() )
pClass = pClass->GetChild();
for ( frag = pHead; NULL != frag; frag = frag->Next )
if ( frag->pClass == pClass )
return frag;
return NULL;
}
//+--------------------------------------------------------------------------
//
// Class: CompositeFormatFragment::AddFragment
//
// Synopsis: Add a fragment to the list if it isn't already present
//
// Parameters: [frag] -- The fragment
//
// Returns: The id of the fragment
//
// Notes: Another way to think of this method is that it provides a
// mapping between CG_CLASS pointers and FormatInfoRef's.
//
//---------------------------------------------------------------------------
FormatInfoRef CompositeFormatFragment::AddFragment( FormatFragment *frag )
{
frag->Next = NULL;
frag->RefID = NextRefID;
frag->Prev = pTail;
frag->Parent = this;
if ( NULL == pTail )
pHead = frag;
else
pTail->Next = frag;
pTail = frag;
NextRefID = (FormatInfoRef) ((size_t) NextRefID + 1);
return frag->RefID;
}
//+--------------------------------------------------------------------------
//
// Class: CompositeFormatFragment::OutputFragmentType
//
// Synopsis: Output a type declaration for this composite
//
// Notes: It looks like:
//
// struct <name>
// {
// <frag1 type>,
// <frag2 type>
// ...
// }
//
//---------------------------------------------------------------------------
void CompositeFormatFragment::OutputFragmentType( CCB *pCCB )
{
ISTREAM *stream = pCCB->GetStream();
stream->WriteOnNewLine("struct ");
stream->Write( GetTypeName() );
stream->WriteOnNewLine("{");
stream->IndentInc();
FormatFragment *pCurrent = pHead;
for ( pCurrent = pHead; NULL != pCurrent; pCurrent = pCurrent->Next )
{
if ( pCurrent->WasOptimizedOut() )
{
// Only top-level stuff should be optimized
MIDL_ASSERT( NULL == GetParent() );
continue;
}
pCurrent->OutputFragmentType( pCCB );
stream->WriteFormat(" frag%d;", (size_t) pCurrent->GetRefID());
}
stream->IndentDec();
stream->WriteOnNewLine("}");
}
//+--------------------------------------------------------------------------
//
// Class: CompositeFormatFragment::OutputFragmentData
//
// Synopsis: Output the initializer data for this composite
//
// Notes: It looks like:
//
// {
// <frag1 data>,
// <frag2 type>
// ...
// }
//
//---------------------------------------------------------------------------
void CompositeFormatFragment::OutputFragmentData( CCB *pCCB )
{
ISTREAM *stream = pCCB->GetStream();
OutputStructDataStart( pCCB );
FormatFragment *pCurrent;
bool FirstFragment = true;
for ( pCurrent = pHead; NULL != pCurrent; pCurrent = pCurrent->Next )
{
if ( pCurrent->WasOptimizedOut() )
{
// Only top-level stuff should be optimized
MIDL_ASSERT( NULL == GetParent() );
continue;
}
if ( !FirstFragment )
stream->Write(",");
FirstFragment = false;
pCurrent->OutputFragmentData( pCCB );
}
OutputStructDataEnd( pCCB );
}
//+--------------------------------------------------------------------------
//
// Class: CompositeFormatFragment::OptimizeFragment
//
// Synopsis: Try to optimize out a fragment in the format string by
// checking to see if it is the same as some other fragment.
//
// Parameters: [frag] -- The fragment to optimize
//
// Returns: The final ID of the fragment
//
// Notes: Optimization can be suppressed by the user by specifiying
// -no_format_opt
//
//---------------------------------------------------------------------------
FormatInfoRef CompositeFormatFragment::OptimizeFragment( FormatFragment *frag )
{
if ( pCommand->IsSwitchDefined( SWITCH_NO_FMT_OPT ) )
return frag->RefID;
// Only stuff in the root composite can be optimized
if ( ! dynamic_cast<RootFormatFragment *>(frag->GetParent()) )
return frag->RefID;
FormatFragment *candidate;
for ( candidate = pHead;
NULL != candidate && candidate != frag;
candidate = candidate->Next )
{
const type_info &frag_type = typeid( *frag );
const type_info &candidate_type = typeid( *candidate );
if ( frag_type != candidate_type )
continue;
if ( candidate->IsEqualTo( frag ) )
{
frag->RefID = candidate->RefID;
while ( NULL != candidate->pNextOptimized )
candidate = candidate->pNextOptimized;
candidate->pNextOptimized = frag;
frag->pPrevOptimized = candidate;
break;
}
}
return frag->RefID;
}
//+--------------------------------------------------------------------------
//
// Method: RootFormatFragement::Output
//
// Synopsis: Output a the whole ndr64 format structure
//
//---------------------------------------------------------------------------
void RootFormatFragment::Output( CCB *pCCB )
{
ISTREAM *stream = pCCB->GetStream();
stream->NewLine();
// REVIEW: Is this the right place to output these?
stream->WriteOnNewLine("#include \"ndr64types.h\"");
stream->WriteOnNewLine("#include \"pshpack8.h\"");
stream->NewLine();
FormatFragment *pCurrent;
for ( pCurrent = pTail; NULL != pCurrent; pCurrent = pCurrent->Prev )
{
if ( pCurrent->WasOptimizedOut() )
continue;
stream->NewLine();
stream->WriteOnNewLine("typedef ");
pCurrent->OutputFragmentType( pCCB );
stream->NewLine();
stream->WriteFormat("__midl_frag%d_t;", pCurrent->GetRefID() );
stream->NewLine();
stream->WriteFormat(
"extern const __midl_frag%d_t __midl_frag%d;",
pCurrent->GetRefID(),
pCurrent->GetRefID() );
}
for ( pCurrent = pTail; NULL != pCurrent; pCurrent = pCurrent->Prev )
{
if ( pCurrent->WasOptimizedOut() )
continue;
stream->NewLine( 2 );
stream->WriteFormat(
"static const __midl_frag%d_t __midl_frag%d =",
pCurrent->GetRefID(),
pCurrent->GetRefID() );
pCurrent->OutputFragmentData( pCCB );
stream->Write(";");
}
stream->NewLine( 2 );
stream->WriteOnNewLine("#include \"poppack.h\"");
stream->NewLine( 2 );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::CreateInstance (static)
//
// Synopsis: Class factory for GenNdr64Format
//
//---------------------------------------------------------------------------
GenNdr64Format * GenNdr64Format::CreateInstance( CCB *pCCB )
{
MIDL_ASSERT( pCommand->IsNDR64Run() );
CG_VISITOR_TEMPLATE<GenNdr64Format> *pVisitor = new CG_VISITOR_TEMPLATE<GenNdr64Format>;
GenNdr64Format *generator = pVisitor;
generator->pCCB = pCCB;
generator->pRoot = new RootFormatFragment;
generator->pCurrent = generator->pRoot;
generator->pVisitor = pVisitor;
// add a dummy entry at the beginning of format string to prevent
// an emptry structure being generated when there is only [local]
// interface in an .idl file.
MIDL_NDR_FORMAT_UINT32 * pDummy = new MIDL_NDR_FORMAT_UINT32;
pDummy->Data = 0;
generator->pRoot->AddFragment( pDummy );
return generator;
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::ContinueGeneration
//
// Synopsis: Do generation for a sub-tree. This is typically called
// by a parent node to generate it's children so that it can
// get offsets to the child format string indices, etc.
//
// Parameters: pClass -- The subtree to generate from
// pCompositeFragment -- The composite to generate into
//
//---------------------------------------------------------------------------
FormatInfoRef GenNdr64Format::ContinueGeneration(
CG_CLASS *pClass,
CompositeFormatFragment *pComposite )
{
CompositeFormatFragment *pOldCurrent = pCurrent;
// Some CG classes don't have children. Those classes should not be
// calling ContinueGeneration on thier non-existant children.
MIDL_ASSERT( NULL != pClass );
// Do this here to save from every visit function from doing it.
FormatFragment* pFrag = pComposite->LookupFragment( pClass );
if ( NULL != pFrag )
return pFrag->GetRefID();
if ( NULL != pComposite )
pCurrent = pComposite;
pClass->Visit( pVisitor );
FormatInfoRef NewFragmentID = pCurrent->LookupFragmentID( pClass );
pCurrent = pOldCurrent;
return NewFragmentID;
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Generate
//
// Synopsis: External routine called to generate the string for a type.
//
// Parameters: pClass -- Type to generate string for.
//
//---------------------------------------------------------------------------
FormatInfoRef GenNdr64Format::Generate( CG_CLASS *pClass )
{
return ContinueGeneration( pClass, GetRoot() );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Output
//
// Synopsis: External routine called to output the string.
//
// Parameters: None
//
//---------------------------------------------------------------------------
void GenNdr64Format::Output( )
{
GetRoot()->Output( pCCB );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::ContinueGenerationInRoot
//
// Synopsis: Same as ContinueGeneration except switches to root.
//
// Parameters: pClass -- The subtree to generate from
//
//---------------------------------------------------------------------------
inline FormatInfoRef GenNdr64Format::ContinueGenerationInRoot( CG_CLASS *pClass )
{
return ContinueGeneration( pClass, GetRoot() );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_CLASS )
//
// Synopsis: Default visitor that handles things that don't need any
// handling.
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_CLASS *pClass )
{
// Should only be called for these classes or drived ones
/*
assert(
NULL != dynamic_cast<CG_AUX *>(pClass)
);
*/
CG_ITERATOR Iterator;
CG_CLASS *pChild;
pClass->GetMembers( Iterator );
while ( ITERATOR_GETNEXT( Iterator, pChild ) )
ContinueGenerationInRoot( pChild );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_BASETYPE )
//
// Synopsis: Generate info for base types and ranges
//
// Notes: FC64_BYTE, FC64_CHAR, FC64_WCHAR, FC64_SMALL, FC64_USMALL,
// FC64_SHORT, FC64_USHORT, FC64_LONG, FC64_ULONG, FC64_HYPER, FC64_UHYPER,
// FC64_INT3264, FC64_UINT3264, FC64_FLOAT, FC64_DOUBLE,
// FC64_ERROR_STATUS_T
//
// (also, pending support elsewhere)
//
// FC64_INT128, FC64_UINT128, FC64_FLOAT128, FC64_FLOAT80
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_BASETYPE *pClass )
{
if ( pClass->GetRangeAttribute() )
{
GenRangeFormat( pClass );
return;
}
MIDL_NDR64_FORMAT_CHAR *frag = new MIDL_NDR64_FORMAT_CHAR( pClass );
GetCurrent()->AddFragment( frag );
GetRoot()->OptimizeFragment( frag );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenRangeFormat
//
// Synopsis: Ranges are a special case of base types so we can't
// distinguish between them polymorphically. The base type
// visitor method calls this if we have a range.
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenRangeFormat( CG_BASETYPE *pRangeCG )
{
MIDL_NDR64_RANGE_FORMAT *format;
node_range_attr *range;
format = new MIDL_NDR64_RANGE_FORMAT( pRangeCG );
range = pRangeCG->GetRangeAttribute();
MIDL_ASSERT( NULL != range );
format->FormatCode = FC64_RANGE;
format->RangeType = (NDR64_FORMAT_CHAR)pRangeCG->GetNDR64FormatChar();
format->Reserved = 0;
format->MinValue = range->GetMinExpr()->GetValue();
format->MaxValue = range->GetMaxExpr()->GetValue();
GetCurrent()->AddFragment( format );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_ENCAPSULATED_STRUCT )
//
// Synopsis: Despite the name, this is for generating info for
// encapsulated unions
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_ENCAPSULATED_STRUCT *pEncapUnion )
{
// A union is represented in the format info by a composite containing
// the union header follow by fragments representing the arm selector
CompositeFormatFragment *composite = new CompositeFormatFragment( pEncapUnion );
GetCurrent()->AddFragment( composite );
MIDL_NDR64_ENCAPSULATED_UNION *format;
format = new MIDL_NDR64_ENCAPSULATED_UNION( pEncapUnion );
composite->AddFragment( format );
CG_FIELD *pSwitchField = (CG_FIELD *) pEncapUnion->GetChild();
CG_BASETYPE *pSwitch = (CG_BASETYPE *) pSwitchField->GetChild();
CG_FIELD *pUnionField = (CG_FIELD *) pSwitchField->GetSibling();
CG_UNION *pUnion = (CG_UNION *) pUnionField->GetChild();
MIDL_ASSERT( NULL != dynamic_cast<CG_BASETYPE *>(pSwitch) );
MIDL_ASSERT( NULL != pUnion && NULL != dynamic_cast<CG_UNION *>(pUnion) );
format->FormatCode = FC64_ENCAPSULATED_UNION;
format->Alignment = ConvertAlignment( pEncapUnion->GetWireAlignment() );
format->Flags = 0;
format->SwitchType = (NDR64_FORMAT_CHAR)pSwitch->GetNDR64SignedFormatChar();
format->MemoryOffset = pUnionField->GetMemOffset()
- pSwitchField->GetMemOffset();
format->MemorySize = pEncapUnion->GetMemorySize();
format->Reserved = 0;
GenerateUnionArmSelector( pUnion, composite );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenerateUnionArmSelector
//
// Synopsis: Generate info about union arms
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenerateUnionArmSelector(
CG_UNION *pUnion,
CompositeFormatFragment *FragmentList )
{
MIDL_NDR64_UNION_ARM_SELECTOR *header;
header = new MIDL_NDR64_UNION_ARM_SELECTOR;
header->Reserved1 = 0;
header->Alignment = ConvertAlignment( pUnion->GetWireAlignment() );
header->Reserved2 = 0;
header->Arms = (NDR64_UINT32) pUnion->GetNumberOfArms();
FragmentList->AddFragment( header );
// Generate the non-default arms
CG_ITERATOR Iterator;
CG_CASE *pCase;
CG_CASE *pDefaultCase = NULL;
NDR64_UINT16 ArmCount = 0;
pUnion->GetMembers( Iterator );
while ( ITERATOR_GETNEXT( Iterator, pCase ) )
{
// The default case is always put at the end
if ( ID_CG_DEFAULT_CASE == pCase->GetCGID() )
{
pDefaultCase = pCase;
continue;
}
MIDL_NDR64_UNION_ARM *arm = new MIDL_NDR64_UNION_ARM;
MIDL_ASSERT( NULL != pCase->GetExpr() );
arm->CaseValue = pCase->GetExpr()->GetValue();
// it's legal to have a case with no type
if ( NULL == pCase->GetChild() || NULL == pCase->GetChild()->GetChild() )
arm->Type = 0;
else
arm->Type = ContinueGenerationInRoot(
pCase->GetChild()->GetChild() );
arm->Reserved = 0;
FragmentList->AddFragment( arm );
++ArmCount;
}
MIDL_ASSERT( ArmCount == header->Arms );
// Generate the default
PNDR64_FORMAT Type;
if ( NULL == pDefaultCase )
Type = (PNDR64_FORMAT) -1;
else
{
CG_CLASS *pType = pDefaultCase->GetChild();
if ( NULL != pType )
pType = pType->GetChild();
if ( NULL == pType )
Type = 0;
else
Type = ContinueGenerationInRoot( pType );
}
FragmentList->AddFragment( new MIDL_NDR64_DEFAULT_CASE( Type ) );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_INTERFACE )
//
// Synopsis: Generate info for interfaces
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_INTERFACE *pInterface )
{
CG_ITERATOR I;
CCB *pCCB = GetCCB();
CG_PROC *pProc;
pInterface->InitializeCCB( pCCB );
pCCB->SetImplicitHandleIDNode( 0 );
if( pInterface->GetMembers( I ) )
while ( ITERATOR_GETNEXT( I, pProc ) )
ContinueGenerationInRoot( pProc );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_PARAM )
//
// Synopsis: Generate info for parameters
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_PARAM *pParam )
{
if ( GetCurrent()->HasClassFragment( pParam ) )
return;
NDR64_PARAM_FLAGS Attributes;
CG_NDR *pChild;
CG_NDR *pOldPlaceholder;
pChild = (CG_NDR *) pParam->GetChild();
if ( pChild->GetCGID() == ID_CG_GENERIC_HDL )
pChild = (CG_NDR *) pChild->GetChild();
// Ignore the following type of arguments that don't go on wire:
// - async handles
// - primitive handles
//
if ( pChild->GetCGID() == ID_CG_PRIMITIVE_HDL || ( (CG_PARAM*) pParam)->IsAsyncHandleParam() )
return;
pOldPlaceholder = pCCB->SetLastPlaceholderClass( pParam );
// Get the parameter attributes. The 32-bit PARAM_ATTRIBUTES structure
// is essentially the same as the NDR64_PARAM_FLAGS structure except
// the 64-bit structure has a single bit (UseCache) for the old
// ServerAllocSize field
PARAM_ATTRIBUTES *pattr = (PARAM_ATTRIBUTES *) &Attributes;
pChild->GetNdrParamAttributes( pCCB, pattr );
Attributes.UseCache = (NDR64_UINT16) ((pattr->ServerAllocSize) ? 1 : 0);
Attributes.Reserved = 0;
// For reasons not understood, GenNdrParamAttributes marks basetypes as
// not by value even if they are (unless they are ranges). Fix the
// insanity.
if ( pChild->IsSimpleType() )
Attributes.IsByValue = 1;
pCCB->SetLastPlaceholderClass( pOldPlaceholder );
// Get the fragment of the pointee.
FormatFragment *pChildFragment;
pChildFragment = GetCurrent()->GetParent()->LookupFragment( pChild );
MIDL_ASSERT( NULL != pChildFragment );
// Fill in the parameter information
MIDL_NDR64_PARAM_FORMAT *format;
format = new MIDL_NDR64_PARAM_FORMAT( pParam );
format->Attributes = Attributes;
format->Reserved = 0;
pParam->GetStackOffsets( pCCB, &format->StackOffset );
// REVIEW: The reason why we have to test base types seperately is
// because of the inconsistent way things are stored. Consider
// a string pointer. It's format info is a simple ref to an
// string array - this can be intrepreted as a pointer to a block
// of memory. However, it's CG representation is just a pointer.
// The missing level of dereference causes the problem. In
// constrast a pointer to a single long is represented as a ref
// pointer to a long both in CG and in the format info.
if ( Attributes.IsBasetype
&& ( Attributes.IsByValue || Attributes.IsSimpleRef ) )
{
if ( Attributes.IsSimpleRef )
pChild = (CG_NDR *) pChild->GetChild();
format->Type = GetRoot()->LookupFragment( pChild )->GetRefID();
}
else
{
FormatFragment *pChildFragment = GetCurrent()->GetParent()->LookupFragment( pChild );
MIDL_ASSERT( NULL != pChildFragment );
// Is the parameter is a simple ref, we need to bypass the pointer
// in the format string.
if ( Attributes.IsSimpleRef )
{
MIDL_NDR64_POINTER_FORMAT* pPointerFrag =
dynamic_cast<MIDL_NDR64_POINTER_FORMAT*>(pChildFragment);
format->Type = pPointerFrag->Pointee;
}
else
{
format->Type = pChildFragment->GetRefID();
}
}
GetCurrent()->AddFragment( format );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_PROC )
//
// Synopsis: Generate info for interfaces
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_PROC *pProc )
{
if ( GetCurrent()->HasClassFragment( pProc ) )
return;
CG_NDR * pOldCGNodeContext;
CG_ITERATOR Iterator;
CG_PARAM * pParam;
short ParamNum;
long ServerBufferSize;
long ClientBufferSize;
long BufSize;
BOOL fServerMustSize;
BOOL fClientMustSize;
// Make sure [call_as] targets are processed
CG_PROC *pCallAs = pProc->GetCallAsCG();
if (pCallAs)
ContinueGenerationInRoot( pCallAs );
CompositeFormatFragment *composite = new CompositeFormatFragment( pProc );
composite->SetParent( GetCurrent() );
GetCurrent()->AddFragment(composite);
MIDL_NDR64_PROC_FORMAT *format = new MIDL_NDR64_PROC_FORMAT( pProc );
composite->AddFragment( format );
pCCB->SetInObjectInterface( pProc->IsObject() );
pOldCGNodeContext = pCCB->SetCGNodeContext( pProc );
//
// If this procedure uses an explicit handle then set the
// NdrBindDescriptionOffset to 0 so that it will not try to output it's
// description when given the GenNdrParamOffLine method in the loop below.
// It's description must be part of the procedure description.
//
if ( pProc->GetHandleUsage() == HU_EXPLICIT )
{
CG_HANDLE * pHandle = pProc->GetHandleClassPtr();
pHandle->SetNdrBindDescriptionOffset( 0 );
if ( pHandle->GetCGID() == ID_CG_CONTEXT_HDL )
{
// The context handle directs the call.
((CG_CONTEXT_HANDLE *)pHandle)->SetCannotBeNull();
}
}
if ( !pProc->IsObject() && !pCCB->IsInCallback() )
{
if ( HU_IMPLICIT == pProc->GetHandleUsage() )
{
if ( pProc->IsGenericHandle() )
pCCB->RegisterGenericHandleType(
pProc->GetHandleClassPtr()->GetHandleType() );
}
}
pProc->GetMembers( Iterator );
ParamNum = 0;
ServerBufferSize = 0;
ClientBufferSize = 0;
fServerMustSize = FALSE;
fClientMustSize = FALSE;
pCCB->SetInterpreterOutSize( 0 );
while( ITERATOR_GETNEXT( Iterator, pParam ) )
{
CG_NDR * pChild;
CG_NDR * pOldPlaceholder;
pChild = (CG_NDR *) pParam->GetChild();
// Ignore the following type of arguments that don't go on wire:
// - async handles
// - primitive handles
//
if ( pChild->GetCGID() == ID_CG_PRIMITIVE_HDL || ( (CG_PARAM*) pParam)->IsAsyncHandleParam() )
continue;
pParam->SetParamNumber( ParamNum++ );
pCCB->SetCurrentParam( (CG_PARAM *) pParam );
pOldPlaceholder = pCCB->SetLastPlaceholderClass( pParam );
ContinueGenerationInRoot( pChild );
// A procedure's buffer size does not depend on pipe arguments
if (pChild->IsPipeOrPipeReference())
{
if (pChild->GetChild()->HasAFixedBufferSize())
pParam->SetInterpreterMustSize(FALSE);
else
// There must be a union in there somewhere
pParam->SetInterpreterMustSize(TRUE);
}
else
{
BufSize = pChild->FixedBufferSize( pCCB );
if ( BufSize != -1 )
{
//
// If either the client's or server's fixed buffer size gets too
// big then we force the parameter to be sized.
//
if ( (pParam->IsParamIn() &&
((ClientBufferSize + BufSize) >= 65356)) ||
(pParam->IsParamOut() &&
((ServerBufferSize + BufSize) >= 65356)) )
{
fClientMustSize = TRUE;
fServerMustSize = TRUE;
}
else
{
pParam->SetInterpreterMustSize( FALSE );
if ( pParam->IsParamIn() )
ClientBufferSize += BufSize;
if ( pParam->IsParamOut() )
ServerBufferSize += BufSize;
}
}
else
{
if ( pParam->IsParamIn() )
fClientMustSize = TRUE;
if ( pParam->IsParamOut() )
fServerMustSize = TRUE;
}
}
pCCB->SetCurrentParam( 0 );
pCCB->SetLastPlaceholderClass( pOldPlaceholder );
}
//
// Generate the format string for the return type if needed.
//
if ( pProc->GetReturnType() )
{
CG_NDR * pChild;
CG_NDR * pOldPlaceholder;
pProc->GetReturnType()->SetParamNumber( ParamNum++ );
pChild = (CG_NDR *) pProc->GetReturnType()->GetChild();
pCCB->SetCurrentParam( pProc->GetReturnType() );
pOldPlaceholder = pCCB->SetLastPlaceholderClass( pProc->GetReturnType() );
ContinueGenerationInRoot( pChild );
BufSize = pChild->FixedBufferSize( pCCB );
if ( BufSize != -1 )
{
if ( (ServerBufferSize + BufSize) >= 65536 )
{
fServerMustSize = TRUE;
}
else
{
ServerBufferSize += BufSize;
pProc->GetReturnType()->SetInterpreterMustSize( FALSE );
}
}
else
fServerMustSize = TRUE;
pCCB->SetCurrentParam( 0 );
pCCB->SetLastPlaceholderClass( pOldPlaceholder );
}
pCCB->SetCurrentParam( 0 );
//
// REVIEW: This routine really needs to be broken up a bit
//
// Figure out the handle type
if ( pProc->IsObject() )
format->Flags.HandleType = NDR64_FC_AUTO_HANDLE;
else if ( pCCB->IsInCallback() )
format->Flags.HandleType = NDR64_FC_CALLBACK_HANDLE;
else if ( HU_IMPLICIT != pProc->GetHandleUsage() )
format->Flags.HandleType = NDR64_FC_EXPLICIT_HANDLE;
else if ( pProc->IsAutoHandle() )
format->Flags.HandleType = NDR64_FC_AUTO_HANDLE;
else if ( pProc->IsPrimitiveHandle() )
format->Flags.HandleType = NDR64_FC_BIND_PRIMITIVE;
else if ( pProc->IsGenericHandle() )
format->Flags.HandleType = NDR64_FC_BIND_GENERIC;
// Set the proc flags
format->Flags.ProcType = 0; // REVIEW: ??
format->Flags.IsInterpreted = 1; // REVIEW: ??
format->Flags.IsObject = pProc->IsObject();
format->Flags.IsAsync = pProc->HasAsyncUUID(); // REVIEW: HasAsyncHandle?
format->Flags.IsEncode = pProc->HasEncode();
format->Flags.IsDecode = pProc->HasDecode();
format->Flags.UsesFullPtrPackage = pProc->HasFullPtr();
format->Flags.UsesRpcSmPackage = pProc->MustInvokeRpcSSAllocate();
format->Flags.HandlesExceptions = pProc->HasStatuses(); // REVIEW: ??
format->Flags.UsesPipes = pProc->HasPipes();
format->Flags.ServerMustSize = fServerMustSize;
format->Flags.ClientMustSize = fClientMustSize;
format->Flags.HasReturn = ( NULL != pProc->GetReturnType() );
format->Flags.HasComplexReturn = pProc->HasComplexReturnType();
format->Flags.ServerHasCorrelation = pProc->HasServerCorr();
format->Flags.ClientHasCorrelation = pProc->HasClientCorr();
format->Flags.HasNotify = pProc->HasNotify() || pProc->HasNotifyFlag();
format->Flags.HasOtherExtensions = 0; // Reset in GenExtendedProcInfo
format->Flags.Reserved = 0;
// Figure out the stack size. Note that stack size in the
// NDR64_PROC_FORMAT structure is meaningless outside of ndr since it is
// different for every processor
format->ia64StackSize = pProc->GetTotalStackSize( pCCB );
// Constant buffer sizes
format->ConstantClientBufferSize = ClientBufferSize;
format->ConstantServerBufferSize = ServerBufferSize;
// RpcFlags
//
// REVIEW: Async is not an operation type!
unsigned int opbits = pProc->GetOperationBits();
format->RpcFlags.Idempotent = (NDR64_UINT16) (opbits & OPERATION_IDEMPOTENT ? 1 : 0);
format->RpcFlags.Broadcast = (NDR64_UINT16) (opbits & OPERATION_BROADCAST ? 1 : 0);
format->RpcFlags.Maybe = (NDR64_UINT16) (opbits & OPERATION_MAYBE ? 1 : 0);
format->RpcFlags.Message = (NDR64_UINT16) (opbits & OPERATION_MESSAGE ? 1 : 0);
format->RpcFlags.InputSynchronous = (NDR64_UINT16) (opbits & OPERATION_INPUT_SYNC ? 1 : 0);
format->RpcFlags.Asynchronous = 0; // !!
format->RpcFlags.Reserved1 = 0;
format->RpcFlags.Reserved2 = 0;
format->RpcFlags.Reserved3 = 0;
// Miscellaneous
format->FloatDoubleMask = pProc->GetFloatArgMask( pCCB );
format->NumberOfParams = ParamNum;
format->ExtensionSize = 0; // Reset in GenExtendedProcInfo
// Generate the extended proc info if any
if ( format->Flags.HasNotify || pProc->GetHandleUsage() == HU_EXPLICIT )
GenExtendedProcInfo( composite );
// Now generate the parameter descriptors
ITERATOR_INIT( Iterator );
while ( ITERATOR_GETNEXT( Iterator, pParam ) )
ContinueGeneration( pParam, composite );
if ( pProc->HasReturn() )
ContinueGeneration( pProc->GetReturnType(), composite );
// REVIEW: Are procs optimized? If so be careful because the stack size
// field is set to 0 for all procs
// composite->OptimizeFragment( format );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenExtendedProcInfo
//
// Synopsis: Generate info concerning explicit binding handles and
// [notify] procs.
//
// Parameters: [composite] -- The composite to add the info to
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenExtendedProcInfo( CompositeFormatFragment *composite )
{
// The proc info should be the first thing in the composite
MIDL_NDR64_PROC_FORMAT *procFormat;
CG_PROC *procCG;
MIDL_NDR64_BIND_AND_NOTIFY_EXTENSION *extension;
extension = new MIDL_NDR64_BIND_AND_NOTIFY_EXTENSION;
composite->AddFragment( extension );
procFormat = dynamic_cast<MIDL_NDR64_PROC_FORMAT *>(composite->GetFirstFragment());
MIDL_ASSERT( NULL != procFormat );
procCG = (CG_PROC *) procFormat->GetCGNode();
procFormat->Flags.HasOtherExtensions = 1; // REVIEW: ??
//
// explicit handles
//
if ( procCG->GetHandleUsage() == HU_EXPLICIT )
{
CG_HANDLE *pHandle = procCG->GetHandleClassPtr();
CG_PARAM *pParam = procCG->GetHandleUsagePtr();
CG_NDR *pOldPlaceholder;
pOldPlaceholder = pCCB->SetLastPlaceholderClass( pParam );
pHandle->GetNdrHandleInfo( pCCB, &extension->Binding );
pParam->GetStackOffsets( pCCB, &extension->StackOffsets );
pCCB->SetLastPlaceholderClass( pOldPlaceholder );
}
else
{
// No explicit handle, zero out the handle info.
memset( &extension->Binding, 0, sizeof( extension->Binding ) );
memset( &extension->StackOffsets, 0, sizeof( extension->StackOffsets ) );
}
//
// [notify]
//
if ( procFormat->Flags.HasNotify )
extension->NotifyIndex = procCG->GetNotifyTableOffset( pCCB );
else
extension->NotifyIndex = 0;
// Update the extension size field.
procFormat->ExtensionSize = sizeof(NDR64_BIND_AND_NOTIFY_EXTENSION);
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_POINTER )
//
// Synopsis: Generate info for simple pointer and add fragment to
// current composite fragment.
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_POINTER *pPointer )
{
MIDL_NDR64_POINTER_FORMAT *format;
format = new MIDL_NDR64_POINTER_FORMAT( pPointer );
// Get the type of the pointer (ref, etc) and any flags
pPointer->GetTypeAndFlags( pCCB, format );
CG_CLASS *pointee = pPointer->GetChild();
if ( format->Flags & FC_SIMPLE_POINTER )
{
// generic handles are represented as thier underlying type and
// otherwise pretty much ignored for our purposes.
if ( ID_CG_GENERIC_HDL == pointee->GetCGID() )
pointee = pointee->GetChild();
MIDL_ASSERT( NULL != dynamic_cast<CG_BASETYPE *> ( pointee ) );
}
GetCurrent()->AddFragment( format );
format->Reserved = 0;
format->Pointee = ContinueGenerationInRoot( pointee );
GetRoot()->OptimizeFragment( format );
}
MIDL_NDR64_POINTER_FORMAT* GenNdr64Format::GenQualifiedPtrHdr( CG_QUALIFIED_POINTER *pPointer )
{
MIDL_NDR64_POINTER_FORMAT* pHeader = new MIDL_NDR64_POINTER_FORMAT( pPointer );
pPointer->GetTypeAndFlags( pCCB, pHeader );
// Always use complex pointer struct
pHeader->Flags &= ~FC_SIMPLE_POINTER;
pHeader->Reserved = 0;
pHeader->Pointee = INVALID_FRAGMENT_ID;
return pHeader;
}
MIDL_NDR64_POINTER_FORMAT* GenNdr64Format::GenQualifiedArrayPtr( CG_ARRAY *pArray )
{
// Ref arrays do not need the pointer header.
if ( pArray->GetPtrType() == PTR_REF )
return NULL;
MIDL_NDR64_POINTER_FORMAT* pFragment = new MIDL_NDR64_POINTER_FORMAT( pArray );
switch ( pArray->GetPtrType() )
{
case PTR_UNKNOWN:
case PTR_REF:
MIDL_ASSERT(0);
break;
case PTR_UNIQUE:
pFragment->FormatCode = FC64_UP;
break;
case PTR_FULL:
pFragment->FormatCode = FC64_FP;
break;
default:
MIDL_ASSERT(0);
break;
}
pFragment->Flags = 0;
pFragment->Reserved = 0;
return pFragment;
}
void GenNdr64Format::Visit( CG_STRING_POINTER *pPointer )
{
// REVIEW: This is essentially two routines distinguished by an if.
// Seperate them.
MIDL_NDR64_POINTER_FORMAT* pPointerHdr = GenQualifiedPtrHdr( pPointer );
GetCurrent()->AddFragment( pPointerHdr );
FormatFragment *pStringFrag = NULL;
if ( dynamic_cast<CG_SIZE_STRING_POINTER *>(pPointer) )
{
CG_CONF_ATTRIBUTE *pConfAttribute = dynamic_cast<CG_CONF_ATTRIBUTE*>( pPointer );
FormatFragment *pFrag = GenerateCorrelationDescriptor( pConfAttribute->GetSizeIsExpr() );
MIDL_NDR64_SIZED_CONFORMANT_STRING_FORMAT *pSizedConfFormat =
new MIDL_NDR64_SIZED_CONFORMANT_STRING_FORMAT( pPointer );
pStringFrag = pSizedConfFormat;
InitStringHeader( pPointer, &pSizedConfFormat->Header, true, true);
pSizedConfFormat->SizeDescription = (PNDR64_FORMAT)pFrag->GetRefID();
}
else
{
MIDL_NDR64_CONFORMANT_STRING_FORMAT *pConfFormat =
new MIDL_NDR64_CONFORMANT_STRING_FORMAT( pPointer );
pStringFrag = pConfFormat;
InitStringHeader( pPointer, &pConfFormat->Header, true, false);
}
GetRoot()->AddFragment( pStringFrag );
GetRoot()->OptimizeFragment( pStringFrag );
pPointerHdr->Pointee = pStringFrag->GetRefID();
GetRoot()->OptimizeFragment( pPointerHdr );
}
void GenNdr64Format::GenerateNonStringQualifiedPtr( CG_QUALIFIED_POINTER *pPointer )
{
MIDL_NDR64_POINTER_FORMAT *pPointerFrag = GenQualifiedPtrHdr( pPointer );
GetCurrent()->AddFragment( pPointerFrag );
FormatFragment *pArrayFrag = GenerateNonStringQualifiedArrayLayout( pPointer, GetRoot() );
pPointerFrag->Pointee = pArrayFrag->GetRefID();
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenInterfacePointer
//
// Synopsis: Generate info for interface pointers
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenInterfacePointer( CG_POINTER *pPointer, BOOL IsConstantIID )
{
MIDL_NDR64_POINTER_FORMAT *format;
format = new MIDL_NDR64_POINTER_FORMAT( pPointer );
// Generate the pointer node
pPointer->GetTypeAndFlags( pCCB, format );
format->FormatCode = FC64_IP; // GetTypeAndFlags doesn't emit FC64_IP
format->Reserved = 0;
GetCurrent()->AddFragment( format );
// generate the interface type node
if ( IsConstantIID )
{
CG_INTERFACE_POINTER *pInterface =
dynamic_cast<CG_INTERFACE_POINTER*>( pPointer );
MIDL_NDR64_CONSTANT_IID_FORMAT *iid;
iid = new MIDL_NDR64_CONSTANT_IID_FORMAT;
iid->FormatCode = FC64_IP;
iid->Flags.ConstantIID = 1;
iid->Flags.Reserved = 0;
iid->Reserved = 0;
node_guid * pGuid = (node_guid *) pInterface
->GetTheInterface()
->GetAttribute( ATTR_GUID );
pGuid->GetGuid( iid->Guid );
format->Pointee = GetRoot()->AddFragment( iid );
}
else
{
CG_IIDIS_INTERFACE_POINTER *piidInterface =
dynamic_cast<CG_IIDIS_INTERFACE_POINTER*>( pPointer );
FormatFragment *pIIDExpr = GenerateCorrelationDescriptor( piidInterface->GetIIDExpr() );
MIDL_NDR64_IID_FORMAT *iid = new MIDL_NDR64_IID_FORMAT;
iid->FormatCode = FC64_IP;
iid->Flags.ConstantIID = 0;
iid->Flags.Reserved = 0;
iid->Reserved = 0;
iid->IIDDescriptor = pIIDExpr->GetRefID();
format->Pointee = GetRoot()->AddFragment( iid );
}
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_UNION )
//
// Synopsis: Generate info for unions
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_UNION *pUnion )
{
// A union is represented in the format info by a composite containing
// the union header follow by fragments representing the arm selector
CompositeFormatFragment *composite = new CompositeFormatFragment( pUnion );
GetCurrent()->AddFragment( composite );
MIDL_NDR64_NON_ENCAPSULATED_UNION *format;
format = new MIDL_NDR64_NON_ENCAPSULATED_UNION( pUnion );
composite->AddFragment( format );
MIDL_ASSERT( dynamic_cast<CG_BASETYPE *> (pUnion->GetSwitchType()) );
NDR64_FORMAT_CHAR SwitchType = (NDR64_FORMAT_CHAR)
( (CG_BASETYPE *) pUnion->GetSwitchType() )
->GetNDR64SignedFormatChar();
// The alignment of a union is the max of the type alignment and the arm alignment.
unsigned short UnionArmAlignment = pUnion->GetWireAlignment();
unsigned short SwitchTypeAlignment =
( (CG_BASETYPE*) pUnion->GetSwitchType() )->GetWireAlignment() ;
unsigned short UnionAlignment = UnionArmAlignment > SwitchTypeAlignment ?
UnionArmAlignment : SwitchTypeAlignment;
format->FormatCode = FC64_NON_ENCAPSULATED_UNION;
format->Alignment = ConvertAlignment( UnionAlignment );
format->Flags = 0;
format->SwitchType = SwitchType;
format->MemorySize = pUnion->GetMemorySize();
format->Reserved = 0;
expr_node *pSwitchExpr = pUnion->GetNdr64SwitchIsExpr();
FormatFragment* pSwitchFormat = GenerateCorrelationDescriptor( pSwitchExpr );
format->Switch = (PNDR64_FORMAT)pSwitchFormat->GetRefID();
GenerateUnionArmSelector( pUnion, composite );
GetRoot()->OptimizeFragment( composite );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_CONTEXT_HANDLE )
//
// Synopsis: Generate info for context handles
//
// Notes: This is an an abreviated version of the same information in
// the proc descriptor.
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_CONTEXT_HANDLE *pHandle )
{
CG_PARAM * pBindParam = (CG_PARAM *) pCCB->GetLastPlaceholderClass();
CG_PROC * pProc = (CG_PROC *) pCCB->GetCGNodeContext();
MIDL_NDR64_CONTEXT_HANDLE_FORMAT *format;
format = new MIDL_NDR64_CONTEXT_HANDLE_FORMAT( pHandle );
format->FormatCode = FC64_BIND_CONTEXT;
//
// Register the rundown routine always, even if the context handle is
// not used as the binding paramter.
//
if ( pHandle->GetHandleType()->NodeKind() == NODE_DEF )
{
pCCB->RegisterContextHandleType( pHandle->GetHandleType() );
}
// Flags
unsigned char uFlags = pHandle->MakeExplicitHandleFlag( pBindParam );
format->ContextFlags.CannotBeNull = pHandle->GetCannotBeNull()
|| pBindParam->IsParamIn()
&& !pBindParam->IsParamOut();
format->ContextFlags.Serialize = (NDR64_UINT8) ( pHandle->HasSerialize() ? 1 : 0 );
format->ContextFlags.NoSerialize = (NDR64_UINT8) ( pHandle->HasNoSerialize() ? 1 : 0 );
format->ContextFlags.Strict = (NDR64_UINT8) ( pHandle->HasStrict() ? 1 : 0 );
format->ContextFlags.IsReturn = (NDR64_UINT8) ( ( uFlags & HANDLE_PARAM_IS_RETURN ) ? 1 : 0 );
format->ContextFlags.IsOut = (NDR64_UINT8) ( ( uFlags & HANDLE_PARAM_IS_OUT ) ? 1 : 0 );
format->ContextFlags.IsIn = (NDR64_UINT8) ( ( uFlags & HANDLE_PARAM_IS_IN ) ? 1 : 0 );
format->ContextFlags.IsViaPointer = (NDR64_UINT8) ( ( uFlags & HANDLE_PARAM_IS_VIA_PTR ) ? 1 : 0 );
// Routine index.
// IndexMgr keeps indexes 1..n, we use indexes 0..n-1
format->RundownRoutineIndex = (NDR64_UINT8)
( pCCB->LookupRundownRoutine( pHandle->GetRundownRtnName() ) - 1);
// Ordinal
format->Ordinal = (NDR64_UINT8) pProc->GetContextHandleCount();
pProc->SetContextHandleCount( (short) (format->Ordinal + 1) );
GetCurrent()->AddFragment( format );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_GENERIC_HANDLE )
//
// Synopsis: Generate info for generic handles
//
// Notes: Actually generic handles are represented by thier underlying
// type, not by any special "generic handle" type. Therefore
// there's nothing to generate. We do need a place to
// register the handle though.
//
// Review: Since we're not generating perhaps registering is better
// placed in some other phase?
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_GENERIC_HANDLE *pHandle )
{
ContinueGenerationInRoot( pHandle->GetChild() );
MIDL_ASSERT( pCCB->GetCGNodeContext()->IsProc() );
CG_HANDLE *pBindingHandle = ((CG_PROC *)pCCB->GetCGNodeContext())
->GetHandleClassPtr();
if ( pBindingHandle == pHandle )
pCCB->RegisterGenericHandleType( pHandle->GetHandleType() );
}
//---------------------------------------------------------------------------
//
// Pointer Layout Functions
//
//---------------------------------------------------------------------------
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenSimplePtrLayout( CG_STRUCT, bool, ulong *)
//
// Synopsis: Generate a pointer layout for a block copyable structure.
//
//---------------------------------------------------------------------------
FormatFragment * GenNdr64Format::GenSimplePtrLayout( CG_STRUCT *pStruct,
bool bGenHeaderFooter,
ulong *pPtrInstances )
{
CompositeFormatFragment *pCompositeFormatFragment = NULL;
CCB *pCCB = GetCCB();
ulong NumberPointerInstances = 0;
CG_FIELD *pOldRegionField = NULL;
CG_NDR *pOldContext = NULL;
if ( dynamic_cast<CG_REGION*>( pStruct ) )
pOldRegionField = pCCB->StartRegion();
else
pOldContext = pCCB->SetCGNodeContext( pStruct );
CG_ITERATOR Iterator;
pStruct->GetMembers( Iterator );
CG_FIELD *pField;
ITERATOR_INIT( Iterator );
while ( ITERATOR_GETNEXT( Iterator, pField ) )
{
CG_NDR * pOldPlaceHolder = pCCB->SetLastPlaceholderClass( pField );
CG_CLASS *pMember = (CG_NDR *) pField->GetChild();
MIDL_ASSERT( !pMember->IsStruct() ); // Structure should have been unrolled.
if ( pMember->IsPointer() )
{
if ( NULL == pCompositeFormatFragment )
pCompositeFormatFragment = new CompositeFormatFragment();
if ( bGenHeaderFooter )
pCompositeFormatFragment->AddFragment(
new MIDL_NDR64_NO_REPEAT_FORMAT() );
pCompositeFormatFragment->AddFragment(
new MIDL_NDR64_POINTER_INSTANCE_HEADER_FORMAT( pField->GetMemOffset() ) );
ContinueGeneration( pMember, pCompositeFormatFragment );
NumberPointerInstances++;
}
// For now, arrays do not have pointers.
else if ( pMember->IsArray() )
{
FormatFragment *pPointerFragment = GenSimplePtrLayout( dynamic_cast<CG_ARRAY*>( pMember ),
false,
pField->GetMemOffset() );
if ( NULL != pPointerFragment )
{
if ( NULL == pCompositeFormatFragment )
pCompositeFormatFragment = new CompositeFormatFragment();
pCompositeFormatFragment->AddFragment( pPointerFragment );
}
}
pCCB->SetLastPlaceholderClass( pOldPlaceHolder );
}
if ( ( NULL != pCompositeFormatFragment ) &&
bGenHeaderFooter )
{
FormatFragment *pFormatFragment = new MIDL_NDR64_FORMAT_CHAR( FC64_END );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
if ( NULL != pPtrInstances )
{
*pPtrInstances = NumberPointerInstances;
}
if ( dynamic_cast<CG_REGION*>( pStruct ) )
pCCB->EndRegion( pOldRegionField );
else
pCCB->SetCGNodeContext( pOldContext );
return pCompositeFormatFragment;
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenSimplePtrLayout( CG_ARRAY, bool, ulong *)
//
// Synopsis: Generate a pointer layout for a block copyable structure.
//
//---------------------------------------------------------------------------
FormatFragment * GenNdr64Format::GenSimplePtrLayout( CG_NDR *pArray,
bool bGenHeaderFooter,
ulong MemoryOffset )
{
CompositeFormatFragment *pArrayLayout = NULL;
CG_CLASS *pChild = pArray->GetChild();
FormatFragment* pChildLayout = NULL;
unsigned long NumberPointers = 0;
// Multidimensional arrays should be bogus
MIDL_ASSERT( !pChild->IsArray() );
if ( pChild->IsPointer() )
{
CompositeFormatFragment *pCompositeFormatFragment =
new CompositeFormatFragment();
pCompositeFormatFragment->AddFragment(
new MIDL_NDR64_POINTER_INSTANCE_HEADER_FORMAT( 0 ) );
ContinueGeneration( pChild, pCompositeFormatFragment );
pChildLayout = pCompositeFormatFragment;
NumberPointers = 1;
}
else if ( pChild->IsStruct() )
{
pChildLayout = GenSimplePtrLayout( dynamic_cast< CG_STRUCT *>( pChild ),
false,
&NumberPointers );
}
if ( pChildLayout )
{
pArrayLayout = new CompositeFormatFragment();
FormatFragment *pHeader;
if ( dynamic_cast<CG_FIXED_ARRAY*>( pArray ) )
{
CG_FIXED_ARRAY *pFixedArray = (CG_FIXED_ARRAY*)( pArray );
pHeader = new MIDL_NDR64_FIXED_REPEAT_FORMAT( dynamic_cast<CG_NDR*>( pChild )->GetMemorySize(),
MemoryOffset,
NumberPointers,
pFixedArray->GetNumOfElements(),
dynamic_cast<CG_NDR*>( pChild )->IsStruct() );
}
else
{
pHeader = new MIDL_NDR64_REPEAT_FORMAT( dynamic_cast<CG_NDR*>( pChild )->GetMemorySize(),
MemoryOffset,
NumberPointers,
dynamic_cast< CG_NDR* >( pChild )->IsStruct() );
}
pArrayLayout->AddFragment( pHeader );
pArrayLayout->AddFragment( pChildLayout );
if ( bGenHeaderFooter )
{
pArrayLayout->AddFragment( new MIDL_NDR64_FORMAT_CHAR( FC64_END ) );
}
}
return pArrayLayout;
}
//--------------------------------------------------------------------------
//
// Structure layout
//
//
//--------------------------------------------------------------------------
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenCmplxPtrLayout( CG_COMPLEX_STRUCT )
//
// Synopsis: Generate a pointer layout for a complex structure.
//
//---------------------------------------------------------------------------
FormatFragment * GenNdr64Format::GenCmplxPtrLayout( CG_COMPLEX_STRUCT *pStruct )
{
CompositeFormatFragment *pCompositeFormatFragment = NULL;
CCB *pCCB = GetCCB();
CG_ITERATOR Iterator;
pStruct->GetMembers( Iterator );
CG_FIELD *pField;
ITERATOR_INIT( Iterator );
while ( ITERATOR_GETNEXT( Iterator, pField ) )
{
CG_NDR * pOldPlaceHolder = pCCB->SetLastPlaceholderClass( pField );
CG_CLASS *pMember = (CG_NDR *) pField->GetChild();
if ( pMember->IsPointer() )
{
if (NULL == pCompositeFormatFragment)
pCompositeFormatFragment = new CompositeFormatFragment();
ContinueGeneration( pMember, pCompositeFormatFragment );
}
pCCB->SetLastPlaceholderClass( pOldPlaceHolder );
}
return pCompositeFormatFragment;
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenerateSimpleStructure( CG_STRUCT, bool )
//
// Synopsis: Generate the format string for a simple structure.
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenerateSimpleStructure( CG_STRUCT *pStruct,
bool IsConformant )
{
// Add the fragment right away so recursive structure definitions
// are caught
CompositeFormatFragment *pMainFragment
= new CompositeFormatFragment( pStruct );
GetCurrent()->AddFragment( pMainFragment );
CG_FIELD *pOldRegionField = NULL;
CG_NDR *pOldContext = NULL;
if ( dynamic_cast<CG_REGION*>( pStruct ) )
pOldRegionField = pCCB->StartRegion();
else
pOldContext = pCCB->SetCGNodeContext( pStruct );
FormatFragment* pMemberLayout = NULL;
FormatFragment* pPointerLayout = GenSimplePtrLayout( pStruct );
if ( pCommand->NeedsNDR64DebugInfo() )
pMemberLayout = GenerateStructureMemberLayout( pStruct, true );
if ( IsConformant )
{
CG_CONFORMANT_STRUCT *pConfStruct = dynamic_cast<CG_CONFORMANT_STRUCT*>( pStruct );
CG_FIELD *pConfField = dynamic_cast<CG_FIELD*>( pConfStruct->GetConformantField() );
MIDL_ASSERT( NULL != pConfField );
CG_ARRAY *pConfArray = dynamic_cast<CG_ARRAY*>( pConfField->GetChild() );
MIDL_ASSERT( NULL != pConfArray );
CG_NDR * pOldPlaceholder = GetCCB()->SetLastPlaceholderClass( pConfField );
FormatFragment *pHeader =
new MIDL_NDR64_CONF_STRUCTURE_HEADER_FORMAT( pConfStruct,
NULL != pPointerLayout,
NULL != pMemberLayout,
ContinueGenerationInRoot( pConfArray ) );
pMainFragment->AddFragment( pHeader );
GetCCB()->SetLastPlaceholderClass( pOldPlaceholder );
}
else
{
FormatFragment *pHeader =
new MIDL_NDR64_STRUCTURE_HEADER_FORMAT( pStruct,
NULL != pPointerLayout,
NULL != pMemberLayout );
pMainFragment->AddFragment( pHeader );
}
if ( NULL != pPointerLayout ) pMainFragment->AddFragment( pPointerLayout );
if ( NULL != pMemberLayout ) pMainFragment->AddFragment( pMemberLayout );
if ( dynamic_cast<CG_REGION*>( pStruct ) )
pCCB->EndRegion( pOldRegionField );
else
pCCB->SetCGNodeContext( pOldContext );
GetRoot()->OptimizeFragment( pMainFragment );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenerateComplexStruct( CG_STRUCT, bool )
//
// Synopsis: Generate the format string for a complex structure.
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenerateComplexStruct( CG_COMPLEX_STRUCT *pStruct,
bool IsConformant )
{
if ( GetCurrent()->HasClassFragment( pStruct ) )
return;
// This must be done first to handle recurion.
CompositeFormatFragment *pMainFragment =
new CompositeFormatFragment( pStruct );
GetCurrent()->AddFragment( pMainFragment );
CG_NDR *pOldContext = pCCB->SetCGNodeContext( pStruct );
FormatInfoRef ArrayID = INVALID_FRAGMENT_ID;
CG_ARRAY *pConfArray = NULL;
if ( IsConformant )
{
CG_FIELD *pConfField = dynamic_cast<CG_FIELD*>( pStruct->GetConformantField() );
MIDL_ASSERT( NULL != pConfField );
pConfArray = dynamic_cast<CG_ARRAY*>( pConfField->GetChild() );
MIDL_ASSERT( NULL != pConfArray );
CG_NDR * pOldPlaceholder = GetCCB()->SetLastPlaceholderClass( pConfField );
ArrayID = ContinueGenerationInRoot( pConfArray );
GetCCB()->SetLastPlaceholderClass( pOldPlaceholder );
}
FormatInfoRef PointerID = INVALID_FRAGMENT_ID;
FormatFragment *pPointerLayout = GenCmplxPtrLayout( pStruct );
if ( NULL != pPointerLayout )
{
// Pointer Layout for complex struct goes in the root fragment.
PointerID = GetRoot()->AddFragment( pPointerLayout );
}
FormatFragment* pMemberLayout = GenerateStructureMemberLayout( pStruct, false );
// The only difference between the member layout with debugging
// and without debugging is that simple region become complex regions
// with member layouts.
FormatInfoRef DebugPointerID = INVALID_FRAGMENT_ID;
FormatInfoRef DebugMemberLayoutID = INVALID_FRAGMENT_ID;
if ( pCommand->NeedsNDR64DebugInfo() )
{
DebugPointerID = PointerID;
FormatFragment *pDebugMemberLayout = GenerateStructureMemberLayout( pStruct, true );
DebugMemberLayoutID = GetRoot()->AddFragment( pDebugMemberLayout );
}
FormatFragment* pHeader;
if ( IsConformant )
{
MIDL_ASSERT( NULL != pConfArray );
pHeader = new MIDL_NDR64_CONF_BOGUS_STRUCTURE_HEADER_FORMAT( pStruct,
pConfArray,
ArrayID,
DebugMemberLayoutID,
DebugPointerID,
PointerID );
}
else
{
pHeader = new MIDL_NDR64_BOGUS_STRUCTURE_HEADER_FORMAT( pStruct,
DebugMemberLayoutID,
DebugPointerID,
PointerID );
}
pMainFragment->AddFragment( pHeader );
pMainFragment->AddFragment( pMemberLayout );
pCCB->SetCGNodeContext( pOldContext );
}
//+--------------------------------------------------------------------------
//
// Class: StructureMemberGenerator
//
// Synopsis: This object is responsible for generating the member layout
// of a structure.
//
//---------------------------------------------------------------------------
class StructureMemberGenerator
{
private:
GenNdr64Format *pMainGenerator;
CompositeFormatFragment *pCompositeFormatFragment;
CG_VISITOR *pVisitor;
bool bIsDebug;
void StartGeneration( CG_STRUCT *pStruct );
void GenerateMember( CG_CLASS *pClass )
{
pClass->Visit( pVisitor );
}
public:
static FormatFragment* Generate( GenNdr64Format *pMainGenerator,
CG_STRUCT *pStruct,
bool bIsDebug )
{
CG_VISITOR_TEMPLATE<StructureMemberGenerator> TemplateVisitor;
StructureMemberGenerator & Visitor = TemplateVisitor;
Visitor.pMainGenerator = pMainGenerator;
Visitor.pCompositeFormatFragment = new CompositeFormatFragment();
Visitor.pVisitor = &TemplateVisitor;
Visitor.bIsDebug = bIsDebug;
Visitor.StartGeneration( pStruct );
return Visitor.pCompositeFormatFragment;
}
// Items that get a unique member layout item.
void Visit( CG_POINTER *pPointer)
{
pPointer;
FormatFragment *pFormatFragment = new MIDL_NDR64_SIMPLE_MEMBER_FORMAT( FC64_POINTER );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
void Visit( CG_IGNORED_POINTER *pPointer )
{
pPointer;
FormatFragment *pFormatFragment = new MIDL_NDR64_SIMPLE_MEMBER_FORMAT( FC64_IGNORE );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
void Visit( CG_BASETYPE *pBaseType )
{
if ( pBaseType->GetRangeAttribute() != NULL )
{
// Send to generic catch all case(Embedded complex)
Visit( (CG_CLASS*) pBaseType );
return;
}
NDR64_FORMAT_CHAR FormatCode = (NDR64_FORMAT_CHAR)pBaseType->GetNDR64FormatChar();
FormatFragment *pFormatFragment = new MIDL_NDR64_SIMPLE_MEMBER_FORMAT( FormatCode );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
void Visit( CG_SIMPLE_REGION *pSimpleRegion )
{
// If this member layout is for debugging, send this
// to the CG_CLASS visit function to create an
// embedded complex.
if ( bIsDebug )
{
Visit( (CG_CLASS*)pSimpleRegion );
return;
}
FormatFragment *pFormatFragment =
new MIDL_NDR64_SIMPLE_REGION_FORMAT( pSimpleRegion );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
void Visit( CG_PAD *pPad )
{
FormatFragment *pFormatFragment
= new MIDL_NDR64_BUFFER_ALIGN_FORMAT( pPad );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
// Do nothing for conformant arrays. They must be at the end.
void Visit( CG_CONFORMANT_ARRAY *pConfArray ) { pConfArray; }
void Visit( CG_CONFORMANT_VARYING_ARRAY *pConfVaryArray ) { pConfVaryArray; }
void Visit( CG_CONFORMANT_STRING_ARRAY *pConfStringArray ) { pConfStringArray; }
// Catch all case. Generates embedded complex
void Visit( CG_CLASS *pClass )
{
FormatInfoRef ID = pMainGenerator->ContinueGenerationInRoot( pClass );
FormatFragment *pFormatFragment = new MIDL_NDR64_EMBEDDED_COMPLEX_FORMAT( ID );
pCompositeFormatFragment->AddFragment( pFormatFragment );
}
};
void StructureMemberGenerator::StartGeneration( CG_STRUCT *pStruct )
{
CCB *pCCB = pMainGenerator->GetCCB();
CG_NDR * pOldPlaceholder = pCCB->GetLastPlaceholderClass();
CG_ITERATOR Iterator;
pStruct->GetMembers( Iterator );
CG_FIELD * pField;
unsigned long BufferOffset = 0;
ITERATOR_INIT( Iterator );
while( ITERATOR_GETNEXT( Iterator, pField ) )
{
//FormatFragment *FormatFragment = NULL;
unsigned long MemPad = pField->GetMemOffset() - BufferOffset;
if ( MemPad != 0 )
{
MIDL_NDR64_MEMPAD_FORMAT *pMemPadFormat =
new MIDL_NDR64_MEMPAD_FORMAT( MemPad );
pCompositeFormatFragment->AddFragment( pMemPadFormat );
}
BufferOffset += MemPad;
pCCB->SetLastPlaceholderClass( pField );
CG_NDR *pMember = (CG_NDR *) pField->GetChild();
// Embedded unknown represent as is not allowed.
MIDL_ASSERT( ! pField->HasEmbeddedUnknownRepAs() );
GenerateMember( pMember );
BufferOffset += pField->GetMemorySize();
}
// Account for padding at the end of the structure.
MIDL_ASSERT( pStruct->GetMemorySize() >= BufferOffset );
unsigned long EndingPad = pStruct->GetMemorySize() - BufferOffset;
if ( EndingPad )
{
MIDL_NDR64_MEMPAD_FORMAT *pMemPadFormat =
new MIDL_NDR64_MEMPAD_FORMAT( EndingPad );
pCompositeFormatFragment->AddFragment( pMemPadFormat );
}
FormatFragment* pEndFormatFragment = new MIDL_NDR64_SIMPLE_MEMBER_FORMAT( FC64_END );
pCompositeFormatFragment->AddFragment( pEndFormatFragment );
pCCB->SetLastPlaceholderClass( pOldPlaceholder );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenerateStructureMemberLayout( CG_STRUCT, bool )
//
// Synopsis: Wrapper for the member layout generator
//
//---------------------------------------------------------------------------
FormatFragment *GenNdr64Format::GenerateStructureMemberLayout( CG_STRUCT *pStruct, bool bIsDebug )
{
return StructureMemberGenerator::Generate( this, pStruct, bIsDebug );
}
//---------------------------------------------------------------------------
//
//
// Arrays
//
//
//
//---------------------------------------------------------------------------
FormatFragment *GenNdr64Format::GenerateArrayElementInfo( CG_CLASS *pChild )
{
CG_NDR *pNdr = dynamic_cast<CG_NDR*>( pChild );
MIDL_ASSERT( NULL != pNdr );
MIDL_NDR64_ARRAY_ELEMENT_INFO *pElementInfo =
new MIDL_NDR64_ARRAY_ELEMENT_INFO();
pElementInfo->ElementMemSize = pNdr->GetMemorySize();
pElementInfo->Element = (PNDR64_FORMAT)ContinueGenerationInRoot( pNdr );
return pElementInfo;
}
void GenNdr64Format::Visit( CG_FIXED_ARRAY *pArray )
{
CompositeFormatFragment *pMainFragment = new CompositeFormatFragment( pArray );
GetCurrent()->AddFragment( pMainFragment );
FormatFragment *pMemberInfo = NULL;
FormatFragment *pPointerLayout = GenSimplePtrLayout( pArray );
if ( pCommand->NeedsNDR64DebugInfo() )
pMemberInfo = GenerateArrayElementInfo( pArray->GetChild() );
MIDL_NDR64_FIX_ARRAY_HEADER_FORMAT *pArrayFormat =
new MIDL_NDR64_FIX_ARRAY_HEADER_FORMAT( pArray );
memset( (NDR64_FIX_ARRAY_HEADER_FORMAT*)pArrayFormat,
0,
sizeof(NDR64_FIX_ARRAY_HEADER_FORMAT*));
pArrayFormat->FormatCode = (NDR64_FORMAT_CHAR) FC64_FIX_ARRAY;
pArrayFormat->Alignment = ConvertAlignment( pArray->GetWireAlignment() );
pArrayFormat->Flags.HasPointerInfo = (NULL != pPointerLayout);
pArrayFormat->Flags.HasElementInfo = (NULL != pMemberInfo );;
pArrayFormat->Reserved = 0;
pArrayFormat->TotalSize = pArray->GetMemorySize();
pMainFragment->AddFragment( pArrayFormat );
if ( pPointerLayout )
pMainFragment->AddFragment( pPointerLayout );
if ( pMemberInfo )
pMainFragment->AddFragment( pMemberInfo );
}
void GenNdr64Format::GenerateFixBogusArrayCommon( CG_FIXED_ARRAY *pArray, bool IsFullBogus )
{
CompositeFormatFragment *pMainFragment = new CompositeFormatFragment( pArray );
GetCurrent()->AddFragment( pMainFragment );
CG_ILANALYSIS_INFO *pAnalysisInfo = pArray->GetILAnalysisInfo();
MIDL_NDR64_BOGUS_ARRAY_HEADER_FORMAT *pHeaderFragment =
new MIDL_NDR64_BOGUS_ARRAY_HEADER_FORMAT( pArray );
memset( (NDR64_BOGUS_ARRAY_HEADER_FORMAT*)pHeaderFragment,
0,
sizeof(NDR64_BOGUS_ARRAY_HEADER_FORMAT));
pHeaderFragment->FormatCode = (NDR64_FORMAT_CHAR)
( IsFullBogus ? FC64_FIX_BOGUS_ARRAY :
FC64_FIX_FORCED_BOGUS_ARRAY );
pHeaderFragment->Alignment = ConvertAlignment( ((CG_NDR*)pArray->GetChild())
->GetWireAlignment() );
pHeaderFragment->Flags.HasPointerInfo = false;
pHeaderFragment->Flags.HasElementInfo = true;
pHeaderFragment->Flags.IsArrayofStrings = pAnalysisInfo->IsArrayofStrings();
pHeaderFragment->Flags.IsMultiDimensional = pAnalysisInfo->IsMultiDimensional();
pHeaderFragment->NumberDims = pAnalysisInfo->GetDimensions();
pHeaderFragment->NumberElements = pArray->GetNumOfElements();
pHeaderFragment->Element = (PNDR64_FORMAT)
ContinueGenerationInRoot( pArray->GetChild() );
pMainFragment->AddFragment( pHeaderFragment );
}
FormatFragment *
GenNdr64Format::GenerateNonStringQualifiedArrayLayout( CG_NDR *pNdr,
CompositeFormatFragment *pComp )
{
CompositeFormatFragment *pMainFragment = new CompositeFormatFragment( pNdr );
pComp->AddFragment( pMainFragment );
FormatFragment *pHeaderFragment;
FormatFragment *pPointerLayout = NULL;
FormatFragment *pMemberInfo = NULL;
CG_ILANALYSIS_INFO *pAnalysisInfo = pNdr->GetILAnalysisInfo();
FormatInfoRef SizeIsDescriptor = INVALID_FRAGMENT_ID;
FormatInfoRef OffsetOfDescriptor = INVALID_FRAGMENT_ID;
FormatInfoRef LengthIsDescriptor = INVALID_FRAGMENT_ID;
if ( pAnalysisInfo->IsConformant() )
{
CG_CONF_ATTRIBUTE *pConfAttribute = dynamic_cast<CG_CONF_ATTRIBUTE*>( pNdr );
expr_node *pSizeIs = pConfAttribute->GetSizeIsExpr();
if ( pSizeIs != NULL )
{
FormatFragment *pSizeIsFrag = GenerateCorrelationDescriptor( pSizeIs );
SizeIsDescriptor = pSizeIsFrag->GetRefID();
}
}
if ( pAnalysisInfo->IsVarying() )
{
CG_VARY_ATTRIBUTE *pVaryAttribute = dynamic_cast<CG_VARY_ATTRIBUTE*>( pNdr );
expr_node *pLengthIs = pVaryAttribute->GetLengthIsExpr();
if ( pLengthIs != NULL )
{
FormatFragment *pLengthIsFrag = GenerateCorrelationDescriptor( pLengthIs );
LengthIsDescriptor = pLengthIsFrag->GetRefID();
}
expr_node *pFirstIs = pVaryAttribute->GetFirstIsExpr();
if ( pFirstIs != NULL )
{
FormatFragment *pOffsetOfFrag = GenerateCorrelationDescriptor( pFirstIs );
OffsetOfDescriptor = pOffsetOfFrag->GetRefID();
}
}
if ( pAnalysisInfo->IsFullBogus() || pAnalysisInfo->IsForcedBogus() )
{
// treat like a bogus array
MIDL_NDR64_CONF_VAR_BOGUS_ARRAY_HEADER_FORMAT *pArrayHeader =
new MIDL_NDR64_CONF_VAR_BOGUS_ARRAY_HEADER_FORMAT( pNdr );
NDR64_BOGUS_ARRAY_HEADER_FORMAT *pFixedArrayHeader = &pArrayHeader->FixedArrayFormat;
pHeaderFragment = pArrayHeader;
bool bIsFullBogus = pAnalysisInfo->IsFullBogus();
NDR64_UINT32 NumberElements = 0;
CG_CONF_ATTRIBUTE *pConfAttribute = dynamic_cast<CG_CONF_ATTRIBUTE *>( pNdr );
if ( !pAnalysisInfo->IsConformant() &&
pConfAttribute &&
pConfAttribute->GetSizeIsExpr() &&
pConfAttribute->GetSizeIsExpr()->IsConstant() )
{
NumberElements = (NDR64_UINT32)pConfAttribute->GetSizeIsExpr()->GetValue();
}
CG_NDR *pChildNdr = dynamic_cast<CG_NDR*>( pNdr->GetChild() );
memset( (NDR64_CONF_VAR_BOGUS_ARRAY_HEADER_FORMAT*)pArrayHeader,
0,
sizeof(NDR64_CONF_VAR_BOGUS_ARRAY_HEADER_FORMAT));
pFixedArrayHeader->FormatCode = (NDR64_FORMAT_CHAR)
( bIsFullBogus ? FC64_BOGUS_ARRAY :
FC64_FORCED_BOGUS_ARRAY );
pFixedArrayHeader->Alignment = ConvertAlignment( pChildNdr->GetWireAlignment() );
pFixedArrayHeader->Flags.HasPointerInfo = false;
pFixedArrayHeader->Flags.HasElementInfo = true;
pFixedArrayHeader->Flags.IsArrayofStrings = pAnalysisInfo->IsArrayofStrings();
pFixedArrayHeader->Flags.IsMultiDimensional = pAnalysisInfo->IsMultiDimensional();
pFixedArrayHeader->NumberDims = pAnalysisInfo->GetDimensions();
pFixedArrayHeader->NumberElements = NumberElements;
pFixedArrayHeader->Element = (PNDR64_FORMAT)
ContinueGenerationInRoot( pChildNdr );
pArrayHeader->ConfDescription = (PNDR64_FORMAT)SizeIsDescriptor;
pArrayHeader->VarDescription = (PNDR64_FORMAT)LengthIsDescriptor;
pArrayHeader->OffsetDescription = (PNDR64_FORMAT)OffsetOfDescriptor;
}
else
{
pPointerLayout = GenSimplePtrLayout( pNdr );
if ( pCommand->NeedsNDR64DebugInfo() )
pMemberInfo = GenerateArrayElementInfo( pNdr->GetChild() );
if ( !pAnalysisInfo->IsConformant() &&
pAnalysisInfo->IsVarying() )
{
MIDL_NDR64_VAR_ARRAY_HEADER_FORMAT *pArrayHeader =
new MIDL_NDR64_VAR_ARRAY_HEADER_FORMAT( pNdr );
pHeaderFragment = pArrayHeader;
CG_NDR *pChildNdr = dynamic_cast<CG_NDR*>(pNdr->GetChild());
memset( (NDR64_VAR_ARRAY_HEADER_FORMAT*)pArrayHeader,
0,
sizeof(NDR64_VAR_ARRAY_HEADER_FORMAT));
pArrayHeader->FormatCode = (NDR64_FORMAT_CHAR) FC64_VAR_ARRAY;
pArrayHeader->Alignment = ConvertAlignment( pChildNdr->GetWireAlignment() );
pArrayHeader->Flags.HasPointerInfo = ( NULL != pPointerLayout );
pArrayHeader->Flags.HasElementInfo = ( NULL != pMemberInfo );
pArrayHeader->Reserved = 0;
pArrayHeader->TotalSize = pNdr->GetMemorySize();
pArrayHeader->ElementSize = pChildNdr->GetMemorySize();
pArrayHeader->VarDescriptor = (PNDR64_FORMAT)LengthIsDescriptor;
pMemberInfo = GenerateArrayElementInfo( pNdr->GetChild() );
}
else if ( pAnalysisInfo->IsConformant() &&
!pAnalysisInfo->IsVarying() )
{
MIDL_NDR64_CONF_ARRAY_HEADER_FORMAT *pArrayHeader =
new MIDL_NDR64_CONF_ARRAY_HEADER_FORMAT( pNdr );
pHeaderFragment = pArrayHeader;
CG_NDR *pChildNdr = dynamic_cast<CG_NDR*>(pNdr->GetChild());
memset( (NDR64_CONF_ARRAY_HEADER_FORMAT*)pArrayHeader,
0,
sizeof(NDR64_CONF_ARRAY_HEADER_FORMAT));
pArrayHeader->FormatCode = (NDR64_FORMAT_CHAR) FC64_CONF_ARRAY;
pArrayHeader->Alignment = ConvertAlignment( pChildNdr->GetWireAlignment() );
pArrayHeader->Flags.HasPointerInfo = (NULL != pPointerLayout);
pArrayHeader->Flags.HasElementInfo = (NULL != pMemberInfo);
pArrayHeader->Reserved = 0;
pArrayHeader->ElementSize = pChildNdr->GetMemorySize();
pArrayHeader->ConfDescriptor = (PNDR64_FORMAT)SizeIsDescriptor;
pMemberInfo = GenerateArrayElementInfo( pNdr->GetChild() );
}
else if ( pAnalysisInfo->IsConformant() &&
pAnalysisInfo->IsVarying() )
{
MIDL_NDR64_CONF_VAR_ARRAY_HEADER_FORMAT *pArrayHeader =
new MIDL_NDR64_CONF_VAR_ARRAY_HEADER_FORMAT( pNdr );
pHeaderFragment = pArrayHeader;
CG_NDR *pChildNdr = dynamic_cast<CG_NDR*>(pNdr->GetChild());
memset( (NDR64_CONF_VAR_ARRAY_HEADER_FORMAT*)pArrayHeader,
0,
sizeof(NDR64_CONF_VAR_ARRAY_HEADER_FORMAT));
pArrayHeader->FormatCode = (NDR64_FORMAT_CHAR) FC64_CONFVAR_ARRAY;
pArrayHeader->Alignment = ConvertAlignment( pChildNdr->GetWireAlignment() );
pArrayHeader->Flags.HasPointerInfo = (NULL != pPointerLayout);
pArrayHeader->Flags.HasElementInfo = (NULL != pMemberInfo);
pArrayHeader->Reserved = 0;
pArrayHeader->ElementSize = pChildNdr->GetMemorySize();
pArrayHeader->ConfDescriptor = (PNDR64_FORMAT)SizeIsDescriptor;
pArrayHeader->VarDescriptor = (PNDR64_FORMAT)LengthIsDescriptor;
}
else
{
// !pAnalysisInfo->IsConformant && !pAnalysisInfo->IsVarying
MIDL_ASSERT(0);
pHeaderFragment = NULL;
}
}
pMainFragment->AddFragment( pHeaderFragment );
if ( pPointerLayout )
pMainFragment->AddFragment( pPointerLayout );
if ( pMemberInfo )
pMainFragment->AddFragment( pMemberInfo );
return pMainFragment;
}
void GenNdr64Format::GenerateNonStringQualifiedArray( CG_ARRAY *pArray )
{
CompositeFormatFragment *pContainer;
MIDL_NDR64_POINTER_FORMAT* pPointerHdr = GenQualifiedArrayPtr( pArray );
if ( pPointerHdr != NULL)
{
GetCurrent()->AddFragment( pPointerHdr );
pContainer = GetRoot();
}
else
pContainer = GetCurrent();
FormatFragment *pArrayFragment =
GenerateNonStringQualifiedArrayLayout( pArray, pContainer );
if ( NULL != pPointerHdr )
{
pPointerHdr->Pointee = pArrayFragment->GetRefID();
}
}
void GenNdr64Format::InitStringHeader( CG_NDR *pString,
NDR64_STRING_HEADER_FORMAT *pHeader,
bool bIsConformant,
bool bIsSized )
{
NDR64_FORMAT_CHAR FormatCode;
CG_BASETYPE *pBT = dynamic_cast<CG_BASETYPE*>( pString->GetChild() );
if ( NULL != pBT )
{
switch( pBT->GetType()->GetBasicType()->NodeKind() )
{
case NODE_BYTE:
case NODE_CHAR:
FormatCode = (NDR64_FORMAT_CHAR)
( bIsConformant ? FC64_CONF_CHAR_STRING : FC64_CHAR_STRING );
break;
case NODE_WCHAR_T:
FormatCode = (NDR64_FORMAT_CHAR)
( bIsConformant ? FC64_CONF_WCHAR_STRING : FC64_WCHAR_STRING );
break;
default:
FormatCode = (NDR64_FORMAT_CHAR)
( bIsConformant ? FC64_CONF_STRUCT_STRING : FC64_STRUCT_STRING );
break;
}
}
else
FormatCode = (NDR64_FORMAT_CHAR)
( bIsConformant ? FC64_CONF_STRUCT_STRING : FC64_STRUCT_STRING );
CG_NDR *pChildNdr = dynamic_cast<CG_NDR*>( pString->GetChild() );
MIDL_ASSERT( pChildNdr->GetMemorySize() <= 0xFFFF );
NDR64_UINT16 ElementSize = (NDR64_UINT16)pChildNdr->GetMemorySize();
MIDL_ASSERT( (FC64_CHAR_STRING == FormatCode) ? (1 == ElementSize) : 1 );
MIDL_ASSERT( (FC64_WCHAR_STRING == FormatCode) ? (2 == ElementSize) : 1 );
memset( pHeader, 0, sizeof(NDR64_STRING_HEADER_FORMAT));
pHeader->FormatCode = FormatCode;
pHeader->Flags.IsSized = bIsSized;
pHeader->ElementSize = ElementSize;
}
void GenNdr64Format::GenerateStringArray( CG_ARRAY *pArray,
bool bIsSized )
{
MIDL_NDR64_POINTER_FORMAT* pPointerHdr = GenQualifiedArrayPtr( pArray );
if ( pPointerHdr != NULL)
GetCurrent()->AddFragment( pPointerHdr );
FormatFragment *pStringFrag = NULL;
if ( bIsSized )
{
CG_CONF_ATTRIBUTE *pConfAttribute = dynamic_cast<CG_CONF_ATTRIBUTE*>( pArray );
expr_node *pSizeIs = pConfAttribute->GetSizeIsExpr();
if ( NULL != pSizeIs )
{
FormatFragment *pFrag = GenerateCorrelationDescriptor( pConfAttribute->GetSizeIsExpr() );
MIDL_NDR64_SIZED_CONFORMANT_STRING_FORMAT *pSizedConfFormat =
new MIDL_NDR64_SIZED_CONFORMANT_STRING_FORMAT( pArray );
pStringFrag = pSizedConfFormat;
InitStringHeader( pArray, &pSizedConfFormat->Header, true, true);
pSizedConfFormat->SizeDescription = (PNDR64_FORMAT)pFrag->GetRefID();
}
else
{
MIDL_NDR64_CONFORMANT_STRING_FORMAT *pConfFormat =
new MIDL_NDR64_CONFORMANT_STRING_FORMAT( pArray );
pStringFrag = pConfFormat;
InitStringHeader( pArray, &pConfFormat->Header, true, false);
}
}
else
{
MIDL_NDR64_NON_CONFORMANT_STRING_FORMAT *pNonConfFormat =
new MIDL_NDR64_NON_CONFORMANT_STRING_FORMAT( pArray );
pStringFrag = pNonConfFormat;
InitStringHeader( pArray, &pNonConfFormat->Header, false, false);
pNonConfFormat->TotalSize = pArray->GetMemorySize();
}
if ( NULL != pPointerHdr )
{
GetRoot()->AddFragment( pStringFrag );
pPointerHdr->Pointee = pStringFrag->GetRefID();
}
else
{
GetCurrent()->AddFragment( pStringFrag );
}
}
class ExpressionGenerator
{
public:
static CompositeFormatFragment *
Generate( CCB * pCCB,
expr_node * pSizeExpr
);
private:
static void
GenExprPadIfNecessary(
CompositeFormatFragment * FragmentList,
ulong * pExprLength,
ulong Align );
static FormatFragment *
GenExprConstant(
NDR64_FORMAT_CHARACTER fc,
EXPR_VALUE lValue,
CompositeFormatFragment * FragmentList,
ulong * pExprLength );
static BOOL
IsConstantExpr (expr_node * pExpr );
static NDR64_FORMAT_CHARACTER
GetTypeForExpression( node_skl *pType );
static FormatFragment *
GenExprFormatString(
CCB *pCCB,
expr_node *pSizeExpr,
CompositeFormatFragment *FragmentList,
BOOL * pIsEarly,
ulong * pExprLength );
static CG_FIELD*
FindField(
node_skl * pFieldType,
CG_STRUCT *pStruct,
const char *pPrintPrefix,
unsigned long *pMemOffset );
static void
ComputeFieldCorrelationOffset(
CCB *pCCB,
node_skl *pFieldType,
// return parameters
CG_FIELD **ppVariableField, // var in expression
long *pOffset,
BOOL *pIsEarly );
};
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenerateCorrelationDescriptor
//
// Synopsis:
//
//---------------------------------------------------------------------------
FormatFragment*
GenNdr64Format::GenerateCorrelationDescriptor( expr_node *pExpr )
{
MIDL_ASSERT(NULL != pExpr);
FormatFragment *pCorrelationDescriptor =
ExpressionGenerator::Generate( pCCB, pExpr );
GetRoot()->AddFragment( pCorrelationDescriptor );
GetRoot()->OptimizeFragment( pCorrelationDescriptor );
return pCorrelationDescriptor;
}
//+--------------------------------------------------------------------------
//
// Method: Generate
//
// Synopsis: generate correlation expression.
//
//
//---------------------------------------------------------------------------
CompositeFormatFragment *
ExpressionGenerator::Generate( CCB * pCCB,
expr_node * pSizeExpr
)
{
if ( NULL == pSizeExpr )
return NULL ;
#if defined(DBG)
CG_CLASS *pAttributedNode = pCCB->GetLastPlaceholderClass( );
MIDL_ASSERT( ( NULL != dynamic_cast<CG_PARAM*>( pAttributedNode ) ) ||
( NULL != dynamic_cast<CG_FIELD*>( pAttributedNode ) ) );
#endif
BOOL IsEarly = FALSE;
CompositeFormatFragment * pExprComposite = new CompositeFormatFragment;
MIDL_NDR_FORMAT_UINT32 * pFlag = new MIDL_NDR_FORMAT_UINT32;
ulong ulExprLength = 0;
// correlation flags. 4 bytes.
pExprComposite->AddFragment( pFlag );
ulExprLength = sizeof( NDR64_UINT32 );
// recursive code to generate the expression stack.
GenExprFormatString( pCCB, pSizeExpr, pExprComposite, &IsEarly, &ulExprLength );
pFlag->Data = 0;
if ( IsEarly )
pFlag->Data |= FC_NDR64_EARLY_CORRELATION;
return pExprComposite;
}
void
ExpressionGenerator::GenExprPadIfNecessary(
CompositeFormatFragment * FragmentList,
ulong * pExprLength,
ulong Align )
{
MIDL_NDR64_EXPR_NOOP * pExprPad = NULL;
if ( *pExprLength & ( Align - 1 ) )
{
// need to add padding to align next element
pExprPad = new MIDL_NDR64_EXPR_NOOP;
pExprPad->Size = (NDR64_UINT8) ( ( Align - 1 ) & *pExprLength );
FragmentList->AddFragment( pExprPad );
*pExprLength += sizeof( NDR64_EXPR_NOOP );
}
else
return;
}
FormatFragment *
ExpressionGenerator::GenExprConstant(
NDR64_FORMAT_CHARACTER fc,
EXPR_VALUE lValue,
CompositeFormatFragment * FragmentList,
ulong * pExprLength )
{
if ( fc == FC64_INT64 )
{
MIDL_NDR64_EXPR_CONST64 * pFormat;
pFormat = new MIDL_NDR64_EXPR_CONST64;
pFormat->ConstValue = lValue;
GenExprPadIfNecessary( FragmentList, pExprLength, 8 );
FragmentList->AddFragment( pFormat );
*pExprLength += sizeof( NDR64_EXPR_CONST64 );
return pFormat;
}
else
{
MIDL_NDR64_EXPR_CONST32 *pFormat;
pFormat = new MIDL_NDR64_EXPR_CONST32;
pFormat->ConstValue = (NDR64_UINT32) lValue;
GenExprPadIfNecessary( FragmentList, pExprLength, 4 );
FragmentList->AddFragment( pFormat );
*pExprLength += sizeof( NDR64_EXPR_CONST64 );
return pFormat;
}
}
// check if the expression (including all sub expressions ) is a constant expression
BOOL
ExpressionGenerator::IsConstantExpr (expr_node * pExpr )
{
if ( pExpr->IsConstant() )
return TRUE;
if ( pExpr->IsAVariable() )
return FALSE;
if ( pExpr->IsBinaryOperator() )
{
return IsConstantExpr( ((expr_op_binary *)pExpr)->GetLeft() ) &&
IsConstantExpr( ((expr_op_binary *)pExpr)->GetRight() );
}
else
{
if ( pExpr->IsUnaryOperator() )
{
return IsConstantExpr( ((expr_op_unary *)pExpr)->GetLeft() );
}
else
{
MIDL_ASSERT( pExpr->IsRelationalOperator() );
return IsConstantExpr( ((expr_ternary *)pExpr)->GetLeft() ) &&
IsConstantExpr( ((expr_ternary *)pExpr)->GetRight() ) &&
IsConstantExpr( ((expr_ternary *)pExpr)->GetRelational() ) ;
}
}
}
NDR64_FORMAT_CHARACTER
ExpressionGenerator::GetTypeForExpression(
node_skl *pType
)
{
// Get the size of the type.
unsigned long Size = pType->GetSize();
// determine if the type is signed or unsigned.
bool IsSigned = true;
node_base_type *pBaseType = dynamic_cast<node_base_type*>( pType->GetBasicType() );
if ( pBaseType )
{
IsSigned = !pBaseType->IsUnsigned();
}
switch ( Size )
{
case 1:
return IsSigned ? FC64_INT8 : FC64_UINT8;
case 2:
return IsSigned ? FC64_INT16 : FC64_UINT16;
case 4:
return IsSigned ? FC64_INT32 : FC64_UINT32;
case 8:
return IsSigned ? FC64_INT64 : FC64_UINT64;
default:
RpcError( NULL, 0, EXPR_NOT_EVALUATABLE, pType->GetSymName() );
return FC64_ZERO; // Keep the compiler happy
}
}
FormatFragment *
ExpressionGenerator::GenExprFormatString(
CCB *pCCB,
expr_node *pSizeExpr,
CompositeFormatFragment *FragmentList,
BOOL * pIsEarly,
ulong * pExprLength )
{
node_skl * pAttributeNodeType;
EXPR_TOKEN Op;
long Offset;
CG_PARAM * pParam = NULL;
CG_NDR * pSwitchNode = NULL;
// BUGBUG: how to generate 32bit constant?
if ( IsConstantExpr( pSizeExpr ) )
{
// Constant expressions are always early!
*pIsEarly = TRUE;
return GenExprConstant( FC64_INT64, pSizeExpr->GetValue(), FragmentList, pExprLength );
}
if ( pSizeExpr->IsAVariable() )
// variable
{
Op = FC_EXPR_VAR;
MIDL_NDR64_EXPR_VAR * pFormat;
CG_NDR * pNdr = pCCB->GetCGNodeContext();
CG_ITERATOR Iterator;
pAttributeNodeType = pSizeExpr->GetType();
if ( pNdr->IsProc() )
// top level param
{
CG_PARAM* pCurrentParam = (CG_PARAM*) pCCB->GetCurrentParam();
CG_PROC* pCurrentProc = (CG_PROC*) pNdr;
*pIsEarly = TRUE;
if ( ( (node_param *) pAttributeNodeType )->IsSaveForAsyncFinish() &&
pCurrentProc->IsFinishProc() )
{
// This is an async split where the finish proc is using
// a param from the begin proc. Since the parameter
// was stored on the split stack, the expression will
// always be early correlation. So all that is needed
// here is to find the parameter in the begin proc.
CG_PROC* pBeginProc = pCurrentProc->GetAsyncRelative();
pBeginProc->GetMembers( Iterator );
for (;;)
{
if (!ITERATOR_GETNEXT( Iterator, pParam ))
{
// Didn't find the parameter. We are in trouble!
MIDL_ASSERT(0);
return NULL;
}
if ( pParam->GetType() == pAttributeNodeType )
{
break;
}
}
}
else
{
// Typical case of correlation in the same parameter.
pCurrentProc->GetMembers( Iterator);
if ( pCurrentParam->GetType() == pAttributeNodeType )
*pIsEarly = FALSE;
for(;;)
{
if (!ITERATOR_GETNEXT( Iterator, pParam ))
{
// Didn't find the parameter. We are in trouble!
MIDL_ASSERT(0);
return NULL;
}
// If we find the current parameter before the variable,
// parameter, then late correlation is needed.
if ( pParam == pCurrentParam )
*pIsEarly = FALSE;
if ( pParam->GetType() == pAttributeNodeType )
break;
}
}
pSwitchNode = (CG_NDR *) pParam->GetChild();
}
else
// structure /union etc.
{
CG_FIELD *pSwitchField;
ComputeFieldCorrelationOffset( pCCB,
pAttributeNodeType,
// return parameters
&pSwitchField,
&Offset,
pIsEarly );
pSwitchNode = (CG_NDR*)pSwitchField->GetChild();
}
// Code the type of the size_is etc. expression.
NDR64_FORMAT_CHARACTER Type = GetTypeForExpression( pSwitchNode->GetType() );
pAttributeNodeType = pSizeExpr->GetType();
pFormat = new MIDL_NDR64_EXPR_VAR;
pFormat->ExprType = FC_EXPR_VAR;
pFormat->VarType = (NDR64_UINT8)Type;
if ( pNdr->IsProc() )
{
CG_NDR* pOld = 0;
BOOL IsFinish = FALSE;
MIDL_NDR64_EXPR_OPERATOR * pAsyncOp;
if ( ( (node_param *) pAttributeNodeType )->IsSaveForAsyncFinish() )
{
pAsyncOp = new MIDL_NDR64_EXPR_OPERATOR;
if ( ( (CG_PROC *)pNdr )->IsFinishProc() )
{
pOld = pCCB->SetCGNodeContext( ( (CG_PROC *)pNdr )->GetAsyncRelative() );
IsFinish = TRUE;
}
pAsyncOp->ExprType = (NDR64_FORMAT_CHAR) FC_EXPR_OPER;
pAsyncOp->Operator = (NDR64_FORMAT_CHAR) OP_ASYNCSPLIT;
FragmentList->AddFragment( pAsyncOp );
*pExprLength += sizeof( NDR64_EXPR_OPERATOR );
}
pFormat->fStackOffset = TRUE;
pFormat->ia64Offset = pParam->GetStackOffset( pCCB, I386_STACK_SIZING );
if ( IsFinish )
pCCB->SetCGNodeContext( pOld );
}
else
{
// Structure
pFormat->fStackOffset = FALSE;
pFormat->Offset = Offset;
}
GenExprPadIfNecessary( FragmentList, pExprLength, 4 );
FragmentList->AddFragment( pFormat );
*pExprLength += sizeof( NDR64_EXPR_VAR );
return pFormat;
}
else
// operator
{
expr_node * pLeftExpr = NULL;
BOOL bLeftExprIsEarly = TRUE;
expr_node * pRightExpr = NULL;
BOOL bRightExprIsEarly = TRUE;
expr_node * pRationalExpr = NULL;
BOOL bRationalExprIsEarly = TRUE;
MIDL_NDR64_EXPR_OPERATOR *pFormat;
NDR64_FORMAT_CHARACTER fcKind = (NDR64_FORMAT_CHARACTER)0;
OPERATOR Operator ;
if ( pSizeExpr->IsBinaryOperator() )
{
Operator = ((expr_op_binary *)pSizeExpr)->GetOperator();
pLeftExpr = ((expr_op_binary *)pSizeExpr)->GetLeft();
pRightExpr = ((expr_op_binary *)pSizeExpr)->GetRight();
}
else
{
if ( pSizeExpr->IsUnaryOperator() )
{
Operator = ((expr_op_unary *)pSizeExpr)->GetOperator();
pLeftExpr = ((expr_op_unary *)pSizeExpr)->GetLeft();
}
else
{
Operator = ((expr_ternary *)pSizeExpr)->GetOperator();
pLeftExpr = ((expr_ternary *)pSizeExpr)->GetLeft();
pRightExpr = ((expr_ternary *)pSizeExpr)->GetRight();
pRationalExpr = ((expr_ternary *)pSizeExpr)->GetRelational();
}
}
switch ( Operator )
{
case OP_UNARY_SIZEOF:
EXPR_VALUE lSize;
lSize = pSizeExpr->GetType()->GetSize();
*pIsEarly = TRUE;
return GenExprConstant( FC64_INT8, lSize, FragmentList, pExprLength);
case OP_UNARY_CAST:
case OP_UNARY_INDIRECTION:
fcKind = GetTypeForExpression( pSizeExpr->GetType() );
break;
}
pFormat = new MIDL_NDR64_EXPR_OPERATOR;
pFormat->ExprType = (NDR64_FORMAT_CHAR) FC_EXPR_OPER;
pFormat->Operator = (NDR64_FORMAT_CHAR) Operator;
// This was initially 0
pFormat->CastType = (NDR64_FORMAT_CHAR) fcKind;
FragmentList->AddFragment( pFormat );
*pExprLength += sizeof( NDR64_EXPR_OPERATOR );
GenExprFormatString( pCCB, pLeftExpr, FragmentList, &bLeftExprIsEarly, pExprLength );
if ( pRightExpr )
GenExprFormatString( pCCB, pRightExpr, FragmentList, &bRightExprIsEarly, pExprLength );
if ( pRationalExpr )
GenExprFormatString( pCCB, pRationalExpr, FragmentList, &bRationalExprIsEarly, pExprLength );
//
// An operator is early if and only if all the arguments to the
// operator are early.
//
*pIsEarly = bLeftExprIsEarly && bRightExprIsEarly && bRationalExprIsEarly;
return pFormat;
}
}
CG_FIELD*
ExpressionGenerator::FindField(
node_skl * pFieldType,
CG_STRUCT *pStruct,
const char *pPrintPrefix,
unsigned long *pMemOffset )
{
CG_ITERATOR Iterator;
pStruct->GetMembers( Iterator );
ITERATOR_INIT( Iterator );
CG_FIELD *pField;
while ( ITERATOR_GETNEXT( Iterator, pField ) )
{
// First check if the fields are the same type. If they
// are, then check the print prefixes to make sure the fields came
// from the same structure.
if ( ( pField->GetType() == pFieldType ) &&
( strcmp( pField->GetPrintPrefix(), pPrintPrefix ) == 0 ) )
{
*pMemOffset = pField->GetMemOffset();
return pField;
}
CG_CLASS * pChildClass = pField->GetChild();
if ( pChildClass->IsStruct() )
{
unsigned long TempMemOffset;
CG_STRUCT *pChildStruct = (CG_STRUCT*)pChildClass;
CG_FIELD *pTempField = FindField( pFieldType,
pChildStruct,
pPrintPrefix,
&TempMemOffset );
if ( NULL != pTempField )
{
*pMemOffset = TempMemOffset + pField->GetMemOffset();
return pTempField;
}
}
}
return NULL;
}
void
ExpressionGenerator::ComputeFieldCorrelationOffset(
CCB *pCCB,
node_skl *pFieldType,
// return parameters
CG_FIELD **ppVariableField, // var in expression
long *pOffset,
BOOL *pIsEarly )
{
//
// Find the fields.
//
CG_FIELD *pCurrentField = dynamic_cast<CG_FIELD*>( pCCB->GetLastPlaceholderClass() );
MIDL_ASSERT( NULL != pCurrentField );
const char *pPrintPrefix = pCurrentField->GetPrintPrefix();
CG_STRUCT *pContext = dynamic_cast<CG_STRUCT*>( pCCB->GetCGNodeContext() );
MIDL_ASSERT( NULL != pContext );
unsigned long VariableOffset;
CG_FIELD* pVariableField = FindField( pFieldType,
pContext,
pPrintPrefix,
&VariableOffset );
MIDL_ASSERT( NULL != pVariableField );
*ppVariableField = pVariableField;
unsigned long CurrentOffset;
CG_FIELD* pAlsoCurrentField = FindField( pCurrentField->GetType(),
pContext,
pPrintPrefix,
&CurrentOffset );
pAlsoCurrentField;
MIDL_ASSERT( pAlsoCurrentField == pCurrentField );
//
// Determine the correlation type.
// The correlation type is early if the variable field will
// be completely marshalled before the current field.
//
BOOL bVariableIsPointer = pVariableField->GetChild()->IsPointer();
BOOL bCurrentFieldIsPointer = pCurrentField->GetChild()->IsPointer();
BOOL bOnlyVariableIsPointer = bVariableIsPointer && !bCurrentFieldIsPointer;
*pIsEarly = ( CurrentOffset > VariableOffset ) && !bOnlyVariableIsPointer;
// In the land of the new transfer syntax, all offsets are a positive offset
// from the start of the stack or the last structure/region that was passed.
CG_FIELD *pRegionField = pCCB->GetCurrentRegionField();
// Make offset relative to start of region
if (NULL != pRegionField)
VariableOffset -= pRegionField->GetMemOffset();
*pOffset = VariableOffset;
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_TRANSMIT_AS )
//
// Synopsis: Generate info for transmit_as types
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_TRANSMIT_AS *pTransmitAs )
{
MIDL_NDR64_TRANSMIT_AS_FORMAT *format;
format = new MIDL_NDR64_TRANSMIT_AS_FORMAT( pTransmitAs );
GetCurrent()->AddFragment( format );
GenXmitOrRepAsFormat(
pTransmitAs,
format,
pTransmitAs->GetPresentedType()->GetSymName(),
pTransmitAs->GetPresentedType(),
pTransmitAs->GetTransmittedType() );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_REPRESENT_AS )
//
// Synopsis: Generate info for represent_as types
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_REPRESENT_AS *pRepresentAs )
{
MIDL_NDR64_REPRESENT_AS_FORMAT *format;
format = new MIDL_NDR64_REPRESENT_AS_FORMAT( pRepresentAs );
GetCurrent()->AddFragment( format );
GenXmitOrRepAsFormat(
pRepresentAs,
format,
pRepresentAs->GetRepAsTypeName(),
pRepresentAs->GetRepAsType(),
pRepresentAs->GetTransmittedType() );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::GenXmitOrRepAsFormat
//
// Synopsis: Do the actual work for transmit_as and represent_as types
//
//---------------------------------------------------------------------------
void GenNdr64Format::GenXmitOrRepAsFormat(
CG_TYPEDEF *pXmitNode,
MIDL_NDR64_TRANSMIT_AS_FORMAT *format,
char *pPresentedTypeName,
node_skl *pPresentedType,
node_skl *pTransmittedType )
{
CG_NDR *pChild = (CG_NDR *) pXmitNode->GetChild();
BOOL fXmit = ( NULL == dynamic_cast<CG_REPRESENT_AS *>(pXmitNode) );
NDR64_UINT16 RoutineIndex = pXmitNode->GenXmitOrRepAsQuintuple(
pCCB,
fXmit,
pXmitNode,
pPresentedTypeName,
(fXmit ? pPresentedType
: pTransmittedType) );
format->FormatCode = (NDR64_FORMAT_CHAR) ( fXmit ? FC64_TRANSMIT_AS : FC64_REPRESENT_AS );
format->RoutineIndex = RoutineIndex;
format->TransmittedTypeWireAlignment = (NDR64_UINT16) ( pChild->GetWireAlignment() - 1 );
// REVIEW: The spec doesn't say whether "MemoryAlignment" is the
// presented type or the transmitted type. Transmitted doesn't
// make much sense but on the other hand presented has some
// alignment flags already.
format->MemoryAlignment = pXmitNode->GetMemoryAlignment();
if ( pPresentedType )
format->PresentedTypeMemorySize = (NDR64_UINT16) pPresentedType->GetSize();
else
MIDL_ASSERT(!"BUGBUG: unknown rep/transmit_as");
if ( pChild->HasAFixedBufferSize() )
format->TransmittedTypeBufferSize = (NDR64_UINT16) pChild->GetWireSize();
else
format->TransmittedTypeBufferSize = 0;
format->Flags.PresentedTypeIsArray = 0;
format->Flags.PresentedTypeAlign4 = 0;
format->Flags.PresentedTypeAlign8 = 0;
format->Flags.Reserved = 0;
if ( pPresentedType )
{
if ( pPresentedType->GetBasicType()->NodeKind() == NODE_ARRAY )
format->Flags.PresentedTypeIsArray = 1;
else
{
if ( pXmitNode->GetMemoryAlignment() == 4 )
format->Flags.PresentedTypeAlign4 = 1;
else if ( pXmitNode->GetMemoryAlignment() == 8 )
format->Flags.PresentedTypeAlign8 = 1;
}
}
if ( pChild->GetCGID() == ID_CG_GENERIC_HDL )
pChild = (CG_NDR *)pChild->GetChild();
format->TransmittedType = ContinueGenerationInRoot( pChild );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_USER_MARSHAL )
//
// Synopsis: Generate info for user_marshal types
//
// REVIEW: user_marshall and represent_as are so close to one another
// they really should probably be merged
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_USER_MARSHAL *pUserMarshal )
{
MIDL_NDR64_USER_MARSHAL_FORMAT *format;
format = new MIDL_NDR64_USER_MARSHAL_FORMAT( pUserMarshal );
GetCurrent()->AddFragment( format );
CG_NDR *pChild = (CG_NDR *) pUserMarshal->GetChild();
format->FormatCode = FC64_USER_MARSHAL;
format->TransmittedTypeWireAlignment = (NDR64_UINT16) (pChild->GetWireAlignment() - 1);
format->MemoryAlignment = pChild->GetMemoryAlignment();
memset( &format->Flags, 0, sizeof(format->Flags) );
if ( pChild->IsPointer() )
{
CG_POINTER *pPointer = (CG_POINTER *) pChild;
MIDL_ASSERT( ! pPointer->IsFull() );
if ( pPointer->IsUnique() )
format->Flags.UniquePointer = 1;
else if ( pPointer->IsRef() )
format->Flags.RefPointer = 1;
}
format->Flags.IID = 0; // Only used by JIT compiler
format->Flags.Reserved = 0;
USER_MARSHAL_CONTEXT * pUserMarshalContext = new USER_MARSHAL_CONTEXT;
pUserMarshalContext->pTypeName = pUserMarshal->GetRepAsTypeName();
pUserMarshalContext->pType = pUserMarshal->GetRepAsType();
BOOL Added = pCCB->GetQuadrupleDictionary()->Add( pUserMarshalContext );
format->RoutineIndex = pUserMarshalContext->Index;
if ( !Added )
delete pUserMarshalContext;
if ( pUserMarshal->GetRepAsType() )
format->UserTypeMemorySize = pUserMarshal->GetRepAsType()->GetSize();
else
MIDL_ASSERT( !"BUGBUG: undefined user_marshal" );
if ( pChild->HasAFixedBufferSize() )
format->TransmittedTypeBufferSize = pChild->GetWireSize();
else
format->TransmittedTypeBufferSize = 0;
format->TransmittedType = ContinueGenerationInRoot( pChild );
}
//+--------------------------------------------------------------------------
//
// Method: GenNdr64Format::Visit( CG_PIPE )
//
// Synopsis: Generate info for [pipe] types
//
//---------------------------------------------------------------------------
void GenNdr64Format::Visit( CG_PIPE *pPipe )
{
pCommand->GetNdrVersionControl().SetHasRawPipes();
MIDL_NDR64_PIPE_FORMAT *format = new MIDL_NDR64_PIPE_FORMAT( pPipe );
GetCurrent()->AddFragment( format );
CG_NDR *pChild = (CG_NDR *) pPipe->GetChild();
NDR64_UINT32 BufferSize = 0;
node_range_attr *range = pPipe->GetRangeAttribute();
NDR64_UINT32 MinValue = 0;
NDR64_UINT32 MaxValue = 0;
CG_ILANALYSIS_INFO *pAnalysisInfo = pChild->GetILAnalysisInfo();
if ( pChild->HasAFixedBufferSize() )
BufferSize = pChild->GetWireSize();
if ( range )
{
MinValue = (NDR64_UINT32) range->GetMinExpr()->GetValue();
MaxValue = (NDR64_UINT32) range->GetMaxExpr()->GetValue();
}
format->FormatCode = FC64_PIPE;
format->Flags.Reserved1 = 0;
format->Flags.HasRange = (bool) ( NULL != range );
format->Flags.BlockCopy = !pAnalysisInfo->IsForcedBogus() && !pAnalysisInfo->IsFullBogus();
format->Flags.Reserved2 = 0;
format->Alignment = (NDR64_UINT8) ( pChild->GetWireAlignment() - 1 );
format->Reserved = 0;
format->Type = ContinueGenerationInRoot( pChild );
format->MemorySize = pChild->GetMemorySize();
format->BufferSize = BufferSize;
format->MinValue = MinValue;
format->MaxValue = MaxValue;
}
//+--------------------------------------------------------------------------
//
// Method: OutputParamFlagDescription
//
// Synopsis: Output a description string for the ndr64 param flags
//
//---------------------------------------------------------------------------
void OutputParamFlagDescription( CCB *pCCB, const NDR64_PARAM_FLAGS &flags )
{
static const PNAME flag_descrip[16] =
{
"MustSize",
"MustFree",
"pipe",
"[in]",
"[out]",
"IsReturn",
"Basetype",
"ByValue",
"SimpleRef",
"DontFreeInst",
"AsyncFinish",
NULL,
NULL,
NULL,
NULL,
"UseCache"
};
pCCB->GetStream()->Write( " " );
OutputFlagDescriptions( pCCB->GetStream(), &flags, sizeof(flags), flag_descrip );
}
//+--------------------------------------------------------------------------
//
// Method: OutputFlagDescription
//
// Synopsis: Given a set of flags and a description for each flags,
// output the corresponding description for flag that is set.
// usused flags can be marked with a NULL description.
//
// Notes: Assumes little-endian memory layout!
//
//---------------------------------------------------------------------------
void OutputFlagDescriptions(
ISTREAM *stream,
const void *pvFlags,
int bytes,
const PNAME *description)
{
unsigned char *flags = (unsigned char *) pvFlags;
bool first = true;
stream->Write( "/*");
for (int i = 0; i < (bytes * 8); i++)
{
if ( ( NULL == description[i] )
|| ( 0 == ( flags[i / 8] & ( 1 << (i % 8) ) ) ) )
{
continue;
}
if ( !first )
stream->Write( "," );
first = false;
stream->Write( " " );
stream->Write( description[i] );
}
stream->Write( " */" );
}