ARM存储格式之 大端小端

本文深入探讨了计算机存储中的大端模式与小端模式,包括它们在内存、网络传输和编程判断处理器模式的应用,提供了具体例子和代码实现。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

开头讲个有关大端小端的故事:

  端模式(Endian)的这个词出自Jonathan Swift书写的《格列佛游记》。这本书根据将鸡蛋敲开的方法不同将所有的人分为两类,从圆头开始将鸡蛋敲开的人被归为Big Endian,从尖头开始将鸡蛋敲开的人被归为Littile Endian。小人国的内战就源于吃鸡蛋时是究竟从大头(Big-Endian)敲开还是从小头(Little-Endian)敲开。在计算机业Big Endian和Little Endian也几乎引起一场战争。  

 

  我们知道在内存中数据是以字节为单位进行存储的,每个地址单元对应着一个字节(byte),一个字节为8位(bite)。但是很多时候数据除了8bit额char外,还有16bit的short,32位的long型(要看具体的编译器),必然存在多字节安排的问题。不同的计算机存放多字节值的顺序不同,有些机器在起始地址存放低位字节(低位先存),即小端模式;有的机器在起始地址存放高位字节(高位先存),即大端模式。基于Intel的CPU,采用的是低位先存。而KEIL C51则为大端模式。大端小端对应着数据在存储器中的存放顺序。

  同时,在网络传输中,网络协议需要指定网络字节顺序,TCP/IP协议中使用16位整数和32位整数的高位先存模式,对应我们的大端模式。

  下面是两个具体例子:

  16bit宽的数0x1234在Little-endian模式(以及Big-endian模式)CPU内存中的存放方式(假设从地址0x4000开始存放)为:

内存地址小端模式存放内容大端模式存放内容
0x40000x340x12
0x40010x120x34

  32bit宽的数0x12345678在Little-endian模式以及Big-endian模式)CPU内存中的存放方式(假设从地址0x4000开始存放)为:

内存地址小端模式存放内容大端模式存放内容
0x40000x780x12
0x40010x560x34
0x40020x340x56
0x40030x120x78

  联合体union的存放顺序是所有成员都从低地址开始存放,利用该特性可以轻松地获得了CPU对内存采用Little-endian还是Big-endian模式读写。

  写程序判断处理器是Little-endian模式,还是Big-endian模式,可以通过以下程序:

  1>.通过将int强制类型转换成char单字节,通过判断起始存储位置。

1 void main(int argc, char **argv)
2 {
3 int i = 1;
4 char *cp = (char *)&i; //前面是指针运算符*,前值类型转换。后面是取地址符号。
5   if (*cp) //如果此时cp指向的内存为1的话,则为小端,否则为大端。
6   printf("Little Endian\n");
7 else
8 printf("Big Endian\n");
9
10 exit(EXIT_SUCCESS);
11 }

注释:如果小端方式中(i占至少两个字节的长度)则i所分配的内存最小地址那个字节中就存着1,其他字节是0.大端的话则1在i的最高地址字节处存放,char是一个字节,所以强制将char型量p指向i则p指向的一定是i的最低地址,那么就可以判断p中的值是不是1(或者为0,也即是假)来确定是不是小端。

或者如下程序:

void main()
{
short int x;

char x0,x1;

x
=0x1122;

x0
=((char*)&x)[0]; //低地址单元
x1=((char*)&x)[1]; //高地址单元

if (0x11 == x0 && 0x22 == x1)
{
cout
<< "Big_endian" << endl;
}
else
{
cout
<< "Little_endian" << endl;
}
}

2>.利用联合体union的存放顺序是所有成员都从低地址开始存放,判断处理器模式。

bool checkCPU( )
{
{
union w
{
int a;
char b;

} c;
c.a
= 1;
return(c.b ==1);
}
}

以及如下程序:

bool isLittleEndian()
{
union _dword
{
int all;
struct _bytes
{
char byte0;
char pad[3];
}bytes;
}dword;

dword.all
=0x87654321;

return (0x21==dword.bytes.byte0);
}

