大端小端

一个16位整数,由两个字节组成。内存中存储这两个字节有两种方法:一种是将低序字节存储在起始地址,称为little-endian字节序,另一种方法 是将高序字节存储在起始地址,称为big-endian字节序。术语little-endian和big-endian表示多字节的哪一端存储在该值的起 始地址。
 
    例如:0x0102
    little-endian: A    0x02
                   A+1  0x01
    big-endian:    A    0x01
                   A+1  0x02
 
    UNP上面有测试字节排序的函数。考虑到网络字节序为big-endian,修改程序来比较一下Host Byte Order和Internet Byte Order的区别。
 

/*
 * filename: byteorder.c
 */


#include <stdio.h>
#include <netinet/in.h>

union {
        short s;
        char c[sizeof(short)];
}un;
short test = 0x0203;

void byteorder(char *msg)
{
        printf("%s", msg);
        if (sizeof(short) == 2) {
                if (un.c[1] == (test & 0x00ff)) {
                        printf("big-endian/n");
                }
                else if (un.c[0] == (test & 0x00ff)) {
                        printf("small-endian/n");
                }
                else {
                        printf("unkowned/n");
                }
        }
        else {
                printf("sizeof(short) = %d/n", sizeof(short));
        }
}

int main(void)
{
        char *msg1 = "Host Byte Order: ";
        char *msg2 = "Network Byte Order: ";

        un.s = test;
        byteorder(msg1);
        un.s = htons(test);
        byteorder(msg2);

        return 0;
}

测试结果如下:

[armlinux@lqm byteorder]$ ./byteorder
Host Byte Order: small-endian
Network Byte Order: big-endian

    在进行嵌入式移植的时候,要考虑到字节序的问题。网络编程更是要考虑字节序的转换,一般使用四个函数htons、htonl、ntohs、ntohl。其 中,h代表host,n代表network,s代表short,l代表long。在这里,把s看成一个16位的值(如TCP或UDP的端口号),把l看作 32位的值(如IPv4)。要进行IPv6的编程的话,需要进行相应的改变了。


2008/01/17

    想到了两个问题。一个是如何可靠的判断存储格式是大端还是小端,另一个是如何进行大小端转换。自己写了程序,可以实现,但是发现在细节上考虑的还是不周全。参考了一下Uboot的源代码,实现如下:

#ifndef ENDIAN_H_
#define ENDIAN_H_

#include <types.h>

#define BIG_ENDIAN 0
#define SMALL_ENDIAN 1

typedef union {
        u16 a;
        u8 b[2];
}ENDIAN;

/* judge cpu is big endian(0) or small endian(1) */
bool judge_endian(void)
{
        ENDIAN t;

        t.a = 0x0001U;

        return (bool)(t.b[0] == 1);
}

/* type translation */
#define SWAB16(x) /
                ((u16)( /
                        ((((u16)(x) & (u16)0x00FFU)) << 8) | /
                        ((((u16)(x) & (u16)0xFF00U)) >> 8)))

#define SWAP32(x) /
                ((u32)( /
                        ((((u32)(x) & (u32)0x000000FFUL)) << 24) | /
                        ((((u32)(x) & (u32)0x0000FF00UL)) << 8) | /
                        ((((u32)(x) & (u32)0x00FF0000UL)) >> 8) | /
                        ((((u32)(x) & (u32)0xFF000000UL)) >> 24)))

#endif /* ENDIAN_H_ */


    其实在htonl等函数的实现中,核心都是SWAP系列的转换,要注意的是数据的类型这个细节。判断大小端则是在编译前设定好了,并不是执行程序时先判断是大端还是小端,而是通过配置选项选定大端还是小端。

 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值