bin文件转化成hex文件的实现

来源:互联网 发布:艺术网站源码 编辑:程序博客网 时间:2024/06/06 02:22
对于嵌入式而言,hex文件可能大家再熟悉不过了,对,我们大学时学习的51单片机编写的代码在keil上编译后就生成了hex文件。那bin文件又是什么意思呢,它又和hex文件的区别在哪?这也不是本文的重点,下面简单的描述下:

最通俗的来讲,hex是带地址的,用下载器下载时,不需要设置偏移地址,它是文件流格式的,都是标准的ASCII码。而bin文件是不带地址的,全部是二进制数据流,打住一下,其实就是我们所谓的机器代码。有兴趣的同学,可以尝试着用反汇编,得到的就是汇编代码了。我所用的开发板S3C2440在ADS1.2上编译形成的代码就是bin格式流,用j-flash打开文件的时候就需要填入偏移地址,三星平台flash偏移地址为0,而stm32平台flash偏移地址就是0x08000000.

本来是应该要描述下hex文件的数据格式,这个就留着下一篇文章来描述,其实百度上也有很多。下一张是hex文件转换为bin文件,刚好和本文相反。说了这么多,下面就直接贴出代码了,有不详细的可以给我留言,同时也欢迎大家喷我。

代码是在VC6.0上面实现的:

首先新建bin2hex.h文件

[cpp] view plain copy print?在CODE上查看代码片派生到我的代码片
  1. #ifndef BIN2HEX_H  
  2. #define BIN2HEX_H  
  3. typedef unsigned char uint8_t;  
  4. typedef unsigned short uint16_t;  
  5. typedef unsigned long uint32_t;  
  6. /*********************************** 
  7. ********************************************* 
  8. 就是每次读写bin文件N个字节,然后再转化为hex格式流,hex格式流长度计算方式 
  9. : + 长度 + 地址 + 类型 + N个数据(N >= 0) + 校验 
  10. 1 + 2    + 4    + 2    + N * 2           + 2 
  11. ********************************************************************************/  
  12. #define NUMBER_OF_ONE_LINE      0x20  
  13. #define MAX_BUFFER_OF_ONE_LINE  (NUMBER_OF_ONE_LINE * 2 + 11)   
  14. typedef struct {  
  15.     uint8_t len;  
  16.     uint8_t addr[2];  
  17.     uint8_t type;  
  18.     uint8_t *data;  
  19. } HexFormat;  
  20.   
  21. typedef enum {  
  22.     RES_OK = 0,                     //操作完成  
  23.     RES_BIN_FILE_NOT_EXIST,         //相当于bin文件不存在,包括输入的路径可能存在不正确  
  24.     RES_HEX_FILE_PATH_ERROR         //目标文件路径可能输入有误            
  25. } RESULT_STATUS;  
  26.   
  27. RESULT_STATUS BinFile2HexFile(char *src, char *dest);  
  28. #endif  

新建bin2hex.c 文件

