|
|
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Copyright (c) 1993 Microsoft Corporation
Module Name :
mrshlp.c
Abstract :
This file contains the routines for marshalling an array's or a structure's embedded pointers and for computing conformance and variance counts and union switch values.
Author :
David Kays dkays September 1993.
Revision History :
---------------------------------------------------------------------*/
#include "ndrp.h"
#include "attack.h"
#include "interp2.h"
#include "mulsyntx.h"
#include "asyncu.h"
#include "pointerq.h"
PFORMAT_STRING NdrpEmbeddedPointerMarshall( PMIDL_STUB_MESSAGE pStubMsg, uchar * pMemory, PFORMAT_STRING pFormat ) /*++
Routine Description :
Marshalls an array's or a structure's embedded pointers.
Arguments :
pStubMsg - Pointer to the stub message. pMemory - Pointer to the structure or array whose embedded pointers are being marshalled. pFormat - The format string pointer layout. Should point to the pointer layout's beginning FC_PP character upon entry.
Return :
Format string pointer after the pointer layout.
--*/ { uchar ** ppMemPtr; uchar * pBufPtr; uchar * pBufferMark; uchar * pMemorySave; ULONG_PTR MaxCountSave; long OffsetSave;
MaxCountSave = pStubMsg->MaxCount; OffsetSave = pStubMsg->Offset; POINTER_BUFFER_SWAP_CONTEXT SwapContext(pStubMsg);
pMemorySave = pStubMsg->Memory;
// This is where the embedding structure or array begins in the buffer.
pBufferMark = pStubMsg->BufferMark;
//
// The Memory field in the stub message keeps track of the pointer to
// the current embedding structure or array. This is needed to handle
// size/length pointers, so that we can get a pointer to the current
// embedding struct when computing conformance and variance.
//
pStubMsg->Memory = pMemory;
// Skip FC_PP and FC_PAD.
pFormat += 2;
for (;;) {
if ( *pFormat == FC_END ) { pStubMsg->Memory = pMemorySave;
return pFormat; }
//
// Check for FC_FIXED_REPEAT and FC_VARIABLE_REPEAT.
//
if ( *pFormat != FC_NO_REPEAT ) { pStubMsg->MaxCount = MaxCountSave; pStubMsg->Offset = OffsetSave;
pStubMsg->BufferMark = pBufferMark;
pFormat = NdrpEmbeddedRepeatPointerMarshall( pStubMsg, pMemory, pFormat ); // Continue to the next pointer.
continue; }
// Compute the pointer to the pointer to marshall.
ppMemPtr = (uchar **)(pMemory + *((signed short *)(pFormat + 2)));
//
// Compute the location in the buffer where the pointer's value will be
// marshalled. Needed for full pointers.
//
pBufPtr = pBufferMark + *((signed short *)(pFormat + 4));
// Increment to the pointer description.
pFormat += 6;
//
// Now marshall the pointer.
//
NdrpPointerMarshall( pStubMsg, pBufPtr, *ppMemPtr, pFormat );
// Increment to the next pointer description.
pFormat += 4;
} // for
}
PFORMAT_STRING NdrpEmbeddedRepeatPointerMarshall( PMIDL_STUB_MESSAGE pStubMsg, uchar * pMemory, PFORMAT_STRING pFormat ) /*++
Routine Description :
Marshalls an array's embedded pointers.
Arguments :
pStubMsg - Pointer to the stub message. pMemory - Array whose embedded pointers are being marshalled. pFormat - Pointer layout format string description.
Return :
Format string pointer after the pointer layout.
--*/ { uchar ** ppMemPtr; uchar * pBufPtr; PFORMAT_STRING pFormatSave; uchar * pMemorySave; uchar * pBufferMark; ulong RepeatCount, RepeatIncrement, Pointers, PointersSave;
pMemorySave = pStubMsg->Memory;
//
// This is where the current embedding structure or array begins in
// the buffer.
//
pBufferMark = pStubMsg->BufferMark;
// Get the number of shipped elements in the array.
switch ( *pFormat ) { case FC_FIXED_REPEAT : pFormat += 2;
RepeatCount = *((ushort *)pFormat);
break;
case FC_VARIABLE_REPEAT : RepeatCount = (ulong)pStubMsg->MaxCount;
//
// Check if this variable repeat instance also has a variable
// offset (this would be the case for a conformant varying array
// of pointers). If so then increment the memory pointer by the
// increment size time the variance offset.
//
if ( pFormat[1] == FC_VARIABLE_OFFSET ) pMemory += *((ushort *)(pFormat + 2)) * pStubMsg->Offset;
// else pFormat[1] == FC_FIXED_OFFSET - do nothing
break;
default : NDR_ASSERT(0,"NdrpEmbeddedRepeatPointerMarshall : bad format char"); RpcRaiseException( RPC_S_INTERNAL_ERROR ); return 0; }
// Increment format string to increment field.
pFormat += 2;
// Get the increment amount between successive pointers.
// This is actually an increment over the array element.
RepeatIncrement = *((ushort * &)pFormat)++;
//
// Add the offset to the beginning of this array to the Memory
// pointer. This is the offset from the current embedding structure
// or array to the array whose pointers we're marshalling.
//
pStubMsg->Memory += *((ushort * &)pFormat)++;
// Get the number of pointers in this repeat instance.
PointersSave = Pointers = *((ushort * &)pFormat)++;
pFormatSave = pFormat;
//
// Loop over the number of shipped elements of the array.
//
for ( ; RepeatCount--; pBufferMark += RepeatIncrement, pMemory += RepeatIncrement, pStubMsg->Memory += RepeatIncrement ) { pFormat = pFormatSave; Pointers = PointersSave;
//
// Loop over the number of pointer per array element (could be
// greater than one for an array of structures).
//
for ( ; Pointers--; ) { ppMemPtr = (uchar **)(pMemory + *((signed short * &)pFormat)++);
pBufPtr = pBufferMark + *((signed short * &)pFormat)++;
NdrpPointerMarshall( pStubMsg, pBufPtr, *ppMemPtr, pFormat );
// Increment to the next pointer's offset_in_memory.
pFormat += 4; } }
pStubMsg->Memory = pMemorySave;
// Return the format string pointer past the pointer descriptions.
return pFormatSave + PointersSave * 8; }
ULONG_PTR NdrpComputeConformance ( PMIDL_STUB_MESSAGE pStubMsg, uchar * pMemory, PFORMAT_STRING pFormat ) /*++
Routine Description :
This routine computes the conformant size for an array or the switch_is value for a union.
Arguments :
pStubMsg - Pointer to the stub message. pMemory - Pointer to the array, string, or union whose size or switch_is is being computed. This is ignored for top level parameters. pFormat - Format string description of the array, string, or union.
Return :
The array or string size or the union switch_is.
Note: Async splitting: The only thing that the compiler should generate with the FC_SPLIT_* operations should be top level conformance. Expr evaluation routines and top level multi-dim sizing is not supported with the async uuid split.
Note64: On 64b platform this routine is can return 8 bytes of a iid_is pointer. However, the MaxCount field does not need to keep it and it stays as a ulong.
--*/ { void * pCount; LONG_PTR Count;
unsigned char FormatCopy[4]; BOOL fAsyncSplit = FALSE;
static const uchar Increments[] = { 4, // Conformant array.
4, // Conformant varying array.
0, 0, // Fixed arrays - unused.
0, 0, // Varying arrays - unused.
4, // Complex array.
2, // Conformant char string.
2, // Conformant byte string.
4, // Conformant stringable struct.
2, // Conformant wide char string.
0, 0, 0, 0, // Non-conformant strings - unused.
0, // Encapsulated union - unused.
2, // Non-encapsulated union.
2, // Byte count pointer.
0, 0, // Xmit/Rep as - unused.
2 // Interface pointer.
};
//
// Advance the format string to the size_is, switch_is, iid_is, or
// byte count description.
//
pFormat += Increments[*pFormat - FC_CARRAY];
pCount = 0;
//
// First check if this is a callback.
//
if ( pFormat[1] == FC_CALLBACK ) { uchar * pOldStackTop; ushort Index;
// Index into expression callback routines table.
Index = *((ushort *)(pFormat + 2));
NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval != 0, "NdrpComputeConformance : no expr eval routines"); NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval[Index] != 0, "NdrpComputeConformance : bad expr eval routine index");
pOldStackTop = pStubMsg->StackTop;
//
// The callback routine uses the StackTop field of the stub message
// to base it's offsets from. So if this is a complex attribute for
// an embedded field of a structure then set StackTop equal to the
// pointer to the structure.
//
if ( (*pFormat & 0xf0) != FC_TOP_LEVEL_CONFORMANCE ) { if ( (*pFormat & 0xf0) == FC_POINTER_CONFORMANCE ) pMemory = pStubMsg->Memory; pStubMsg->StackTop = pMemory; }
//
// This call puts the result in pStubMsg->MaxCount.
//
(*pStubMsg->StubDesc->apfnExprEval[Index])( pStubMsg );
pStubMsg->StackTop = pOldStackTop;
return pStubMsg->MaxCount; }
if ( (*pFormat & 0xf0) == FC_NORMAL_CONFORMANCE ) { // Get the address where the conformance variable is in the struct.
pCount = pMemory + *((signed short *)(pFormat + 2)); goto ComputeConformantGetCount; }
// See if this is an async split
if ( pFormat[1] & 0x20 ) { fAsyncSplit = TRUE; RpcpMemoryCopy( & FormatCopy[0], pFormat, 4 ); pFormat = (PFORMAT_STRING) & FormatCopy[0];
// Remove the async marker
FormatCopy[1] = pFormat[1] & (unsigned char)~0x20; }
//
// Get a pointer to the conformance describing variable.
//
if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_CONFORMANCE ) { //
// Top level conformance. For /Os stubs, the stubs put the max
// count in the stub message. For /Oi stubs, we get the max count
// via an offset from the stack top.
//
if ( pStubMsg->StackTop ) { if ( fAsyncSplit ) { PNDR_DCOM_ASYNC_MESSAGE pAsyncMsg;
pAsyncMsg = (PNDR_DCOM_ASYNC_MESSAGE) pStubMsg->pAsyncMsg;
pCount = pAsyncMsg->BeginStack + *((ushort *)(pFormat + 2)); } else pCount = pStubMsg->StackTop + *((ushort *)(pFormat + 2)); goto ComputeConformantGetCount; } else { //
// If this is top level conformance with /Os then we don't have
// to do anything, the proper conformance count is placed in the
// stub message inline in the stubs.
//
return pStubMsg->MaxCount; } }
//
// If we're computing the size of an embedded sized pointer then we
// use the memory pointer in the stub message, which points to the
// beginning of the embedding structure.
//
if ( (*pFormat & 0xf0) == FC_POINTER_CONFORMANCE ) { pMemory = pStubMsg->Memory; pCount = pMemory + *((signed short *)(pFormat + 2)); goto ComputeConformantGetCount; }
//
// Check for constant size/switch.
//
if ( (*pFormat & 0xf0) == FC_CONSTANT_CONFORMANCE ) { //
// The size/switch is contained in the lower three bytes of the
// long currently pointed to by pFormat.
//
Count = (LONG_PTR) ((ulong)pFormat[1] << 16); Count |= (LONG_PTR) *((ushort *)(pFormat + 2));
goto ComputeConformanceEnd; }
//
// Check for conformance of a multidimensional array element in
// a -Os stub.
//
if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_MULTID_CONFORMANCE ) { long Dimension;
if ( fAsyncSplit ) RpcRaiseException( RPC_X_WRONG_STUB_VERSION );
//
// If pArrayInfo is non-null than we have a multi-D array. If it
// is null then we have multi-leveled sized pointers.
//
if ( pStubMsg->pArrayInfo ) { Dimension = pStubMsg->pArrayInfo->Dimension; pStubMsg->MaxCount = pStubMsg->pArrayInfo->MaxCountArray[Dimension]; } else { Dimension = *((ushort *)(pFormat + 2)); pStubMsg->MaxCount = pStubMsg->SizePtrCountArray[Dimension]; }
return pStubMsg->MaxCount; }
NDR_ASSERT(0, "NdrpComputeConformance:, Invalid Conformance type"); RpcRaiseException( RPC_S_INTERNAL_ERROR ); return 0; //bogus return
ComputeConformantGetCount:
//
// Must check now if there is a dereference op.
//
if ( pFormat[1] == FC_DEREFERENCE ) { pCount = *(void **)pCount; }
//
// Now get the conformance count.
//
switch ( *pFormat & 0x0f ) { case FC_HYPER : // iid_is on 64b platforms only.
Count = *((LONG_PTR *)pCount); break;
case FC_ULONG : Count = (LONG_PTR)*((ulong *)pCount); break;
case FC_LONG : Count = *((long *)pCount); break;
case FC_ENUM16: case FC_USHORT : Count = (long) *((ushort *)pCount); break;
case FC_SHORT : Count = (long) *((short *)pCount); break;
case FC_USMALL : Count = (long) *((uchar *)pCount); break;
case FC_SMALL : Count = (long) *((char *)pCount); break;
default : NDR_ASSERT(0,"NdrpComputeConformance : bad count type"); RpcRaiseException( RPC_S_INTERNAL_ERROR ); return 0; }
//
// Check the operator.
//
switch ( pFormat[1] ) { case FC_DIV_2 : Count /= 2; break; case FC_MULT_2 : Count *= 2; break; case FC_SUB_1 : Count -= 1; break; case FC_ADD_1 : Count += 1; break; default : // OK
break; }
ComputeConformanceEnd:
// Max count is not used for iid_is.
pStubMsg->MaxCount = (ulong) Count;
return (ULONG_PTR) Count; }
void NdrpComputeVariance ( PMIDL_STUB_MESSAGE pStubMsg, uchar * pMemory, PFORMAT_STRING pFormat ) /*++
Routine Description :
Computes the variance (offset and actual count) for an array.
Arguments :
pStubMsg - Pointer to the stub message. pMemory - Pointer to the array whose variance is being computed. This is unused for a top level parameter. pFormat - Format string description of the array.
Return :
None.
--*/ { void * pLength; LONG_PTR Length;
unsigned char FormatCopy[4]; BOOL fAsyncSplit = FALSE;
uchar FcType = *pFormat;
//
// Advance the format string to the variance description.
//
static uchar Increments[] = { 8, // Conformant varying array.
0, 0, // Fixed arrays - unsed.
8, 12, // Varying array.
8 // Complex array.
};
pFormat += Increments[ FcType - FC_CVARRAY];
if (FcType == FC_CVARRAY || FcType == FC_BOGUS_ARRAY) { CORRELATION_DESC_INCREMENT( pFormat ); }
pLength = 0;
//
// First check if this is a callback.
//
if ( pFormat[1] == FC_CALLBACK ) { ULONG_PTR OldMaxCount; uchar * pOldStackTop; ushort Index;
Index = *((ushort *)(pFormat + 2));
NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval != 0, "NdrpComputeConformance : no expr eval routines"); NDR_ASSERT(pStubMsg->StubDesc->apfnExprEval[Index] != 0, "NdrpComputeConformance : bad expr eval routine index");
pOldStackTop = pStubMsg->StackTop;
// This gets trampled by the callback routine.
OldMaxCount = pStubMsg->MaxCount;
//
// The callback routine uses the StackTop field of the stub message
// to base it's offsets from. So if this is a complex attribute for
// an embedded field of a structure then set StackTop equal to the
// pointer to the structure.
//
if ( (*pFormat & 0xf0) != FC_TOP_LEVEL_CONFORMANCE ) { if ( (*pFormat & 0xf0) == FC_POINTER_VARIANCE ) pMemory = pStubMsg->Memory; pStubMsg->StackTop = pMemory; }
//
// This puts the computed offset in pStubMsg->Offset and the length
// in pStubMsg->MaxCount.
//
(*pStubMsg->StubDesc->apfnExprEval[Index])( pStubMsg );
// Put the length in the proper field.
pStubMsg->ActualCount = (ulong)pStubMsg->MaxCount;
pStubMsg->MaxCount = OldMaxCount;
pStubMsg->StackTop = pOldStackTop;
return; }
if ( (*pFormat & 0xf0) == FC_NORMAL_VARIANCE ) { // Get the address where the variance variable is in the struct.
pLength = pMemory + *((signed short *)(pFormat + 2)); goto ComputeVarianceGetCount; }
// See if this is an async split
if ( pFormat[1] & 0x20 ) { fAsyncSplit = TRUE; RpcpMemoryCopy( & FormatCopy[0], pFormat, 4 ); pFormat = (PFORMAT_STRING) & FormatCopy[0];
// Remove the async marker
FormatCopy[1] = pFormat[1] & 0xdf; }
//
// Get a pointer to the variance variable.
//
if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_VARIANCE ) { //
// Top level variance. For /Os stubs, the stubs put the actual
// count and offset in the stub message. For /Oi stubs, we get the
// actual count via an offset from the stack top. The first_is must
// be zero if we get here.
//
if ( pStubMsg->StackTop ) { if ( fAsyncSplit ) { PNDR_DCOM_ASYNC_MESSAGE pAsyncMsg;
pAsyncMsg = (PNDR_DCOM_ASYNC_MESSAGE) pStubMsg->pAsyncMsg;
pLength = pAsyncMsg->BeginStack + *((signed short *)(pFormat + 2)); } else pLength = pStubMsg->StackTop + *((signed short *)(pFormat + 2)); goto ComputeVarianceGetCount; } else { //
// If this is top level variance with /Os then we don't have
// to do anything, the proper variance values are placed in the
// stub message inline in the stubs.
//
return; } }
//
// If we're computing the length of an embedded size/length pointer then we
// use the memory pointer in the stub message, which points to the
// beginning of the embedding structure.
//
if ( (*pFormat & 0xf0) == FC_POINTER_VARIANCE ) { pMemory = pStubMsg->Memory; pLength = pMemory + *((signed short *)(pFormat + 2)); goto ComputeVarianceGetCount; }
//
// Check for constant length.
//
if ( (*pFormat & 0xf0) == FC_CONSTANT_VARIANCE ) { //
// The length is contained in the lower three bytes of the
// long currently pointed to by pFormat.
//
Length = (LONG_PTR) ((ulong)pFormat[1] << 16); Length |= (LONG_PTR) *((ushort *)(pFormat + 2));
goto ComputeVarianceEnd; }
//
// Check for variance of a multidimensional array element in
// a -Os stub.
//
if ( (*pFormat & 0xf0) == FC_TOP_LEVEL_MULTID_CONFORMANCE ) { long Dimension;
if ( fAsyncSplit ) RpcRaiseException( RPC_X_WRONG_STUB_VERSION );
//
// If pArrayInfo is non-null than we have a multi-D array. If it
// is null then we have multi-leveled sized pointers.
//
if ( pStubMsg->pArrayInfo ) { Dimension = pStubMsg->pArrayInfo->Dimension;
pStubMsg->Offset = pStubMsg->pArrayInfo->OffsetArray[Dimension]; pStubMsg->ActualCount = pStubMsg->pArrayInfo->ActualCountArray[Dimension]; } else { Dimension = *((ushort *)(pFormat + 2));
pStubMsg->Offset = pStubMsg->SizePtrOffsetArray[Dimension]; pStubMsg->ActualCount = pStubMsg->SizePtrLengthArray[Dimension]; }
return; }
ComputeVarianceGetCount:
//
// Must check now if there is a dereference op.
//
if ( pFormat[1] == FC_DEREFERENCE ) { pLength = *(void **)pLength; }
//
// Now get the conformance count.
//
switch ( *pFormat & 0x0f ) { case FC_ULONG : Length = (LONG_PTR)*((ulong *)pLength); break;
case FC_LONG : Length = *((long *)pLength); break;
case FC_USHORT : Length = (long) *((ushort *)pLength); break;
case FC_SHORT : Length = (long) *((short *)pLength); break;
case FC_USMALL : Length = (long) *((uchar *)pLength); break;
case FC_SMALL : Length = (long) *((char *)pLength); break;
default : NDR_ASSERT(0,"NdrpComputeVariance : bad format"); RpcRaiseException( RPC_S_INTERNAL_ERROR ); return; }
//
// Check the operator.
//
switch ( pFormat[1] ) { case FC_DIV_2 : Length /= 2; break; case FC_MULT_2 : Length *= 2; break; case FC_SUB_1 : Length -= 1; break; case FC_ADD_1 : Length += 1; break; default : // OK
break; }
ComputeVarianceEnd:
// Get here if the length was computed directly.
pStubMsg->Offset = 0; pStubMsg->ActualCount = (ulong) Length; }
|