bin文件轉換為hex文件C語言實現

2022-01-03 嵌入式應用研究院

文章轉載自CSDN,如有侵權,請聯繫刪除,地址:

https://blog.csdn.net/hwb_1988/article/details/45132417

對於嵌入式而言,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文件

#ifndef BIN2HEX_H
#define BIN2HEX_H
typedef unsigned char uint8_t;
typedef unsigned short uint16_t;
typedef unsigned long uint32_t;
/***********************************
*********************************************
就是每次讀寫bin文件N個字節,然後再轉化為hex格式流,hex格式流長度計算方式
: + 長度 + 地址 + 類型 + N個數據(N >= 0) + 校驗
1 + 2 + 4 + 2 + N * 2 + 2
********************************************************************************/
#define NUMBER_OF_ONE_LINE 0x20
#define MAX_BUFFER_OF_ONE_LINE (NUMBER_OF_ONE_LINE * 2 + 11)
typedef struct {
uint8_t len;
uint8_t addr[2];
uint8_t type;
uint8_t *data;
} HexFormat;

typedef enum {
RES_OK = 0, //操作完成
RES_BIN_FILE_NOT_EXIST, //相當於bin文件不存在,包括輸入的路徑可能存在不正確
RES_HEX_FILE_PATH_ERROR //目標文件路徑可能輸入有誤
} RESULT_STATUS;

RESULT_STATUS BinFile2HexFile(char *src, char *dest);
#endif

新建bin2hex.c 文件

#include "bin2hex.h"
#include <stdio.h>
/********************************************************************************
input:
dest: 為轉換後的結果
p->addr[0]: 高地址
p->addr[1]: 低地址
p->type: 記錄類型
p->data: 為bin格式流有效數據指針
p->len: 為bin格式流有效數據長度
output:
返回有效數據的長度
********************************************************************************/
uint16_t BinFormatEncode(uint8_t *dest, HexFormat *p)
{
uint16_t offset = 0;
uint8_t check = 0, num = 0; //:(1) + 長度(2) + 地址(4) + 類型(2)
sprintf(&dest[offset], ":%02X%02X%02X%02X", p->len, p->addr[0], p->addr[1], p->type);
offset += 9; //hex格式流數據指針偏移2
check = p->len + p->addr[0] + p->addr[1] + p->type; //計算校驗和
while (num < p->len) //當數據長度不為0,繼續在之前的hex格式流添加數據
{
sprintf(&dest[offset], "%02X", p->data[num]);
check += p->data[num]; //計算校驗和
offset += 2; //hex格式數據流數據指針偏移2
num++; //下一個字符
}
check = ~check + 1; //反碼+1
sprintf(&dest[offset], "%02X", check);
offset += 2;
return offset; //返回hex格式數據流的長度
}

RESULT_STATUS BinFile2HexFile(char *src, char *dest)
{
FILE *src_file, *dest_file;
uint16_t tmp;
HexFormat gHexFor;
uint32_t low_addr = 0, hign_addr = 0;
uint8_t buffer_bin[NUMBER_OF_ONE_LINE], buffer_hex[MAX_BUFFER_OF_ONE_LINE];
uint32_t src_file_length;
uint16_t src_file_quotient, cur_file_page = 0;
uint8_t src_file_remainder;
src_file = fopen(src, "rb"); //源文件為bin文件,以二進位的形式打開
if (!src_file) //這裡也是相當於用來檢查用戶的輸入是否準備
{
return RES_BIN_FILE_NOT_EXIST;
}
dest_file = fopen(dest, "w"); //目的文件為hex文件,以文本的形式打開
if (!dest_file)
{
return RES_HEX_FILE_PATH_ERROR;
}
fseek(src_file, 0, SEEK_END); //定位到文件末
src_file_length = ftell(src_file);
fseek(src_file, 0, SEEK_SET); //重新定位到開頭,準備開始讀取數據
src_file_quotient = (uint16_t)(src_file_length / NUMBER_OF_ONE_LINE); //商,需要讀取多少次
src_file_remainder = (uint8_t)(src_file_length % NUMBER_OF_ONE_LINE); //餘數,最後一次需要多少個字符
gHexFor.data = buffer_bin; //指向需要轉換的bin數據流
while (cur_file_page < src_file_quotient)
{
fread(buffer_bin, 1, NUMBER_OF_ONE_LINE, src_file);
gHexFor.len = NUMBER_OF_ONE_LINE;
if ((low_addr & 0xffff0000) != hign_addr && hign_addr != 0) //只有大於64K以後才寫入擴展線性地址,第一次一般是沒有
{
hign_addr = low_addr & 0xffff0000;
gHexFor.addr[0] = (uint8_t)((hign_addr & 0xff000000) >> 24);
gHexFor.addr[1] = (uint8_t)((hign_addr & 0xff0000) >> 16);
gHexFor.type = 4;
gHexFor.len = 0; //記錄擴展地址
tmp = BinFormatEncode(buffer_hex, &gHexFor);
fwrite(buffer_hex, 1, tmp, dest_file);
fprintf(dest_file, "\n"); ;
}
gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);
gHexFor.addr[1] = (uint8_t)(low_addr & 0xff);
gHexFor.type = 0; //數據記錄
tmp = BinFormatEncode(buffer_hex, &gHexFor);
fwrite(buffer_hex, 1, tmp, dest_file);
fprintf(dest_file, "\n"); ;
cur_file_page++;
low_addr += NUMBER_OF_ONE_LINE;
}
if (src_file_remainder != 0) //最後一次讀取的個數不為0,這繼續讀取
{
fread(buffer_bin, 1, src_file_remainder, src_file);
gHexFor.addr[0] = (uint8_t)((low_addr & 0xff00) >> 8);
gHexFor.addr[1] = (uint8_t)(low_addr & 0x00ff);
gHexFor.len = src_file_remainder;
gHexFor.type = 0; //數據記錄
tmp = BinFormatEncode(buffer_hex, &gHexFor);
fwrite(buffer_hex, 1, tmp, dest_file);
fprintf(dest_file, "\n"); ;
}
gHexFor.addr[0] = 0;
gHexFor.addr[1] = 0;
gHexFor.type = 1; //結束符
gHexFor.len = 0;
tmp = BinFormatEncode(buffer_hex, &gHexFor);
fwrite(buffer_hex, 1, tmp, dest_file);
fprintf(dest_file, "\n"); ;
fclose(src_file);
fclose(dest_file);
return RES_OK;
}

