hex格式介绍及转bin格式的源程序
源:
  Intel HEX⽂件是记录⽂本⾏的ASCII⽂本⽂件,在Intel HEX⽂件中,每⼀⾏是⼀个HEX记录,由⼗六进制数组成的机器码或者数据常量。Intel HEX⽂件经常被⽤于将程序或数据传输存储到ROM、EPROM,⼤多数编程器和模拟器使⽤Intel HEX⽂件。
  很多编译器的⽀持⽣成HEX格式的烧录⽂件,尤其是Keil c。但是编程器能够下载的往往是BIN格式,因此HEX转BIN是每个编程器都必须⽀持的功能。
  HEX格式⽂件以⾏为单位,每⾏由“:”(0x3a)开始,以回车键结束(0x0d,0x0a)。⾏内的数据都是由两个字符表⽰⼀个16进制字节,⽐如”01”就表⽰数0x01;”0a”,就表⽰0x0a。对于16位的地址,则⾼位在前低位在后,⽐如地址0x010a,在HEX格式⽂件中就表⽰为字符串”010a”。下⾯为HEX⽂件中的⼀⾏:
:10000000FF0462FF051EFF0A93FF0572FF0A93FFBC
  “:”表⽰⼀⾏的开始。
  “:”后的第1,2个字符“10”表⽰本⾏包含的数据的长度,这⾥就是0x10即16个。
  第3,4,5,6个字符“0000”表⽰数据存储的起始地址,这⾥表⽰从0x0000地址开始存储16个数据,其中⾼位地址在前,低位地址在后。
  第7,8个字符“00”表⽰数据的类型。该类型总共有以下⼏种:
00 ----数据记录      01 ----⽂件结束记录 02 ----扩展段地址记录 04 ----扩展线性地址记录
这⾥就是0x00即为普通数据记录。
⾃后的32个字符就是本⾏包含的数据,每两个字符表⽰⼀个字节数据,总共有16个字节数据跟⾏⾸的记录的长度相⼀致。
最后两个字符表⽰校验码。
每个HEX格式的最后⼀⾏都是固定为:
:00000001FF
  以上的信息其实就⾜够进⾏HEX转BIN格式的程序的编写。⾸先我们只处理数据类型为0x00及0x01的情况。0x02表⽰对应的存储地址超过了64K,由于我的编程器只针对64K以下的单⽚机,因此在次不处理,0x04也是如此。
  我的编程思路是从⽂件中⼀个⼀个读出字符,根据“:”判断⼀⾏的开始,然后每两个字符转换成⼀个字节,并解释其对应的意义。然后将数据从该⾏中剥离出来保存到缓冲区中,并最终输出到⽂件中。
  具体程序如下,该程序在VC2005下采⽤控制台项⽬编译,需要在release下编译,在debug模式中会提⽰⼀个dll⽂件⽆法到,这可能是VC⾃⾝的错误。
// hextobin.cpp : 定义控制台应⽤程序的⼊⼝点。
#include "stdafx.h"
#include <malloc.h>
#include <memory.h>
typedef unsigned char BYTE;
//将两个字符转化为⼀个字节量
void CharToByte(char* pChar,BYTE* pByte)
{
char h,l;
h=pChar[0];    //⾼位
l=pChar[1];    //低位
if(l>='0'&&l<='9')
l=l-'0';
else if(l>='a' && l<='f')
l=l-'a'+0xa;
else if(l>='A' && l<='F')
l=l-'A'+0xa;
if(h>='0'&&h<='9')
h=h-'0';
else if(h>='a' && h<='f')
h=h-'a'+0xa;
else if(h>='A' &&h <='F')
h=h-'A'+0xa;
*pByte=(BYTE)h*16+l;
}
int _tmain(int argc, _TCHAR* argv[])
{
char fileName[100];
char data[2];
BYTE *outBuf;
FILE *myFile;
int len;
int i;
BYTE adressHigh;
BYTE adressLow;
BYTE dataLen;
BYTE dataType;
BYTE byteData;
int totalLen;
totalLen = 0;
len = 0;
adressHigh = 0;
adressLow = 0;
dataLen = 0;
dataType = 0;
printf("请输⼊HEX格式⽂件名:");
scanf_s("%s",fileName);
printf("\n");
if (fopen_s(&myFile,fileName,"r") != 0)
{
printf("打开⽂件%s失败!",fileName);
}
/
/将⽂件长度计算出来⽤于申请存储数据的缓冲区
while (!feof(myFile))
{
++len;
fgetc(myFile);
}
rewind(myFile);
//因为是每两个字符表⽰⼀个字节,所以最⼤的数据个数要少于⽂件字符个数的⼀半    outBuf = (BYTE*)malloc(len/2);
memset(outBuf,0xff,len/2);
while (!feof(myFile))
{
//:号表⽰⼀⾏的开始
if (fgetc(myFile) == ':')
{
//⼀⾏的头两个字符表⽰该⾏包含的数据长度
data[0] = fgetc(myFile);
data[1] = fgetc(myFile);
CharToByte(data,&dataLen);
//⼀⾏的第、个字符表⽰数据存储起始地址的⾼位
data[0] = fgetc(myFile);
data[1] = fgetc(myFile);
CharToByte(data,&adressHigh);
/
/⼀⾏的第、个字符表⽰数据存储起始地址的低位
data[0] = fgetc(myFile);
data[1] = fgetc(myFile);
hex字符串是什么CharToByte(data,&adressLow);
//⼀⾏的第、个字符表⽰数据类型
data[0] = fgetc(myFile);
data[1] = fgetc(myFile);
CharToByte(data,&dataType);
//当数据类型为时,表⽰本⾏包含的是普通数据记录
if (dataType == 0x00)
{
for (i=0;i<dataLen;i++)
{
data[0] = fgetc(myFile);
data[1] = fgetc(myFile);
CharToByte(data,&byteData);
outBuf[adressHigh*256+adressLow+i] = byteData;
}
totalLen += dataLen;
}
//当数据类型为时,表⽰到了最后⼀⾏
if (dataType == 0x01)
{
printf("⽂件结束记录!");
}
//当数据类型为时,表⽰本⾏包含的是扩展段地址记录
if (dataType == 0x02)
{
printf("不⽀持扩展段地址记录!");
return0;
}
//当数据类型为时,表⽰本⾏包含的是扩展线性地址记录
if (dataType == 0x04)
{
printf("不⽀持扩展线性地址记录!");
return0;
}
}
}
fclose(myFile);
printf("请输⼊保存的BIN格式⽂件名:");
scanf_s("%s",fileName);
if (fopen_s(&myFile,fileName,"w") != 0)
{
printf("打开⽂件%s失败!",fileName);
}
for (i=0;i<totalLen;i++)
{
fputc(outBuf[i],myFile);
}
return0;
}

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。