關鍵是big endian和little endian的概念。注意16bit機器和32bit機器(以32位爲訪問單位)下排列不同。
內存地址從低地址向高地址增長,big endian是高位數據優先,即高位放在低地址。而little endian是低位數據優先,低位放在低地址。網絡字節序採用大端。
判斷字節序:
main()
{
int x = 0x1;
if ((*(char *)&x) == 0x1)
printf("little endian!\n");
else
printf("big endian!\n");
}
字節序轉換:
ntohs(n) = __swab16(n),
ntohl = __swab32(n)。
__swab16與__swab32函數定義如下所示。
#define ___swab16(x)
{
__u16 __x = (x);
((__u16)(
(((__u16)(__x) & (__u16)0x00ffU) << 8) |
(((__u16)(__x) & (__u16)0xff00U) >> 8) ));
}
#define ___swab32(x)
{
__u32 __x = (x);
((__u32)(
(((__u32)(__x) & (__u32)0x000000ffUL) << 24) |
(((__u32)(__x) & (__u32)0x0000ff00UL) << 8) |
(((__u32)(__x) & (__u32)0x00ff0000UL) >> 8) |
(((__u32)(__x) & (__u32)0xff000000UL) >> 24) ));
}
寫法很嚴謹。簡化版的:
#define swab32(x) ((x&0x000000ff) << 24 | (x&0x0000ff00) << 8 | (x&0x00ff0000) >> 8 | (x&0xff000000) >> 24)
#define swab16(x) ((x&0x00ff) << 8 | (x&0xff00) >> 8)
內存地址從低地址向高地址增長,big endian是高位數據優先,即高位放在低地址。而little endian是低位數據優先,低位放在低地址。網絡字節序採用大端。
判斷字節序:
main()
{
int x = 0x1;
if ((*(char *)&x) == 0x1)
printf("little endian!\n");
else
printf("big endian!\n");
}
字節序轉換:
ntohs(n) = __swab16(n),
ntohl = __swab32(n)。
__swab16與__swab32函數定義如下所示。
#define ___swab16(x)
{
__u16 __x = (x);
((__u16)(
(((__u16)(__x) & (__u16)0x00ffU) << 8) |
(((__u16)(__x) & (__u16)0xff00U) >> 8) ));
}
#define ___swab32(x)
{
__u32 __x = (x);
((__u32)(
(((__u32)(__x) & (__u32)0x000000ffUL) << 24) |
(((__u32)(__x) & (__u32)0x0000ff00UL) << 8) |
(((__u32)(__x) & (__u32)0x00ff0000UL) >> 8) |
(((__u32)(__x) & (__u32)0xff000000UL) >> 24) ));
}
寫法很嚴謹。簡化版的:
#define swab32(x) ((x&0x000000ff) << 24 | (x&0x0000ff00) << 8 | (x&0x00ff0000) >> 8 | (x&0xff000000) >> 24)
#define swab16(x) ((x&0x00ff) << 8 | (x&0xff00) >> 8)