成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Linux C語言字節對齊的那些事

開發 后端
隨手整理一下C語言中字節對齊的問題與大家一起分享。一起來看一下吧。

[[418052]]

 最近一口君在做一個項目,遇到一個問題,運行于ARM上的threadx在與DSP通信采用消息隊列的方式傳遞消息(最終實現原理是中斷+共享內存的方式),在實際操作過程中發現threadx總是crash,于是經過排查,是因為傳遞消息的結構體沒有考慮字節對齊的問題。

隨手整理一下C語言中字節對齊的問題與大家一起分享。

一、概念 

對齊跟數據在內存中的位置有關。如果一個變量的內存地址正好位于它長度的整數倍,他就被稱做自然對齊。比如在32位cpu下,假設一個整型變量的地址為0x00000004,那它就是自然對齊的。

首先了解什么位、字節、字

名稱 英文名 含義
bit 1個二進制位稱為1個bit
字節 Byte 8個二進制位稱為1個Byte
word 電腦用來一次性處理事務的一個固定長度

字長

一個字的位數,現代電腦的字長通常為16,32, 64位。(一般N位系統的字長是N/8字節。)

不同的CPU一次可以處理的數據位數是不同的,32位CPU可以一次處理32位數據,64位CPU可以一次處理64位數據,這里的位,指的就是字長。

而所謂的字長,我們有時會稱為字(word)。在16位的CPU中,一個字剛好為兩個字節,而32位CPU中,一個字是四個字節。若以字為單位,向上還有雙字(兩個字),四字(四個字)。

二、對齊規則

對于標準數據類型,它的地址只要是它的長度的整數倍就行了,而非標準數據類型按下面的原則對齊:   數組 :按照基本數據類型對齊,第一個對齊了后面的自然也就對齊了。聯合 :按其包含的長度最大的數據類型對齊。結構體:結構體中每個數據類型都要對齊。

三、如何限制定字節對齊位數?

1. 缺省

在缺省情況下,C編譯器為每一個變量或是數據單元按其自然對界條件分配空間。一般地,可以通過下面的方法來改變缺省的對界條件:

2. #pragma pack(n)

·  使用偽指令#pragma pack (n),C編譯器將按照n個字節對齊。· 使用偽指令#pragma pack (),取消自定義字節對齊方式。

#pragma pack(n) 用來設定變量以n字節對齊方式。n字節對齊就是說變量存放的起始地址的偏移量有兩種情況:

  1.  如果n大于等于該變量所占用的字節數,那么偏移量必須滿足默認的對齊方式
  2.  如果n小于該變量的類型所占用的字節數,那么偏移量為n的倍數,不用滿足默認的對齊方式。

結構的總大小也有一個約束條件,如果n大于等于所有成員變量類型所占用的字節數,那么結構的總大小必須為占用空間最大的變量占用的空間數的倍數;否則必須是n的倍數。

3. __attribute

另外,還有如下的一種方式:· __attribute((aligned (n))),讓所作用的結構成員對齊在n字節自然邊界上。如果結構中有成員的長度大于n,則按照最大成員的長度來對齊。· attribute ((packed)),取消結構在編譯過程中的優化對齊,按照實際占用字節數進行對齊。

4. 匯編.align

匯編代碼通常用.align來制定字節對齊的位數。

.align:用來指定數據的對齊方式,格式如下:

  1. .align [absexpr1, absexpr2] 

以某種對齊方式,在未使用的存儲區域填充值. 第一個值表示對齊方式,4, 8,16或 32. 第二個表達式值表示填充的值。

四、為什么要對齊?

操作系統并非一個字節一個字節訪問內存,而是按2,4,8這樣的字長來訪問。因此,當CPU從存儲器讀數據到寄存器,IO的數據長度通常是字長。如32位系統訪問粒度是4字節(bytes), 64位系統的是8字節。當被訪問的數據長度為n字節且該數據地址為n字節對齊時,那么操作系統就可以高效地一次定位到數據, 無需多次讀取,處理對齊運算等額外操作。數據結構應該盡可能地在自然邊界上對齊。如果訪問未對齊的內存,CPU需要做兩次內存訪問。

