Search code examples
cpaddingmicroprocessorsword-size

struct size, padding and the future


Consider the following program

#include <cstdio>
#include <stdint.h>

struct Foo
    {
    int32_t a;
    int32_t b;
    int32_t c;
    };

struct Bar
    {
    int32_t a;
    int32_t b;
    int16_t c;
    };

int main()
    {
    printf("sizeof(Foo)=%u\nsizeof(Bar)=%u",sizeof(Foo),sizeof(Bar));
    return 0;
    }

When compiling using MinGW I get sizeof(Foo)=12 and sizeof(Bar)=12 on both 32-bit and 64-bit systems (I guess that in a 16-bit environment, sizeof(Bar)=10). Is it likely that in the future, the size of these structures will be 16 to make the fit better in the 64-bit environment?

I'm storing structs in files, and do not want to add work-arounds in the future. The Doom thing structure layout was probably a good choice 1993 but not today.


Solution

  • The size of the Bar structure has little to do with whether you're on a 16-bit environment, it has solely to do with the alignment requirements of the fields within your structure.

    The alignment of the structure is generally identical to the most restrictive alignment of all of the fields within it. In this case, it's the 32-bit alignment requirement of an int32_t type that's affecting it.

    Now you may think that shouldn't be the case since the final field is a 16-bit value and there's nothing after it requiring further alignment, but you'd be wrong. Think of what happens with an array of those things:

    0000  index 0  a      32 bits
    0004           b      32 bits
    0008           c      16 bits
    000a           filler 16 bits
    000c  index 1  a      32 bits
    0010           b      32 bits
    0014           c      16 bits
    0018           filler 16 bits
    

    You can see that filler is required after c due to the requirement for the following a to be correctly aligned.

    In terms of the future, it would be unlikely that any of the alignments of those fields would change, simply because they are exact bit-width types. They will always be 32-bit values and probably always require a 32-bit alignment.

    Having said that, there's no guarantee that you won't find an implementation that requires 32 bit values to be aligned on 128-bit boundaries at some point in the future. Section 6.2.8 Alignment of objects is the controlling section in C11, and it doesn't seem to discount this possibility since it doesn't state the the alignment is related to the object size:

    An object type imposes an alignment requirement on every object of that type: stricter alignment can be requested using the _Align keyword.

    If you want to catch the possibility that the structure size and/or alignment changes, you can use sizeof and alignof in your main function to output a message and exit if they're not correct. This won't affect code in the field but will help enormously if you move to a compiler with stricter alignments.

    And that's the point where you change it (see YAGNI). The problem with planning for all possible futures is that you have no idea what will happen :-)