分析:如果你的处理器调用函数isLittleEndian返回1,那么说明你的处理器为little endian,否则为big endian.注意,如果在little endian处理器上,byte0和pad按内存从低到高的存放顺序:LOW->byte0 pad[0] pad[1] pad[2] ->HIGH;0x87654321按内存从低到高的存放顺序:  0x21  0x43   0x65   0x87, 可见byte0对应到0x21。所以通过判断dword中第一个字节dword.bytes.byte0是否与0x21相等就可以看出是否是little endian。

 

 

 PS:本文后面几个程序都是从网上摘抄过来的,对于联合体Union不是很清楚的可以参看http://blog.sina.com.cn/s/blog_3fa943920100ob37.html 我觉得这篇博文写的很详细。通过本文对大端小端模式有了一个比较深的理解吧。也希望对读者有些许帮助,谢谢!接下来我会总结下联合Union结构和Struct的区别。

转载于:https://www.cnblogs.com/yin-jingyu/archive/2011/09/30/2196110.html

<think>嗯,我需要回答用户关于ARM指令是大端还是小端存储的问题。首先,我得回忆一下ARM架构的基本知识。ARM处理器作为广泛使用的架构,其存储方式可能因版本或配置不同而有所变化。 记得ARM架构支持两种字节序:大端(Big-endian)和小端(Little-endian)。但默认情况下,多ARM处理器使用的是小端模式。例如,Cortex-M系列通常是小端。不过,某些应用场景可能需要大端,比如网络设备,这时候处理器可以配置为大端模式。 另外,需要区分ARM指令本身的字节序和存储的字节序。用户的问题可能更多关注存储,但指令存储通常是小端,尤其是在Thumb模式下。不过,指令的字节序可能更复杂,可能涉及半字的排列。 还需要确认不同ARM版本是否有变化,比如ARMv7和ARMv8是否有差异。查阅资料的话,ARM官方文档指出,默认是小端,但支持大端配置。因此,回答时应强调灵活性,同时说明常见情况。 最后,要确保回答结构清晰,分点说明,并给出示例,比如内存据的存储方式,帮助用户理解。可能还需要提醒用户具体应用时查阅手册,因为配置可能影响结果。</think>ARM处理器的字节序(大端/小端)设计具有灵活性,具体取决于架构版本和配置。以下分步说明: 1. **基本设计原则** ARM架构**原生支持小端(Little-endian)存储模式**,这也是绝大多ARM芯片的默认配置。例如: - Cortex-M系列全系强制使用小端 - 移动设备处理器(如Cortex-A系列)默认小端 2. **大端模式支持** 部分ARM处理器通过配置可切换到大端模式(Big-endian),典型应用场景包括: - 网络协议处理(如TCP/IP头的大端格式) - 传统大端系统兼容 - ARMv7及以上架构提供BE-8(大端,指令小端)配置选项 3. **指令据的区别** - **指令存储**:固定为小端格式(包括Thumb/ARM指令集) - **存储**:可通过程序状态寄存器(CPSR)的E位动态切换 示例代码: ```asm ; 切换到大端模式 MRS r0, CPSR ORR r0, r0, #0x200 ; 设置第9位(E位) MSR CPSR_c, r0 ``` 4. **内存访问示例** 假设将0x12345678存入地址0x1000: | 地址 | 小端模式内容 | 大端模式内容 | |--------|--------------|--------------| | 0x1000 | 0x78 | 0x12 | | 0x1001 | 0x56 | 0x34 | | 0x1002 | 0x34 | 0x56 | | 0x1003 | 0x12 | 0x78 | 5. **实际开发注意事项** - 使用联合体检测字节序: ```c union EndianTest { uint32_t i; uint8_t c[4]; } test = {0x12345678}; // 若test.c[0]==0x78则为小端 ``` - 网络通信时必须显式使用htonl()等转换函 - 外设寄存器访问需确认设备要求的字节序 结论:**ARM指令存储本身固定为小端据处理默认小端但可配置大端**。开发时应通过芯片手册确认具体配置,在跨平台场景中建议显式处理字节序转换。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值