字節對齊可能帶來的隱患:

代碼中關于對齊的隱患,很多是隱式的。比如在強制類型轉換的時候。例如: 

  1. unsigned int i = 0x12345678 
  2. unsigned char *p=NULL 
  3. unsigned short *p1=NULL 
  4. p=&i;  
  5. *p=0x00 
  6. p1=(unsigned short *)(p+1);  
  7. *p1=0x0000

最后兩句代碼,從奇數邊界去訪問unsignedshort型變量,顯然不符合對齊的規定。在x86上,類似的操作只會影響效率,但是在MIPS或者sparc上,可能就是一個error,因為它們要求必須字節對齊.

五、舉例

例1:os基本數據類型占用的字節數

首先查看操作系統的位數

在64位操作系統下查看基本數據類型占用的字節數: 

  1. #include <stdio.h>  
  2. int main()  
  3.  
  4.     printf("sizeof(char) = %ld\n", sizeof(char));  
  5.     printf("sizeof(int) = %ld\n", sizeof(int));  
  6.     printf("sizeof(float) = %ld\n", sizeof(float));  
  7.     printf("sizeof(long) = %ld\n", sizeof(long));                                    
  8.     printf("sizeof(long long) = %ld\n", sizeof(long long));  
  9.     printf("sizeof(double) = %ld\n", sizeof(double));  
  10.     return 0;  

例2:結構體占用的內存大小--默認規則

考慮下面的結構體占用的位數 

  1. struct yikou_s  
  2.  
  3.     double d;  
  4.     char c;  
  5.     int i;  
  6. } yikou_t; 

執行結果

  1. sizeof(yikou_t) = 16 

在內容中各變量位置關系如下

其中成員C的位置還受字節序的影響,有的可能在位置8

編譯器給我們進行了內存對齊,各成員變量存放的起始地址相對于結構的起始地址的偏移量必須為該變量類型所占用的字節數的倍數, 且結構的大小為該結構中占用最大空間的類型所占用的字節數的倍數。

對于偏移量:變量type n起始地址相對于結構體起始地址的偏移量必須為sizeof(type(n))的倍數結構體大小:必須為成員最大類型字節的倍數 

  1. char: 偏移量必須為sizeof(char) 即1的倍數  
  2. int: 偏移量必須為sizeof(int) 即4的倍數 
  3. float: 偏移量必須為sizeof(float) 即4的倍數  
  4. double: 偏移量必須為sizeof(double) 即8的倍數 

例3:調整結構體大小

我們將結構體中變量的位置做以下調整: 

  1. struct yikou_s  
  2.  
  3.     char c;  
  4.     double d;  
  5.     int i;  
  6. } yikou_t; 

執行結果 

  1. sizeof(yikou_t) = 24 

各變量在內存中布局如下:

當結構體中有嵌套符合成員時,復合成員相對于結構體首地址偏移量是復合成員最寬基本類型大小的整數倍。

例4:#pragma pack(4) 

  1. #pragma pack(4)  
  2. struct yikou_s  
  3.  
  4.     char c; 
  5.     double d;  
  6.     int i;  
  7. } yikou_t;  
  8. sizeof(yikou_t) = 16 

例5:#pragma pack(8) 

  1. #pragma pack(8)  
  2. struct yikou_s  
  3.  
  4.     char c;  
  5.     double d;  
  6.     int i;  
  7. } yikou_t;  
  1. sizeof(yikou_t) = 24 

例6:匯編代碼

舉例:以下是截取的uboot代碼中異常向量irq、fiq的入口位置代碼:

六、匯總實力

