C struct size alignment
Asked Answered
A

5

14

I want the size of a C struct to be multiple of 16 bytes (16B/32B/48B/..). It does not matter which size it gets to; it only needs to be multiple of 16 bytes.

How could I enforce the compiler to do that?

Attainable answered 21/6, 2012 at 0:40 Comment(4)
Your title uses the word alignment however the body of your question talks about size of the struct. What is it that you actually want to accomplish? And I am also curious as to why.Ehrman
And here is an existing article that provides a solution using a union to pad to a particular size. However that solution requires knowing the size of the struct Size of Struct however what you would need to do is to use a size calculation for the char array such as (sizeof(struct it)/16+1)*16 rather than a number constant.Ehrman
@RichardChambers The reason is because I have a 16B aligned chunk of data and this struct is going to be at the beginning of the chunk, and I still want the beginning of the actual data portion of the chunk (after the struct) to be aligned to 16B for good SSE performance.Attainable
@RichardChambers Yeah. The title seems to be somewhat misleading. My concern is the size of the struct, and not the alignment.Attainable
E
7

The size of a C struct will depend on the members of the struct, their types and how many of them there are. There is really no standard way to force the compiler to make structs to be a multiple of some size. Some compilers provide a pragma that will allow you to set the alignment boundary however that is really a different thing. And there may be some that would have such a setting or provide such a pragma.

However if you insist on this one method would be to do memory allocation of the struct and to force the memory allocation to round up to the next 16 byte size.

So if you had a struct like this.

struct _simpleStruct {
   int iValueA;
   int iValueB;
};

Then you could do something like the following.

{
    struct _simpleStruct *pStruct = 0;
    pStruct = malloc ((sizeof(*pStruct)/16 + 1)*16);
    // use the pStruct for whatever
    free(pStruct);
}

What this would do is to push the size up to the next 16 byte size so far as you were concerned. However what the memory allocator does may or may not be to give you a block that is actually that size. The block of memory may actually be larger than your request.

If you are going to do something special with this, for instance lets say that you are going to write this struct to a file and you want to know the block size then you would have to do the same calculation used in the malloc() rather than using the sizeof() operator to calculate the size of the struct.

So the next thing would be to write your own sizeof() operator using a macro such as.

#define SIZEOF16(x) ((sizeof(x)/16 + 1) * 16)

As far as I know there is no dependable method for pulling the size of an allocated block from a pointer. Normally a pointer will have a memory allocation block that is used by the memory heap management functions that will contain various memory management information such as the allocated block size which may actually be larger than the requested amount of memory. However the format for this block and where it is located relative to the actual memory address provided will depend on the C compiler's run time.

Ehrman answered 21/6, 2012 at 1:5 Comment(3)
Good answer, but looks like if the type is already a multiple of 16 bytes then this macro will add an extra 16 bytes to its size, potentially wasting memory. Here is an alternative using the ternary operator to eliminate this case: #define SIZEOF16(x) ((sizeof(x) & 0x0F) ? (sizeof(x) + (0x10 - (sizeof(x) & 0x0F))) : sizeof(x))Vivid
@Vivid a good point. I suppose my concern is with the multiple use of the macro argument. Most people know not to use an expression that modifies such as x++ in a macro and that most function looking things that are all uppercase are typically a macro though so I suppose it would probably be fairly innocuous. I suppose an inline function would do it as well.Ehrman
Multiple uses in a macro is a valid concern, though sizeof is figured out at compile-time. I can't imagine it being used like this: SIZEOF16 = (x++) ever, but if so I think there may be some way to tell GCC to only evaluate a macro argument once (can't recall it off the top of my head).Vivid
F
23

For Microsoft Visual C++:

#pragma pack(push, 16)

struct _some_struct
{
     ...
}

#pragma pack(pop)

For GCC:

struct _some_struct { ... } __attribute__ ((aligned (16)));

Example:

#include <stdio.h>

struct test_t {
    int x;
    int y;
} __attribute__((aligned(16)));

int main()
{
    printf("%lu\n", sizeof(struct test_t));
    return 0;
}

compiled with gcc -o main main.c will output 16. The same goes for other compilers.

Filamentary answered 21/6, 2012 at 0:53 Comment(4)
For gcc attribute aligned, is it actually forcing the struct size to be multiple of 16B? Or does it just guarantee that the start address of the struct is aligned to 16B?Attainable
Both the start address and the size of the structure will be aligned to 16B. This is done in order to assure correct alignment of elements of the array of structures.Filamentary
This is not true. The __attribute(aligned(x))) macro allocates the struct on a x-byte boundary and has nothing to do with the size of the struct gcc.gnu.org/onlinedocs/gcc-3.2.3/gcc/Variable-Attributes.htmlRepeat
It is true (it is implied). Check it yourself. I have updated my answer with an example.Filamentary
E
7

