博客首页 注册 建议与交流 排行榜 加入友情链接
推荐 投诉 搜索: 帮助

CalmArrow

【打好基础】全力以赴,顺其自然【每天进步一点点】
  piaoxiang.cublog.cn

关于作者
姓名:CalmArrow(lqm)
职业:硕士在读
位置:山东济南
研究:嵌入式系统设计
联系:calmarrow@gmail.com

信仰:
    1、永远保持积极向上(积极的心态,积极的思考,积极的行动),享受生活给予的一切!
    2、做正确的事,正确的做事;用心去做,做到最好!
    3、如果你觉得不幸福,那么请你把幸福的门槛降低一些,不要把幸福拒之门外。。。

目标:
    软硬结合,打好基础,提高学习能力,完善知识体系,建立核心优势。

方法:
    理论与实践相结合
    深度与广度相结合
    理解与记忆相结合

说明:
    本Blog仅供学习之用,转载文章如涉及版权,请通知。原创作品如转载,请注明出处。
|| << >> ||
我的分类


big-endian and little-endian

文章说明:calmarrow(lqm)原创

文章引自:http://piaoxiang.cublog.cn

 
    一个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_ */

    测试程序包如下:

文件: endian.tar.gz
大小: 1KB
下载: 下载

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

发表于: 2006-12-30,修改于: 2008-01-17 16:05,已浏览1697次,有评论0条 推荐 投诉


网友评论
 发表评论