原创 hex格式介绍及转bin格式的源程序

2007-11-1 20:31 5435 4 5 分类: MCU/ 嵌入式

最近在写一个编程器,遇到hex格式文件转换为bin格式文件的问题,通过查阅各类资料,对该格式有了了解。所以在此介绍一下hex格式,并将自己的程序思路写下来,供大家探讨一下。<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />


       Intel HEX文件是记录文本行的ASCII文本文件,在Intel HEX文件中,每一行是一个HEX记录,由十六进制数组成的机器码或者数据常量。Intel HEX文件经常被用于将程序或数据传输存储到ROMEPROM,大多数编程器和模拟器使用Intel HEX文件。


       很多编译器的支持生成HEX格式的烧录文件,尤其是Keil c。但是编程器能够下载的往往是BIN格式,因此HEXBIN是每个编程器都必须支持的功能。


       HEX格式文件以行为单位,每行由“:”(0x<?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />3a)开始,以回车键结束(0x0d,0x0a)。行内的数据都是由两个字符表示一个16进制字节,比如”01”就表示数0x01”0a”,就表示0x0a。对于16位的地址,则高位在前低位在后,比如地址0x010a,在HEX格式文件中就表示为字符串”010a”。下面为HEX文件中的一行:


:10000000FF0462FF051EFF0A93FF0572FF0A93FFBC


       “:”表示一行的开始。


       “:”后的第12个字符“10表示本行包含的数据的长度,这里就是0x1016个。


       3456个字符“0000表示数据存储的起始地址,这里表示从0x0000地址开始存储16个数据,其中高位地址在前,低位地址在后。


       78个字符“00表示数据的类型。该类型总共有以下几种:


00 ----数据记录      
01 ----
文件结束记录
02 ----
扩展段地址记录
04 ----
扩展线性地址记录


这里就是0x00即为普通数据记录。


自后的32个字符就是本行包含的数据,每两个字符表示一个字节数据,总共有16个字节数据跟行首的记录的长度相一致。


最后两个字符表示校验码。


每个HEX格式的最后一行都是固定为:


:00000001FF


       以上的信息其实就足够进行HEXBIN格式的程序的编写。首先我们只处理数据类型为0x000x01的情况。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);


              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("不支持扩展段地址记录!");


                   return 0;


              }


              //当数据类型为时,表示本行包含的是扩展线性地址记录


              if (dataType == 0x04)


              {


                   printf("不支持扩展线性地址记录!");


                   return 0;


              }


         }


     }


     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,myFile);


     }


     return 0;


}


 

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户131114 2008-8-20 04:26

这个程序经过多少测试? 如果测试足够完全的话,想直接使用,但没有看到License,是否可以免费使用? 得到文件大小: fseek(flash_file, 0L, SEEK_END); flash_size = (uint16)ftell(flash_file);
相关推荐阅读
用户1211959 2009-12-26 23:07
avr上的贪吃蛇游戏源码
∷�a class="" title="" href="http://www.helianpcb.com/" target="">线路板--公元1903年,Albert Hanson最先采用了“...
用户1211959 2009-08-03 21:09
[原创]蓝牙设备开发的三种方式
蓝牙设备开发一般包含蓝牙芯片及主机的开发。主机部分根据应用情况可以是PC,单片机,ARM等。蓝牙通讯协议是一组协议的集合,从最底层的硬件驱动,到上层的通讯协议,都由明确的规定。蓝牙设备必须实现这些协议...
用户1211959 2009-08-03 21:04
CSR 并口下载线原理图
...
用户1211959 2009-08-03 21:03
[原创]蓝牙串口与PC蓝牙管理器建立链接
本示例采用windows自带的蓝牙管理器与蓝牙模块建立链接。步骤如下:(1)    打开蓝牙管理器 (2)点击添加按钮后复选设备已经准备好的复选框 (3)    点击下一步进行设备的搜索,搜索到串口蓝...
用户1211959 2009-08-03 21:02
[原创]蓝牙串口的使用
进行串口通讯(1) 使用串口调试工具打开指定端口     (2) 如同普通串口一样,进行收发操作 AVR串口jtag的改造(1) 将蓝牙串口上的波特率设置跳线放到19200档上。<?xml:na...
用户1211959 2009-07-08 22:07
[出售]蓝牙串口
串口蓝牙适配器主要是用于替代传统的有线串口使用的场合。本蓝牙串口可以配置为主设备或者从设备,从而可以单独使用或者两个配对使用。<?xml:namespace prefix = o ns = "u...
EE直播间
更多
我要评论
1
4
关闭 站长推荐上一条 /3 下一条