This depends entirely on the compiler and other tools since alignment is not specified that deeply in the ISO C standard (it specifies that alignment may happen at the compilers behest but does not go into detail as to how to enforce it).

You'll need to look into the implementation-specific stuff for your compiler toolchain. It may provide a #pragma pack (or align or some other thing) that you can add to your structure defininition.

It may also provide this as a language extension. For example, gcc allows you to add attributes to a definition, one of which controls alignment:

struct mystruct { int val[7]; } __attribute__ ((aligned (16)));
Eclosion answered 21/6, 2012 at 0:51 Comment(0)
E
7

The size of a C struct will depend on the members of the struct, their types and how many of them there are. There is really no standard way to force the compiler to make structs to be a multiple of some size. Some compilers provide a pragma that will allow you to set the alignment boundary however that is really a different thing. And there may be some that would have such a setting or provide such a pragma.

However if you insist on this one method would be to do memory allocation of the struct and to force the memory allocation to round up to the next 16 byte size.

So if you had a struct like this.

struct _simpleStruct {
   int iValueA;
   int iValueB;
};

Then you could do something like the following.

{
    struct _simpleStruct *pStruct = 0;
    pStruct = malloc ((sizeof(*pStruct)/16 + 1)*16);
    // use the pStruct for whatever
    free(pStruct);
}

What this would do is to push the size up to the next 16 byte size so far as you were concerned. However what the memory allocator does may or may not be to give you a block that is actually that size. The block of memory may actually be larger than your request.

If you are going to do something special with this, for instance lets say that you are going to write this struct to a file and you want to know the block size then you would have to do the same calculation used in the malloc() rather than using the sizeof() operator to calculate the size of the struct.

So the next thing would be to write your own sizeof() operator using a macro such as.

#define SIZEOF16(x) ((sizeof(x)/16 + 1) * 16)

As far as I know there is no dependable method for pulling the size of an allocated block from a pointer. Normally a pointer will have a memory allocation block that is used by the memory heap management functions that will contain various memory management information such as the allocated block size which may actually be larger than the requested amount of memory. However the format for this block and where it is located relative to the actual memory address provided will depend on the C compiler's run time.

Ehrman answered 21/6, 2012 at 1:5 Comment(3)
Good answer, but looks like if the type is already a multiple of 16 bytes then this macro will add an extra 16 bytes to its size, potentially wasting memory. Here is an alternative using the ternary operator to eliminate this case: #define SIZEOF16(x) ((sizeof(x) & 0x0F) ? (sizeof(x) + (0x10 - (sizeof(x) & 0x0F))) : sizeof(x))Vivid
@Vivid a good point. I suppose my concern is with the multiple use of the macro argument. Most people know not to use an expression that modifies such as x++ in a macro and that most function looking things that are all uppercase are typically a macro though so I suppose it would probably be fairly innocuous. I suppose an inline function would do it as well.Ehrman
Multiple uses in a macro is a valid concern, though sizeof is figured out at compile-time. I can't imagine it being used like this: SIZEOF16 = (x++) ever, but if so I think there may be some way to tell GCC to only evaluate a macro argument once (can't recall it off the top of my head).Vivid
T
5

You could perhaps do a double struct, wrapping your actual struct in a second one that can add padding:

struct payload {
  int a;    /*Your actual fields. */
  float b;
  char c;
  double d;
};

struct payload_padded {
  struct payload p;
  char   padding[16 * ((sizeof (struct payload) + 15) / 16)];
};

Then you can work with the padded struct:

struct payload_padded a;

a.p.d = 43.3;

Of course, you can make use of the fact that the first member of a structure starts 0 bytes from where the structure starts, and treat a pointer to struct payload_padded as if it's a pointer to a struct payload (because it is):

float d_plus_2(const struct payload *p)
{
  return p->d + 2;
}

/* ... */

struct payload_padded b;
const double dp2 = d_plus_2((struct payload *) &b);
Testis answered 21/6, 2012 at 10:0 Comment(0)
A
0

In C++11 you can use alignas to specify packing.

struct alignas(16) _simpleStruct {
   int iValueA;
   int iValueB;
};

It is the equivalent of using __attribute__((aligned(16)));

https://en.cppreference.com/w/cpp/language/alignas

Alsoran answered 19/7, 2024 at 16:43 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.