QuietE

转:HEX文件格式

0
阅读(1435)

 

记录一下:

HEX文件以行为单位。每行以字符 ‘:’ (0x3a)开头,以回车换行符0x0d, 0x0a为结束。每行开始和结束之间的所有内容,都是以字符形式表现的。例如数据如果是 0x1A ,那么转换到HEX格式的行里面就是0x31 0x41。如果数据是16bit的,例如地址,则先显示高位,后显示底位。例如 0x1234,转换成HEX格式文件后变成 0x31 0x32 0x33 0x34,显示出来以后就是1234。将数据部分内容每2个字符看做一个HEX数据,例如:

:020000040000FA , 我把它看做 0x02 0x00 0x00 0x04 0x00 0x00 0xFA

第一个 0x02 为数据长度。

紧跟着后面的0x00 0x00 为地址。

再后面的0x04为数据类型,类型共分一下几类:

'00' Data Record

'01' End of File Record

'02' Extended Segment Address Record

'03' Start Segment Address Record

'04' Extended Linear Address Record

'05' Start Linear Address Record

然后,接着0x04后面的两个 0x00 0x00就是数据。最后一个0xFA是校验码。

       HEX文件的每一行都是这样的格式:

<0x3a>

[数据长度1Byte]

[数据地址2Byte]

[数据类型1Byte]

[数据nByte]

[校验1Byte]

<0x0d>

<0x0a>

在例如:

:1000000018F09FE518F09FE518F09FE518F09FE5C0

安装上面的数据行格式分析如下:

<0x3a>

[数据长度1Byte]

10

[数据地址2Byte]

00 00

[数据类型1Byte]

00

[数据nByte]

18F09FE518F09FE518F09FE518F09FE5

[校验1Byte]

C0

<0x0d>

<0x0a>

每行中的数据并不是一定有的,第二个直接的数据长度为0,那么这行就没有数据。

由于每行标识数据地址的只有2Byte,所以最大只能到64K,为了可以保存更大数据地址数据,就有了Extended LinearAddress Record。如果这行的数据类型是0x04,那么,这行的数据就是随后数据的基地址。例如:

:020000040004F6

:1000000018F09FE518F09FE518F09FE518F09FE5C0

:1000100018F09FE5805F20B9F0FF1FE518F09FE51D

第一行,是Extended LinearAddress Record,里面的基地址是0x0004,第二行是Data Record,里面的地址值是0x0000。那么数据18F09FE518F09FE518F09FE518F09FE5要写入FLASH中的地址为(0x0004 << 16) | 0x0000,也就是写入FLASH的0x40000这个地址。同样,第三行的数据的写入地址为0x40010。当一个HEX文件的数据超过64k的时候,文件中就会出现多个Extended Linear Address Record。

End of File Record 行是每一个HEX文件的最后一行。例如:

:00000001FF

这样的一行数据内容是固定的,数据长度为0,地址为0。

校验值:每一行的最后一个值为此行数据的校验和。例如:

:1000000018F09FE518F09FE518F09FE518F09FE5C0 这行中的 0xC0

:1000100018F09FE5805F20B9F0FF1FE518F09FE51D 这行中的 0x1D

校验和的算法为:计算从0x3A 以后(不包括0x3A)的所有各字节的和模256的余。即各字节二进制算术和,不计超过256的溢出值,然后用0x100减去这个算数累加和,得出得值就是此行得校验和。

Byyifan         2006.3.16

http://yifan.blog.com

什么是Intel HEX格式?

回答:

Intel HEX文件是记录文本行的ASCII文本文件,在Intel HEX文件中,每一行是一个HEX记录

由十六进制数组成的机器码或者数据常量,Intel HEX文件经常被用于将程序或数据传输

存储到ROM.EPROM,大多数编程器和模拟器使用Intel HEX文件.

记录格式

一个Intel HEX文件可以包含任意多的十六进制记录,每条记录有五个域,下面是一个记录的格式.

:llaaaatt[dd...]cc

每一组字母是独立的一域,每一个字母是一个十六进制数字,每一域至少由两个十六进制数字组成,下面是字节的描述.

:冒号 是每一条Intel HEX记录的开始

ll 是这条记录的长度域,他表示数据(dd)的字节数目.

aaaa 是地址域,他表示数据的起始地址

<如果是数据记录,这表示将要烧录的这条记录中的数据在EPROM中的偏移地址,

对于不支持扩展段地址和扩展线性地址的,如89C51,这就是此条记录的起始地址>

tt 这个域表示这条HEX记录的类型,他有可能是下面这几种类型