有手懶的同學,直接貼一個完整的例子給你們: 

  1. #include <stdio.h>  
  2. main()  
  3.  
  4. struct A {  
  5.     int a;  
  6.     char b;  
  7.     short c;  
  8. };  
  9. struct B {  
  10.     char b;  
  11.     int a;  
  12.     short c;  
  13. };  
  14. struct AA {  
  15.    // int a;  
  16.     char b;  
  17.     short c; 
  18.  };  
  19. struct BB {  
  20.     char b; 
  21.     // int a;  
  22.     short c;  
  23. };   
  24. #pragma pack (2) /*指定按2字節對齊*/  
  25. struct C {  
  26.     char b;  
  27.     int a;  
  28.     short c;  
  29. };  
  30. #pragma pack () /*取消指定對齊,恢復缺省對齊*/  
  31. #pragma pack (1) /*指定按1字節對齊*/  
  32. struct D {  
  33.     char b;  
  34.     int a;  
  35.     short c;  
  36. };  
  37. #pragma pack ()/*取消指定對齊,恢復缺省對齊*/  
  38. int s1=sizeof(struct A);  
  39. int s2=sizeof(struct AA);  
  40. int s3=sizeof(struct B);  
  41. int s4=sizeof(struct BB);  
  42. int s5=sizeof(struct C);  
  43. int s6=sizeof(struct D);  
  44. printf("%d\n",s1);  
  45. printf("%d\n",s2);  
  46. printf("%d\n",s3);  
  47. printf("%d\n",s4);  
  48. printf("%d\n",s5);  
  49. printf("%d\n",s6);  
  50.  

 

責任編輯:龐桂玉 來源: C語言與C++編程
相關推薦

2021-08-30 12:05:46

Linux字節對齊代碼

2021-08-06 11:50:49

Linux 字節對齊Linux 系統

2017-05-15 21:50:54

Linux引號

2018-04-11 08:54:16

Linux ARM存儲分布

2014-06-06 16:08:17

初志科技

2011-09-19 15:40:35

2020-07-29 08:14:59

云計算云遷移IT

2017-11-20 09:00:34

Linux服務器時間同步

2011-05-19 16:47:50

軟件測試

2012-05-01 08:06:49

手機

2024-02-04 17:03:30

2012-05-31 09:53:38

IT風云15年

2013-12-12 17:50:36

開源系統Linux

2015-08-20 09:17:36

Java線程池

2015-09-14 09:28:47

2009-02-19 10:21:00

路由多WAN口

2017-03-08 08:53:44

Git命令 GitHub

2021-08-11 21:46:47

MySQL索引join

2021-03-18 16:05:20

SSD存儲故障

2012-10-08 11:55:05

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产高清在线精品 | 久久精品久久精品久久精品 | 日本午夜网站 | 国产精品久久久久久久久久三级 | 欧美一级欧美一级在线播放 | 超碰3| 日韩亚洲一区二区 | 国产欧美日韩在线一区 | 日韩视频在线播放 | 精品国产黄a∨片高清在线 www.一级片 国产欧美日韩综合精品一区二区 | 国产精品99精品久久免费 | 天天狠狠| 国产一区精品 | 一级免费毛片 | 欧美精品被| 欧美一区二区在线免费观看 | 91精品午夜窝窝看片 | a黄视频| 日韩av成人在线观看 | 自拍偷拍小视频 | 五月婷婷导航 | 日本欧美视频 | 亚洲免费人成在线视频观看 | 一级片在线观看视频 | 亚洲高清视频一区二区 | 国产精品日韩欧美一区二区三区 | 成人性生交a做片 | 国产精品成人国产乱一区 | 一区二区三区四区电影视频在线观看 | 亚洲精品1区 | 国产96色在线 | 黄网站在线播放 | 91精品国产91久久久久青草 | 亚洲精品视频久久 | 国产成人精品a视频一区www | 精品国产一二三区 | 欧美一二三 | 欧美精品一区二区三区四区 在线 | 中文久久| 亚洲国产成人av好男人在线观看 | 免费成人av网站 |