windows-nt/Source/XPSP1/NT/com/rpc/ndr20/mrshlp.cxx

867 lines
24 KiB
C++
Raw Normal View History

2020-09-26 03:20:57 -05:00
/*+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
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 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;
}