I\'ve been poring over the draft standard and can\'t seem to find what I\'m looking for.
If I have a standard-layout type
struct T {
unsigned hand
You essentially are asking, given:
struct T {
U handle;
};
whether it's guaranteed that sizeof(T) == sizeof(U)
. No, it is not.
Section 9.2/17 of the ISO C++03 standard says:
A pointer to a POD-struct object, suitably converted using a
reinterpret_cast
, points to its initial member (or if that member is a bit-field, then to the unit in which it resides) and vice versa.
Suppose you have an array of struct T
. The vice versa part means that the address of any of the T::handle
members must also be a valid address of a struct T
. Now, suppose that these members are of type char
and that your claim is true. This would mean that struct T
would be allowed to have an unaligned address, which seems rather unlikely. The standard usually tries to not tie the hands of implementations in such a way. For your claim to be true, the standard would have to require that struct T
be allowed to have unaligned addresses. And it would have to be allowed for all structures, because struct T
could be a forward-declared, opaque type.
Furthermore, section 9.2/17 goes on to state:
[Note: There might therefore be unnamed padding within a POD-struct object, but not at its beginning, as necessary to achieve appropriate alignment.]
Which, taken a different way, means that there is no guarantee that there will never be padding.
RETRACTION: The argument is erroneous. The proof of Lemma 2 relies on a hidden premise that the alignment of an aggregate type is determined strictly by the alignments of its member types. As Dyp points out in the commentary, that premise is not supported by the standard. It is therefore admissible for struct { Foo f }
to have a more strict alignment requirement that Foo
.
sizeof(T) == sizeof(U)
when T
is a standard layout class (9/7) with default alignment having a single default-aligned non-static data member U
, e.g,
struct T { // or class, or union
U u;
};
It's well-established that:
sizeof(T) >= sizeof(U)
offsetof(T, u) == 0
(9.2/19)U
must be a standard layout type for T
to be a standard layout classu
has a representation consisting of exactly sizeof(U)
contiguous bytes of memory (1.8/5)Together these facts imply that the first sizeof(U)
bytes of the representation of T
are occupied by the representation of u
. If sizeof(T) > sizeof(U)
, the excess bytes must then be tail padding: unused padding bytes inserted after the representation of u
in T
.
The argument is, in short:
Under what circumstances does the standard allow an implementation to add such padding to the representation of a standard layout class? When it's necessary for alignment: per 3.11/1, "An alignment is an implementation-defined integer value representing the number of bytes between successive addresses at which a given object can be allocated." There are two mentions of adding padding, both for alignment reasons:
5.3.3 Sizeof [expr.sizeof]/2 states "When applied to a reference or a reference type, the result is the size of the referenced type. When applied
to a class, the result is the number of bytes in an object of that class including any padding required for placing objects of that type in an array. The size of a most derived class shall be greater than zero (1.8). The result of applying sizeof
to a base class subobject is the size of the base class type.77 When applied to an array, the result is the total number of bytes in the array. This implies that the size of an array of n elements is n times the size of an element."
9.2 Class members [class.mem]/13 states "Implementation alignment requirements might cause two adjacent members not to be allocated immediately after each other; so might requirements for space for managing virtual functions (10.3) and virtual base classes (10.1)."
(Notably the C++ standard does not contain a blanket statement allowing implementations to insert padding in structures as in the C standards, e.g., N1570 (C11-ish) §6.7.2.1/15 "There may be unnamed padding within a structure object, but not at its beginning." and /17 "There may be unnamed padding at the end of a structure or union.")
Clearly the text of 9.2 doesn't apply to our problem, since (a) T
has only one member and thus no "adjacent members", and (b) T
is standard layout and hence has no virtual functions or virtual base classes (per 9/7). Demonstrating that 5.3.3/2 doesn't allow padding in our problem is more challenging.
Lemma 1: For any type W
with default alignment, alignof(W)
divides sizeof(W)
: By 5.3.3/2, the size of an array of n elements of type W
is exactly n times sizeof(W)
(i.e., there is no "external" padding between array elements). The addresses of consecutive array elements are then sizeof(W)
bytes apart. By the definition of alignment, it must then be that alignof(W)
divides sizeof(W)
.
Lemma 2: The alignment alignof(W)
of a default-aligned standard layout class W
with only default-aligned data members is the least common multiple LCM(W)
of the alignments of the data members (or 1 if there are none): Given an address at which an object of W
can be allocated, the address LCM(W)
bytes away must also be appropriately aligned: the difference between the addresses of member subobjects would also be LCM(W)
bytes, and the alignment of each such member subobject divides LCM(W)
. Per the definition of alignment in 3.11/1, we have that alignof(W)
divides LCM(W)
. Any whole number of bytes n < LCM(W)
must not be divisible by the alignment of some member v
of W
, so an address that is only n
bytes away from an address at which an object of W
can be allocated is consequently not appropriately aligned for an object of W
, i.e., alignof(W) >= LCM(W)
. Given that alignof(W)
divides LCM(W)
and alignof(W) >= LCM(W)
, we have alignof(W) == LCM(W)
.
Application of this lemma to the original problem has the immediate consequence that alignof(T) == alignof(U)
. So how much padding might be "required for placing objects of that type in an array"? None. Since alignof(T) == alignof(U)
by the second lemma, and alignof(U)
divides sizeof(U)
by the first, it must be that alignof(T)
divides sizeof(U)
so zero bytes of padding are required to place objects of type T
in an array.
Since all possible sources of padding bytes have been eliminated, an implementation may not add padding to T
and we have sizeof(T) == sizeof(U)
as required.
I am used to Borland environments and for them:
T is a struct in your case so sizeof(T) is size of struct
for the same reason if you have 4Byte struct (uint32) and 16Byte allign