00 ----数据记录

01 ----文件结束记录

02 ----扩展段地址记录

04 ----扩展线性地址记录

dd   是数据域,表示一个字节的数据,一个记录可能有多个数据字节,字节数目可以

查看ll域的说明

cc   是效验和域,表示记录的效验和,计算方法是将本条记录冒号开始的所有字母对

<不包括本效验字和冒号> 所表示的十六进制数字

<一对字母表示一个十六进制数,这样的一个十六进制数为一个字节>

都加起来然后模除256得到的余数最后求出余数的补码即是本效验字节cc.

<例如:

:0300000002005E9D

cc=0x01+NOT((0x03+0x00+0x00+0x00+0x02+0x00+0x5E)%0x100)=0x01+0x9C=0x9D

C语言描述:

UCHAR cc;

cc=(UCHAR)~(0x03+0x00+0x00+0x00+0x02+0x00+0x5E);

cc++;

>

数据记录

IntelHEX文件由若干个数据记录组成,一个数据记录以一个回车和一个换行结束

                                         <回车为0x0d换行为0x0a>

比如下面的一条数据记录

:10246200464C5549442050524F46494C4500464C33

10   是此行记录数据的字节数目

2462   是数据在内存<将要烧写的eprom地址>中的起始地址

00     是记录类型00(是一个数据记录)

464C 到 464C 是数据

33   是此行记录的效验和

扩展线性地址记录(HEX386)

扩展线性地址记录也可称为 32位地址记录 和   HEX386记录,这个纪录包含高16(16-31位)位数据地址,这种扩展的线性记录总是有两个字节数据,像下面这样:

:02000004FFFFFC

02     是记录的数据字节数目

0000   是地址域这在扩展地址记录中总是0000

04     是记录类型04(扩展地址记录)

FFFF   是高16位地址

FC     是记录效验和,计算方法如下:

        01h + NOT(02h + 00h + 00h + 04h + FFh + FFh)

当一个扩展线性地址记录被读到后,扩展线性地址记录的数据区域将被保存

并应用到后面从Intel HEX文件中读出的记录,这个扩展线性记录一直有效,

直到读到下一个扩展线性记录.

绝对内存地址 = 数据记录中的地址 + 移位后的扩展线性地址

下面举例说明这个过程

从数据记录的地址域得到地址          2462

从扩展线性地址记录的地址域得到地址   FFFF

绝对内存地址                          FFFF2462

扩展段地址记录 (HEX86)

扩展段地址记录也被称为 HEX86记录,   包含 4-19位的数据地址段,

这个扩展段地址记录总是有两字节数据,如下:

:020000021200EA

02 是 记录中的数据字节数目

0000           是地址域,在扩展段地址记录中,这个域总是0000

02           是记录类型02(扩展段地址的标示)

1200     是该段的地址

EA     是效验和

计算如下:

01h + NOT(02h + 00h + 00h + 02h + 12h + 00h).

当扩展段地址记录被读后,扩展段地址将被存储并应用到以后从Intel HEX文件读出的记录,这个段地址一直有效直到读到下一个扩展段地址记录

绝对内存地址 = 数据记录中的地址 + 移位后的扩展段地址

数据记录中的地址域 移位后扩展段地址记录中的地址域

下面举例说明这个过程

从数据记录的地址域得到地址                  2 4 6 2

从扩展段地址记录的地址域得到地址          1 2 0 0

绝对内存地址                        0 0 0 1 4 4 6 2

文件结束记录(EOF)

一个Intel HEX文件必须有一个文件结束记录,这个记录的类型域必须是01,

一个EOF记录总是这样:

:00000001FF

00是记录中数据字节的数目

0000这个地址对于EOF记录来说无任何意义

01记录类型是01(文件结束记录标示)

FF是效验和计算如下

01h + NOT(00h + 00h + 00h + 01h).

========================

总结

形如

:BBAAAATTHHHH...HHHHCC

BB: Byte

AAAA:数据记录的开始地址,高位在前,地位在后

因为这个格式只支持8bits,地址被倍乘

所以,为了得到实际的PIC的地址,需要将 地址除以2

TT: Type

        00 数据记录

        01 记录结束

        04 扩展地址记录(表示32位地址的前缀,当然这种只能在 INHX32)

HHHH:一个字(Word)的数据记录,高Byte在前,低Byte在后

      TT之后,总共有 BB/2 个字 的数据

CC: 一个Byte的CheckSum

因为PIC16F873A只有4K的程序空间

所以,不会有 TT=04的 Linear Address Record