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); 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[i],myFile); } return 0; }