Structures (RPC)
There are several categories of structures, progressively more complicated in terms of actions required for marshaling. They begin with a simple structure that can be block-copied as a whole, and continue to a complex structure that has to be serviced field by field.
- Simple Structure
- Simple Structure with Pointers
- Conformant Structure
- Conformant Structure with Pointers
- Conformant Varying Structure (with or without Pointers)
- Hard Structure
- Complex Structure
- Structure Member Layout Description
Note
When compared to array categories, it becomes clear that only structures up to 64k in size can be described (the size is for the flat part of the structure), that is there is no equivalent of SM and LG arrays.
Members Common to Structures
alignment
The necessary alignment of the buffer before unmarshaling the structure. Valid values are 0, 1, 3, and 7 (the actual alignment minus 1).
memory_size
Size of the structure in memory in bytes; for conformant structures this size does not include the array's size.
offset_to_array_description
Offset from the current format string pointer to the description of the conformant array contained in a structure.
member_layout
Description of each element of the structure. The NDR routines only need to examine this portion of a type's format string if endian transformation is needed or if the type is a complex structure.
pointer_layout
See the Pointer Layout section.
Simple Structure
A simple structure contains only base types, fixed arrays, and other simple structures. The main feature of the simple structure is that it can be block-copied as a whole.
FC_STRUCT alignment<1>
memory_size<2>
member_layout<>
FC_END
Simple Structure with Pointers
A simple structure with pointers contains only base types, pointers, fixed arrays, simple structures, and other simple structures with pointers. Because the layout<> will only have to be visited when doing an endianness conversion, it is placed at the end of the description.
FC_PSTRUCT alignment<1>
memory_size<2>
pointer_layout<>
member_layout<>
FC_END
Conformant Structure
A conformant structure contains only base types, fixed arrays, and simple structures, and must contain either a conformant string or a conformant array. This array could actually be contained in another conformant structure or conformant structure with pointers which is embedded in this structure.
FC_CSTRUCT alignment<1>
memory_size<2>
offset_to_array_description<2>
member_layout<>
FC_END
Conformant Structure with Pointers
A conformant structure with pointers contains only base types, pointers, fixed arrays, simple structures, and simple structures with pointers; a conformant structure must contain a conformant array. This array could actually be contained in another conformant structure or conformant structure with pointers that is embedded in this structure.
FC_CPSTRUCT alignment<1>
memory_size<2>
offset_to_array_description<2>
pointer_layout<>
member_layout<> FC_END
Conformant Varying Structure (with or without Pointers)
A conformant varying structure contains only simple types, pointers, fixed arrays, simple structures, and simple structures with pointers; a conformant varying structure must contain either a conformant string or a conformant-varying array. The conformant string or array can actually be contained in another conformant structure or conformant structure with pointers that is embedded in this structure.
FC_CVSTRUCT alignment<1>
memory_size<2>
offset_to_array_description<2>
[pointer_layout<>]
layout<>
FC_END
Hard Structure
The hard structure was a concept aimed at eliminating steep penalties related to processing complex structures. It is derived from an observation that a complex structure typically has only one or two conditions that prevent block-copying, and therefore spoil its performance compared to a simple structure. The culprits are usually unions or enumeration fields.
A hard structure is a structure that has an enum16, end-padding in memory, or a union as the last member. These three elements prevent the structure from falling into one of the previous structure categories, which enjoy small interpretation overhead and maximum optimization potential, but do not force it into the very costly complex structure category.
The enum16 must not cause the memory and wire sizes of the structure to differ. The structure cannot have any conformant array, nor any pointers (unless part of the union); the only other members allowed are base types, fixed arrays, and simple structures.
FC_HARD_STRUCTURE alignment<1>
memory_size<2>
reserved<4>
enum_offset<2>
copy_size<2>
mem_copy_incr<2>
union_description_offset<2>
member_layout<>
FC_END
The enum_offset<2> field provides the offset from the beginning of the structure in memory to an enum16 if it contains one; otherwise the enum_offset<2> field is –1.
The copy_size<2> field provides the total number of bytes in the structure, which may be block-copied into/from the buffer. This total does not include any trailing union nor any end-padding in memory. This value is also the amount the buffer pointer should be incremented following the copy.
The mem_copy_incr<2> field is the number of bytes the memory pointer should be incremented following the block-copy before handling any trailing union. Incrementing by this amount (not by copy_size<2> bytes) yields a proper memory pointer to any trailing union.
Complex Structure
A complex structure is any structure containing one or more fields that either prevent the structure from being block-copied, or for which additional checking must be performed during marshaling or unmarshaling (for example, bound checks on an enumeration). The following NDR types fall in this category:
- simple types: ENUM16, __INT3264 (on 64-bit platforms only), an integral with [range]
- alignment padding at the end of the structure
- interface pointers (they go using an embedded complex)
- ignored pointers (that is related to [ignore] attribute and FC_IGNORE token)
- complex arrays, varying arrays, string arrays
- multidimensional conformant arrays with at least one nonfixed dimension
- unions
- elements defined with [transmit_as], [represent_as], [wire_marshal], [user_marshal]
- embedded complex structures
- padding at the end of the structure
A complex structure has the following format description:
FC_BOGUS_STRUCT alignment<1>
memory_size<2>
offset_to_conformant_array_description<2>
offset_to_pointer_layout<2>
member_layout<>
FC_END
[pointer_layout<>]
The memory_size<2> field is the size of the structure in memory, in bytes.
If the structure contains a conformant array, the offset_to_conformant_array_description<2> field provides the offset to the conformant array's description, otherwise it is zero.
If the structure has pointers, the offset_to_pointer_layout<2> field provides the offset past the structure's layout to the pointer layout, otherwise this field is zero.
The pointer_layout<> field of a complex structure is handled somewhat differently than for other structures. The pointer_layout<> field of a complex structure only contains descriptions of actual pointer fields in the structure itself. Any pointers contained within any embedded arrays, unions, or structures are not described in the complex structure's pointer_layout<> field.
Note
This is in contrast to other structures, which duplicate the description of any pointers contained in embedded arrays or structures in their own pointer _layout<> field as well.
The format of a complex structure's pointer layout is also radically different. Because it only contains descriptions of actual pointer members and because a complex structure is marshaled and unmarshaled one field at a time, the pointer_layout<> field simply contains the pointer description of all pointer members. There is no beginning FC_PP, and none of the usual pointer_layout<> information.
Structure Member Layout Description
A structure's layout description contains one or more of the following format characters:
Any of the base type characters, like FC_CHAR, and so on
Alignment directives. There are three format characters specifying alignment of the memory pointer: FC_ALIGNM2, FC_ALIGNM4, and FC_ALIGNM8.
Note
There are also buffer alignment tokens, FC_ALIGNB2 through FC_ALIGNM8; these are not used.
Memory padding. These occur only at the end of a structure's description and denote the number of bytes of padding in memory before the conformant array in the structure: FC_STRUCTPADn, where n is the number of bytes of padding.
Any embedded nonbase type (note, however, that a conformant array never occurs in the structure layout). This has a 4-byte description:
FC_EMBEDDED_COMPLEX memory_pad<1> offset_to_description<2>,
where the offset is not guaranteed to be 2-byte aligned.
memory_pad<1> is a padding needed in memory before the complex field.
offset_to_description<2> is a relative type offset to the embedded type.
There may also be an FC_PAD before the terminating FC_END if needed to ensure that the format string will be aligned at a 2-byte boundary following the FC_END.