Partilhar via


align (C++)

Específico da Microsoft

Use __declspec(align(#)) para controlar precisamente o alinhamento dos dados definidos pelo usuário (por exemplo, alocações estáticas ou dados automáticos em uma função).

__declspec( align( # ) ) declarator

Comentários

Criar aplicativos que usam as instruções mais recentes do processador acarreta algumas restrições e problemas novos. Em particular, muitas instruções novas exigem que os dados sejam alinhados a limites de 16 bytes. Além disso, ao alinhar os dados usados com frequência ao tamanho da linha de cache de um processador específico, você melhora o desempenho do cache. Por exemplo, se você definir uma estrutura cujo tamanho seja menor de 32 bytes, alinhe-a para 32 bytes a fim de garantir que os objetos desse tipo de estrutura sejam armazenados em cache com eficiência.

# é o valor do alinhamento. As entradas válidas são potências inteiras de dois de 1 a 8192 (bytes), como 2, 4, 8, 16, 32 ou 64. declarator são os dados que você está declarando como alinhados.

Para obter informações sobre como retornar um valor do tipo size_t que é o requisito de alinhamento do tipo, consulte __alignof. Para obter informações sobre como declarar ponteiros não alinhados ao redirecionar para processadores de 64 bits, consulte __unaligned.

Você pode usar __declspec(align(#)) ao definir uma struct, union ou class ou ao declarar uma variável.

O compilador não garante nem tenta preservar o atributo de alinhamento de dados durante uma cópia ou operação de transformação de dados. Por exemplo, memcpy pode copiar uma estrutura declarada com __declspec(align(#)) em qualquer local. Alocadores normais como, por exemplo, malloc, C++ novo operador, e os alocadores Win32 retornam uma memória que provavelmente não será suficientemente alinhada para estruturas __declspec(align(#)) ou matrizes de estruturas. Para garantir que o destino de uma cópia ou operação de transformação de dados seja alinhado corretamente, use _aligned_malloc, ou grave seu próprio alocador.

Não é possível especificar o alinhamento dos parâmetros da função. Quando os dados que têm um atributo de alinhamento são passados pelo valor na pilha, seu alinhamento é controlado pela convenção de chamada. Se o alinhamento de dados for importante na função chamada, copie o parâmetro na memória alinhada corretamente antes de usar.

Sem __declspec(align(#)), o Visual C++ geralmente alinha os dados em limites naturais com base em processador de destino e o tamanho dos dados, até o limite de 4 bytes em processadores de 32 bits e limite de 8 bytes em processadores de 64 bits. Os dados nas classes ou estruturas são alinhados dentro da classe ou da estrutura no mínimo de seu alinhamento natural e da configuração atual da compactação (de #pragma pack ou da opção de compilador /Zp).

Este exemplo mostra o uso de __declspec(align(#)):

__declspec(align(32)) struct Str1{
   int a, b, c, d, e;
};

Esse tipo agora tem um atributo de alinhamento de 32 bytes. Isso significa que todas as instâncias estáticas e automáticas iniciam em um limite de 32 bytes. Os tipos adicionais de estrutura declarados com esse tipo como um membro preservam o atributo de alinhamento desse tipo, isto é, qualquer estrutura com Str1 como um elemento terá um atributo de alinhamento de pelo menos 32.

Observe que sizeof(struct Str1) é igual a 32. Isso significa que se uma matriz de objetos Str1 for criada e a base da matriz for 32 bytes alinhados, cada membro da matriz também será de 32 bytes alinhados. Para criar uma matriz cuja base esteja alinhada corretamente, use _aligned_malloc ou crie seu próprio alocador.

O valor de sizeof para qualquer estrutura é o deslocamento do membro final, mais o tamanho desse membro, arredondado até o múltiplo mais próximo do maior valor de alinhamento de membro ou do valor inteiro de alinhamento de estrutura, o que for maior.

O compilador usa essas regras para alinhamento de estrutura:

  • A menos que seja substituído por __declspec(align(#)), o alinhamento de um membro de estrutura escalar é o mínimo do seu tamanho e da compactação atual.

  • A menos que seja substituído por __declspec(align(#)), o alinhamento de uma estrutura é o máximo de alinhamentos individuais dos seus membros.

  • Um membro de estrutura é colocado em um deslocamento do início de sua estrutura pai, que é o menor múltiplo de seu alinhamento maior ou igual ao deslocamento do final do membro anterior.

  • O tamanho de uma estrutura é o menor múltiplo de seu maior alinhamento maior que ou igual ao deslocamento do final de seu último membro.

__declspec(align(#)) só pode aumentar as restrições de alinhamento.

Para obter mais informações, consulte:

  • Exemplos de align

  • Definindo novos tipos com __declspec(align(#))

  • Alinhando dados no armazenamento local de threads

  • Como alinhar trabalhos com a compactação de dados

  • Exemplos de alinhamento estrutura (específico para x64)

Exemplos de align

Os exemplos a seguir mostram como __declspec(align(#)) afeta o tamanho e o alinhamento de estruturas de dados. Os exemplos assumem as seguintes definições:

#define CACHE_LINE  32
#define CACHE_ALIGN __declspec(align(CACHE_LINE))

Neste exemplo, a estrutura S1 é definida usando __declspec(align(32)). Todos os usos de S1 para uma definição de variável ou em outras declarações de tipo são alinhados como 32 bytes. sizeof(struct S1) retorna 32 e S1 tem 16 bytes de preenchimento depois dos 16 bytes necessários para armazenar os quatro inteiros. Cada int membro requer alinhamento de 4 bytes, mas o alinhamento da estrutura em si está declarado como 32. Portanto, o alinhamento geral é 32.

struct CACHE_ALIGN S1 { // cache align all instances of S1
   int a, b, c, d;
};
struct S1 s1;   // s1 is 32-byte cache aligned

Neste exemplo, sizeof(struct S2) retorna 16, que é exatamente a soma dos tamanhos do membro, pois esse é um múltiplo do maior requisito de alinhamento (um múltiplo de 8).

__declspec(align(8)) struct S2 {
   int a, b, c, d;
};

No exemplo a seguir, sizeof(struct S3) retorna 64.

struct S3 {
   struct S1 s1;   // S3 inherits cache alignment requirement
                  // from S1 declaration
   int a;         // a is now cache aligned because of s1
                  // 28 bytes of trailing padding
};

Neste exemplo, observe que a tem o alinhamento de seu natural tipo, nesse caso, 4 bytes. No entanto, S1 deve ser alinhado para 32 bytes. Vinte e oito bytes de preenchimento seguem a, de modo que s1 comece no deslocamento de 32. S4, então, herda o requisito de alinhamento de S1, pois é o maior requisito de alinhamento da estrutura. sizeof(struct S4) retorna 64.

struct S4 {
   int a;
   // 28 bytes padding
    struct S1 s1;      // S4 inherits cache alignment requirement of S1
};

As três declarações de variável a seguir também usam __declspec(align(#)). Em cada caso, a variável deve ser alinhada para 32 bytes. No caso da matriz, o endereço básico da matriz, não cada membro dela, é alinhado para 32 bytes. O valor de sizeof para cada membro da matriz não é afetado quando você usa __declspec(align(#)).

CACHE_ALIGN int i;
CACHE_ALIGN int array[128];
CACHE_ALIGN struct s2 s;

Para alinhar cada membro de uma matriz, códigos como estes deverão ser usados:

typedef CACHE_ALIGN struct { int a; } S5;
S5 array[10];

Neste exemplo, observe que alinhar a estrutura em si e alinhar o primeiro elemento tem o mesmo efeito:

CACHE_ALIGN struct S6 {
   int a;
   int b;
};

struct S7 {
   CACHE_ALIGN int a;
               int b;
};

S6 e S7 têm alinhamento, alocação e características de tamanho idênticos.

Neste exemplo, o alinhamento dos endereços iniciais de a, b, c e d são 4, 1, 4 e 1, respectivamente.

void fn() { 
   int a;
   char b;
   long c;
   char d[10]
} 

O alinhamento, se a memória tiver sido alocada no heap, depende de qual função de alocação é chamada. Por exemplo, se você usar malloc, o resultado dependerá do tamanho do operando. Se arg >= 8, a memória retornada estará alinhada em 8 bytes. Se arg < 8, o alinhamento da memória retornada será a primeira potência de 2 menor que arg. Por exemplo, se você usar malloc (7), o alinhamento será de 4 bytes.

Definindo novos tipos com __declspec(align(#))

Você pode definir um tipo com uma característica de alinhamento.

Por exemplo, você pode definir uma struct com um valor de alinhamento desta maneira:

struct aType {int a; int b;};
typedef __declspec(align(32)) struct aType bType;

Agora, aType e bType são do mesmo tamanho (8 bytes), mas as variáveis do tipo bType serão alinhadas para 32 bytes.

Alinhando dados no armazenamento local de threads

O armazenamento local de thread estático (TLS ) criado com o atributo __declspec(thread) e colocado na seção TLS na imagem funciona para o alinhamento exatamente como dados estáticos normais. Para criar dados TLS, o sistema operacional aloca memória do tamanho da seção do TLS e respeita o atributo de alinhamento da seção do TLS.

Este exemplo mostra várias maneiras de colocar dados alinhados no armazenamento local de threads.

// put an aligned integer in TLS
__declspec(thread) __declspec(align(32)) int a;   

// define an aligned structure and put a variable of the struct type
// into TLS
__declspec(thread) __declspec(align(32)) struct F1 { int a; int b; } a;

// create an aligned structure 
struct CACHE_ALIGN S9 {
   int a;
   int b;
};
// put a variable of the structure type into TLS
__declspec(thread) struct S9 a;

Como alinhar trabalhos com a compactação de dados

A opção de compilador /Zp e o pragma pack têm o efeito de compactar dados para membros da estrutura e da união. Este exemplo mostra como o /Zp e o __declspec(align(#)) funcionam juntos:

struct S {
   char a;
   short b;
   double c;
   CACHE_ALIGN double d;
   char e;
   double f;
};

A tabela a seguir lista o deslocamento de cada membro em diversos valores /Zp (ou #pragma pack), mostrando como os dois interagem.

Variável

/Zp1

/Zp2

/Zp4

/Zp8

a

0

0

0

0

b

1

2

2

2

c

3

4

4

8

d

32

32

32

32

e

40

40

40

40

f

41

42

44

48

sizeof(S)

64

64

64

64

Para obter mais informações, consulte /Zp (alinhamento de membro do Struct).

O deslocamento de um objeto baseia-se no deslocamento do objeto anterior e na configuração atual de compactação, a menos que o objeto tenha um atributo __declspec(align(#)). Nesse caso, o alinhamento será baseado no deslocamento do objeto anterior e no valor de __declspec(align(#)) para o objeto.

Consulte também

Referência

__declspec

Conceitos

Visão geral das convenções ARM ABI

Visão geral do x 64 convenções de chamada