新建main.c文件,這裡是帶參數的,主要是方便批處理,是另有用途。

#include <stdio.h>
#include "bin2hex.h"
int main(int argc, char *argv[])
{
RESULT_STATUS res;
if (argc != 3)
{
printf("input para doesn't match\r\n");
return -1;
}
res = BinFile2HexFile(argv[1], argv[2]);
switch (res)
{
case RES_OK:
printf("hex file to bin file success!\r\n");
return -1;
case RES_BIN_FILE_NOT_EXIST:
printf("bin file doesn't exist!\r\n");
return -1;
case RES_HEX_FILE_PATH_ERROR:
printf("hex file path is error, please check it!\r\n");
return -1;
}
return 0;
}

就三個源文件,編譯生成bin2hex.c文件。

下面描述下用法:

把bin2hex.c文件拷貝到c盤根目錄下,再拷貝一個需要轉換的bin文件,這裡我就拷貝了一個hwb.bin文件。然後點擊菜單開始->運行->輸入cmd->進入dos窗口->調整當前目錄為c:,這個如果不知道的可以百度一下,無法就是就是命令cd.

最後輸入命令:bin2hex hwb.bin hwb.hex,輸入後,可以看到提示說轉換成功,大家再檢查下是否有一個hex文件,本文的代碼支持大於64K,大家把轉換後的hex下載到單片機上運行試試看。

往期精彩

再談指針:大佬給你撥開 C指針 的雲霧

C語言#和##連接符在項目中的應用(漂亮)

C語言將xxx.bin文件轉為數組

C語言常用的幾種排序

若覺得本次分享的文章對您有幫助,隨手點[在看]並轉發分享,也是對我的支持。