[cpp] view plain copy print?在CODE上查看代码片派生到我的代码片
  1. #include "bin2hex.h"  
  2. #include <stdio.h>  
  3. /******************************************************************************** 
  4. input: 
  5.     dest: 为转换后的结果 
  6.     p->addr[0]: 高地址 
  7.     p->addr[1]: 低地址 
  8.     p->type: 记录类型 
  9.     p->data: 为bin格式流有效数据指针 
  10.     p->len: 为bin格式流有效数据长度 
  11. output: 
  12.     返回有效数据的长度 
  13. ********************************************************************************/  
  14. uint16_t BinFormatEncode(uint8_t *dest, HexFormat *p)  
  15. {  
  16.     uint16_t offset = 0;  
  17.     uint8_t check = 0, num = 0;     //:(1) + 长度(2) + 地址(4) + 类型(2)  
  18.     sprintf(&dest[offset], ":%02X%02X%02X%02X", p->len, p->addr[0], p->addr[1], p->type);  
  19.     offset += 9;                    //hex格式流数据指针偏移2  
  20.     check = p->len + p->addr[0] + p->addr[1] + p->type; //计算校验和  
  21.     while (num < p->len)          //当数据长度不为0,继续在之前的hex格式流添加数据  
  22.     {                                 
  23.         sprintf(&dest[offset], "%02X", p->data[num]);  
  24.         check += p->data[num];       //计算校验和  
  25.         offset += 2;                //hex格式数据流数据指针偏移2  
  26.         num++;                      //下一个字符  
  27.     }  
  28.     check = ~check + 1;             //反码+1  
  29.     sprintf(&dest[offset], "%02X", check);  
  30.     offset += 2;          
  31.     return offset;                  //返回hex格式数据流的长度  
  32. }  
  33.   
  34. RESULT_STATUS BinFile2HexFile(char *src, char *dest)  
  35. {  
  36.     FILE *src_file, *dest_file;  
  37.     uint16_t tmp;  
  38.     HexFormat gHexFor;  
  39.     uint32_t low_addr = 0, hign_addr = 0;  
  40.     uint8_t buffer_bin[NUMBER_OF_ONE_LINE], buffer_hex[MAX_BUFFER_OF_ONE_LINE];  
  41.     uint32_t src_file_length;  
  42.     uint16_t src_file_quotient, cur_file_page = 0;  
  43.     uint8_t src_file_remainder;  
  44.     src_file = fopen(src, "rb");        //源文件为bin文件,以二进制的形式打开  
  45.     if (!src_file)                      //这里也是相当于用来检查用户的输入是否准备  
  46.     {  
  47.         return RES_BIN_FILE_NOT_EXIST;  
  48.     }  
  49.     dest_file = fopen(dest, "w");       //目的文件为hex文件,以文本的形式打开  
  50.     if (!dest_file)                       
  51.     {  
  52.         return RES_HEX_FILE_PATH_ERROR;  
  53.     }  
  54.     fseek(src_file, 0, SEEK_END);       //定位到文件末   
  55.     src_file_length = ftell(src_file);  
  56.     fseek(src_file, 0, SEEK_SET);       //重新定位到开头,准备开始读取数据  
  57.     src_file_quotient = (uint16_t)(src_file_length / NUMBER_OF_ONE_LINE);   //商,需要读取多少次  
  58.     src_file_remainder = (uint8_t)(src_file_length % NUMBER_OF_ONE_LINE);   //余数,最后一次需要多少个字符  
  59.     gHexFor.data = buffer_bin;          //指向需要转换的bin数据流  
  60.     while (cur_file_page < src_file_quotient)  
  61.     {  
  62.         fread(buffer_bin, 1, NUMBER_OF_ONE_LINE, src_file);  
  63.         gHexFor.len = NUMBER_OF_ONE_LINE;  
  64.         if ((low_addr & 0xffff0000) != hign_addr && hign_addr != 0)     //只有大于64K以后才写入扩展线性地址,第一次一般是没有  
  65.         {  
  66.             hign_addr = low_addr & 0xffff0000;                            
  67.             gHexFor.addr[0] = (uint8_t)((hign_addr & 0xff000000) >> 24);  
  68.             gHexFor.addr[1] = (uint8_t)((hign_addr & 0xff0000) >> 16);  
  69.             gHexFor.type = 4;                                         
  70.             gHexFor.len = 0;                                            //记录扩展地址              
  71.             tmp = BinFormatEncode(buffer_hex, &gHexFor);  
  72.             fwrite(buffer_hex, 1, tmp, dest_file);  
  73.             fprintf(dest_file, "\n"); ;  
  74.         }  
  75.         gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);  
  76.         gHexFor.addr[1] = (uint8_t)(low_addr & 0xff);  
  77.         gHexFor.type = 0;                                               //数据记录  
  78.         tmp = BinFormatEncode(buffer_hex, &gHexFor);  
  79.         fwrite(buffer_hex, 1, tmp, dest_file);  
  80.         fprintf(dest_file, "\n"); ;  
  81.         cur_file_page++;  
  82.         low_addr += NUMBER_OF_ONE_LINE;   
  83.     }  
  84.     if (src_file_remainder != 0)        //最后一次读取的个数不为0,这继续读取  
  85.     {  
  86.         fread(buffer_bin, 1, src_file_remainder, src_file);  
  87.         gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);  
  88.         gHexFor.addr[1] = (uint8_t)(low_addr & 0x00ff);  
  89.         gHexFor.len = src_file_remainder;  
  90.         gHexFor.type = 0;                                               //数据记录                        
  91.         tmp = BinFormatEncode(buffer_hex, &gHexFor);  
  92.         fwrite(buffer_hex, 1, tmp, dest_file);  
  93.         fprintf(dest_file, "\n"); ;  
  94.     }  
  95.     gHexFor.addr[0] = 0;                  
  96.     gHexFor.addr[1] = 0;  
  97.     gHexFor.type = 1;                   //结束符  
  98.     gHexFor.len = 0;  
  99.     tmp = BinFormatEncode(buffer_hex, &gHexFor);  
  100.     fwrite(buffer_hex, 1, tmp, dest_file);  
  101.     fprintf(dest_file, "\n"); ;  
  102.     fclose(src_file);  
  103.     fclose(dest_file);  
  104.     return RES_OK;  
  105. }  

新建main.c文件,这里是带参数的,主要是方便批处理,是另有用途。

[cpp] view plain copy print?在CODE上查看代码片派生到我的代码片
  1. #include <stdio.h>  
  2. #include "bin2hex.h"  
  3. int main(int argc, char *argv[])  
  4. {  
  5.     RESULT_STATUS res;  
  6.     if (argc != 3)  
  7.     {  
  8.         printf("input para doesn't match\r\n");  
  9.         return -1;  
  10.     }  
  11.     res = BinFile2HexFile(argv[1], argv[2]);  
  12.     switch (res)  
  13.     {  
  14.         case RES_OK:  
  15.             printf("hex file to bin file success!\r\n");  
  16.             return -1;  
  17.         case RES_BIN_FILE_NOT_EXIST:  
  18.             printf("bin file doesn't exist!\r\n");  
  19.             return -1;  
  20.         case RES_HEX_FILE_PATH_ERROR:  
  21.             printf("hex file path is error, please check it!\r\n");  
  22.             return -1;  
  23.     }  
  24.     return 0;  
  25. }  

三个源文件,编译生成bin2hex.c文件。

下面描述下用法:

把bin2hex.c文件拷贝到c盘根目录下,再拷贝一个需要转换的bin文件,这里我就拷贝了一个hwb.bin文件。然后点击菜单开始->运行->输入cmd->进入dos窗口->调整当前目录为c:,这个如果不知道的可以百度一下,无法就是就是命令cd.

最后输入命令:bin2hex hwb.bin hwb.hex,输入后,可以看到提示说转换成功,大家再检查下是否有一个hex文件,本文的代码支持大于64K,大家把转换后的hex下载到单片机上运行试试看。

0 0
原创粉丝点击