TShopping

 找回密碼
 註冊
搜索
查看: 2371|回復: 0

[分享] bin檔案和hex檔案互轉

[複製鏈接]
發表於 2020-9-13 22:48:06 | 顯示全部樓層 |閱讀模式
 
Push to Facebook
對於嵌入式而言,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檔案
  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. typedef enum {
  21.     RES_OK = 0,                     //操作完成
  22.     RES_BIN_FILE_NOT_EXIST,         //相当于bin文件不存在,包括输入的路径可能存在不正确
  23.     RES_HEX_FILE_PATH_ERROR         //目标文件路径可能输入有误         
  24. } RESULT_STATUS;

  25. RESULT_STATUS BinFile2HexFile(char *src, char *dest);
  26. #endif
複製代碼



新建bin2hex.c 檔案
  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. RESULT_STATUS BinFile2HexFile(char *src, char *dest)
  34. {
  35.     FILE *src_file, *dest_file;
  36.     uint16_t tmp;
  37.     HexFormat gHexFor;
  38.     uint32_t low_addr = 0, hign_addr = 0;
  39.     uint8_t buffer_bin[NUMBER_OF_ONE_LINE], buffer_hex[MAX_BUFFER_OF_ONE_LINE];
  40.     uint32_t src_file_length;
  41.     uint16_t src_file_quotient, cur_file_page = 0;
  42.     uint8_t src_file_remainder;
  43.     src_file = fopen(src, "rb");        //源文件为bin文件,以二进制的形式打开
  44.     if (!src_file)                      //这里也是相当于用来检查用户的输入是否准备
  45.     {
  46.         return RES_BIN_FILE_NOT_EXIST;
  47.     }
  48.     dest_file = fopen(dest, "w");       //目的文件为hex文件,以文本的形式打开
  49.     if (!dest_file)                     
  50.     {
  51.         return RES_HEX_FILE_PATH_ERROR;
  52.     }
  53.     fseek(src_file, 0, SEEK_END);       //定位到文件末
  54.     src_file_length = ftell(src_file);
  55.     fseek(src_file, 0, SEEK_SET);       //重新定位到开头,准备开始读取数据
  56.     src_file_quotient = (uint16_t)(src_file_length / NUMBER_OF_ONE_LINE);   //商,需要读取多少次
  57.     src_file_remainder = (uint8_t)(src_file_length % NUMBER_OF_ONE_LINE);   //余数,最后一次需要多少个字符
  58.     gHexFor.data = buffer_bin;          //指向需要转换的bin数据流
  59.     while (cur_file_page < src_file_quotient)
  60.     {
  61.         fread(buffer_bin, 1, NUMBER_OF_ONE_LINE, src_file);
  62.         gHexFor.len = NUMBER_OF_ONE_LINE;
  63.         if ((low_addr & 0xffff0000) != hign_addr && hign_addr != 0)     //只有大于64K以后才写入扩展线性地址,第一次一般是没有
  64.         {
  65.             hign_addr = low_addr & 0xffff0000;                          
  66.             gHexFor.addr[0] = (uint8_t)((hign_addr & 0xff000000) >> 24);
  67.             gHexFor.addr[1] = (uint8_t)((hign_addr & 0xff0000) >> 16);
  68.             gHexFor.type = 4;                                       
  69.             gHexFor.len = 0;                                            //记录扩展地址            
  70.             tmp = BinFormatEncode(buffer_hex, &gHexFor);
  71.             fwrite(buffer_hex, 1, tmp, dest_file);
  72.             fprintf(dest_file, "\n"); ;
  73.         }
  74.         gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);
  75.         gHexFor.addr[1] = (uint8_t)(low_addr & 0xff);
  76.         gHexFor.type = 0;                                               //数据记录
  77.         tmp = BinFormatEncode(buffer_hex, &gHexFor);
  78.         fwrite(buffer_hex, 1, tmp, dest_file);
  79.         fprintf(dest_file, "\n"); ;
  80.         cur_file_page++;
  81.         low_addr += NUMBER_OF_ONE_LINE;
  82.     }
  83.     if (src_file_remainder != 0)        //最后一次读取的个数不为0,这继续读取
  84.     {
  85.         fread(buffer_bin, 1, src_file_remainder, src_file);
  86.         gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);
  87.         gHexFor.addr[1] = (uint8_t)(low_addr & 0x00ff);
  88.         gHexFor.len = src_file_remainder;
  89.         gHexFor.type = 0;                                               //数据记录                     
  90.         tmp = BinFormatEncode(buffer_hex, &gHexFor);
  91.         fwrite(buffer_hex, 1, tmp, dest_file);
  92.         fprintf(dest_file, "\n"); ;
  93.     }
  94.     gHexFor.addr[0] = 0;               
  95.     gHexFor.addr[1] = 0;
  96.     gHexFor.type = 1;                   //结束符
  97.     gHexFor.len = 0;
  98.     tmp = BinFormatEncode(buffer_hex, &gHexFor);
  99.     fwrite(buffer_hex, 1, tmp, dest_file);
  100.     fprintf(dest_file, "\n"); ;
  101.     fclose(src_file);
  102.     fclose(dest_file);
  103.     return RES_OK;
  104. }</stdio.h>
複製代碼


新建main.c檔案,這裡是帶引數的,主要是方便批處理,是另有用途。
  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. }</stdio.h>
複製代碼


就三個原始檔,編譯生成bin2hex.c檔案。
下面描述下用法:
把bin2hex.c檔案拷貝到c盤根目錄下,再拷貝一個需要轉換的bin檔案,這裡我就拷貝了一個hwb.bin檔案。然後點選選單開始->執行->輸入cmd->進入dos視窗->調整當前目錄為c:,這個如果不知道的可以百度一下,無法就是就是命令cd.
最後輸入命令:bin2hex hwb.bin hwb.hex,輸入後,可以看到提示說轉換成功,大家再檢查下是否有一個hex檔案,本文的程式碼支援大於64K,大家把轉換後的hex下載到微控制器上執行試試看。

bin,檔案,hex,檔案互轉

bin,檔案,hex,檔案互轉



 

臉書網友討論
*滑块验证:
您需要登錄後才可以回帖 登錄 | 註冊 |

本版積分規則



Archiver|手機版|小黑屋|免責聲明|TShopping

GMT+8, 2024-4-19 06:50 , Processed in 0.203037 second(s), 26 queries .

本論壇言論純屬發表者個人意見,與 TShopping綜合論壇 立場無關 如有意見侵犯了您的權益 請寫信聯絡我們。

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回復 返回頂部 返回列表