相關焦點

  • 單片機中hex、bin文件的區別
    單片機程序寫好之後,我們都要把程序下載到單片機的內存中,單片機才會按照程式設計師的邏輯執行命令實現功能。之前也講過下載單片機的幾種方式,比如ISP下載,JTAG下載,下載文件的格式最常見的是hex文件,這個格式大家都是知道的,還有一種bin文件是單片機的下載文件。下面介紹這兩種格式的區別。
  • 一文看懂hex文件、bin文件、axf文件的區別
    在STM32開發中,經常會碰到hex文件、bin文件與axf文件,這些都是可以燒寫到板子裡運行的文件。
  • STM32單片機中Hex、Bin文件的區別與應用
    STM32、51等單片機程序經過編譯後,生成的hex文件、bin文件,它們都是單片機燒寫文件,本文介紹它們的區別與應用。
  • 單片機hex和bin文件的區別
    _u=ukgdp5a7629&id=524088004171】ex文件是可以燒寫到單片機中,被單片機執行的一種文件格式,生成Hex文件的方式由很多種,可以通過不同的編譯器將C程序或者彙編程序編譯生成hex。Hex文件如果用特殊的程序來查看(一般記事本就可以實現)。打開後可發現,真箇文件以行為單位,每行以冒號開頭,內容全部為16進位碼。
  • Python 讀取hex文件
    PYTYHON 讀取hex文件藉助bincopy庫,可以讀取hex指定地址的數據。
  • Linux下分析bin文件的10種方法
    $ file /bin/pwd/bin/pwd: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), dynamically linked (uses shared libs), for GNU/Linux 2.6.32, BuildID[sha1]=0d264bacf2adc568f0e21cbcc9576df434c44380
  • MATLAB用GUI計算十進位轉換為二、八、十六進位
    1)dec2bin(A):此函數將十進位數或矩陣轉換為它的二進位形式的字符串。A本身或A的元素(A是矩陣時)都必須小於2的52次方的非負整數(2)dec2bin(A,n):此函數將A轉換成n個字符組成的字符串表示的A的n位二進位數。如果實際轉換成的二進位數的位數小於n,則其餘位上為0,如果實際轉換成的二進位數的位數大於n,則忽略此限制。
  • 最正宗的方法,如何生成單片機程序的bin文件(基於keil)
    圖1還是以孔丙火(微信公眾號:孔丙火)前面文章中的FreeModbus從站設計的單片機程序做為例子,如圖2所示,在生成bin文件的fromelf指令要寫在「After Build/Rebuild」裡面。
  • bin文件怎麼打開?bin是什麼文件?
    bin文件怎麼打開?
  • 【C語言-52】文件操作
    為了實現這種需求,我們可以藉助文件。在需要時,可以將程序中的數據保存為一個文件。待下次重新啟動程序時,從之前保存的文件中提取數據。這樣,程序就不會在重啟後失憶了。為了操作文件,我們需要藉助幾個在頭文件stdio.h中聲明的庫函數。
  • C\C++語言5|控制臺和文件的輸入、輸出
    一、C語言的輸入輸出程序輸入源和輸出目標統稱為設備(device)。無論什麼設備,是用於輸入還是輸出,C語言都通過流進行輸入和輸出操作。流與文件息息相關。這裡的文件指的不是磁碟文件,而是程序待處理的流和實際物理設備之間的媒介。大多數情況下,C語言的初學者無需了解這些文件,因為流、文件、設備之間的交互都由C庫函數和作業系統自動完成。
  • 收集整理的23種文件下載的方式
    PowerShell File DownloadPowerShell 是一種winodws原生的腳本語言,對於熟練使用它的人來說,可以實現很多複雜的功能。下面這兩條指令實現了從Internet網絡下載一個文件。
  • 在 Linux 上分析二進位文件的 10 種方法
    通常情況下,當你用一個應用程式打開一個文件,而它不知道如何處理該文件時,就會出現這種情況。嘗試用 vim 打開一個可執行文件或視頻文件,你屏幕上會看到的只是拋出的亂碼。在 hexdump 中打開未知文件,可以幫助你看到文件的具體內容。你也可以選擇使用一些命令行選項來查看用 ASCII 表示的文件數據。這可能會幫助你了解到它是什麼類型的文件。
  • Unable to open HEX file解決方法及如何生成HEX文件
    經常有童鞋在運行仿真的時候的時候遇到這個問題:Proteus顯示Unable to open HEX file 錯誤。
  • C語言之文件操作
    可能有很多的新手,在遇到格式的問題的時候,會認為直接更改擴展名,就能實現格式轉換。不瞞你們說,我小時候也有過這種想法。但是後來發現,不行。舉個例子,現在有一個 MP3 的文件,要轉成 AAC。這兩個文件從編碼上來講,就是不一樣的。MP3 只能用 MP3 的方式去讀取,AAC 只能用 AAC 的方式去讀取。
  • C語言——文件的基本操作
    C語言基礎合集,點我點我~~~C語言進階合集,點我點我~~~C語言高級:01.C語言實現字符串的加密和解密02.C語言數據結構——鍊表C語言中文件的基本操作包括:文件的打開、文件的關閉以及文件的輸入和輸出。
  • Hexdump 如何工作 | Linux 中國
    用 hexdump 實現 cat 命令閱讀 PNG 規範的時候你可能會注意到頭 8 個字節中的數據與 hexdump 提供的結果看上去不一樣。實際上,那是一樣的數據,但以一種不同的轉換方式展現出來。所以 hexdump 的輸出是正確的,但取決於你在尋找的信息,其輸出結果對你而言不總是直接了當的。
  • C語言操作EXCEL文件(讀寫)
    操作十分複雜,當然我也對這種方法進行了嘗試,也實現了功能,這種方法實現的功能比較多,一般我們只是進行簡單的讀寫操作,所以並不是很想使用這種方法。下面通過C語言讀寫程序來實現。一、寫操作第一步:單純C語言寫入Excel文件只能是 *.csv的後綴文件(是和txt一樣,以二進位文本形式存儲,它是以都逗號分隔符做個單元格內容的劃分, .xls存儲比較複雜, .csv文件可以可以通過.xls或者.xlsx文件另存為,選擇.csv文件格式),它們可以通過Notepad++等記事本軟體當做txt文件打開。
  • C語言 | 文件打開關閉
    「要成為絕世高手,並非一朝一夕,除非是天生武學奇才,但是這種人…萬中無一
  • 乾貨|C語言文件的基本操作!
    本文轉載自【微信公眾號:手機電腦雙黑客,ID:heikestudio】經微信公眾號授權轉載,如需轉載與原文作者聯繫c語言對文件的操作主要分為:按字符操作,按行操作,按內存塊操作主要的函數:fopen():FILE * fopen(_In_z_ const char * _Filename, _In_z_ const char