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

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

文章轉載自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文件的區別
    單片機程序寫好之後,我們都要把程序下載到單片機的內存中,單片機才會按照程式設計師的邏輯執行命令實現功能。
  • 關於bin、hex和axf文件格式的描述
    本文接著上一篇文章講述兩點內容:1.格式轉換2.bin、hex和axf的描述上一篇文章講述的人【MDK-ARM如何生產Bin文件】,只是簡單的講述了實現過程,這裡再簡單提一下有朋友問的問題:fromelf的格式和使用。
  • 輕鬆理解bin、hex、axf和elf文件格式
    作為普通嵌入式軟體開發者,可能只知道如何使用他們,並不會在意這些文件裡面具體是什麼內容。bin 和 hex 大家都不陌生,就是我們下載到晶片的程序文件。bin文件只是單純的程序數據,hex除程序數據之外還有一定格式數據。而 axf 和 bin、 hex 同樣也屬於程序文件,差別在於 axf 具有更多的調試信息。
  • 程序燒錄中SREC、Hex、Bin文件格式有啥區別?一文看懂!
    關於燒錄文件在典型的應用程式中,編譯器或彙編器將程序的原始碼(例如C或彙編語言)轉換為機器代碼,並將其輸出到一個文件中。然後,這個文件由程式設計師導入,以將機器代碼「燒錄」到ROM(或Flash Memory)中,或傳輸到目標系統以進行加載和執行。
  • HEX文件如何直接放在程序數組中是直接使用?
    文件直接在APP中寫入到Flash的指定位置,所以就需要查看BootLoader代碼生成的bin文件的具體內容,然後通過static const uint8_t bootload_src[]__attribute__((at(0x08000000)))={}寫入Flash的指定位置然而網上的bin查看器以及可以打開bin
  • 單片機hex和bin文件的區別
    hex。Hex文件如果用特殊的程序來查看(一般記事本就可以實現)。打開後可發現,整個文件以行為單位,每行以冒號開頭,內容全部為16進位碼。Hex文件可以按照如下的方式進行拆分來分析其中的內容:例如::020000040000FA , 我把它看做 0x02 0x00 0x00 0x04 0x00 0x00 0xFA第一個 0x02 為數據長度。
  • DS80C400的Keil C語言編程
    簡單程序,如網絡揚聲器,可以用彙編語言輕鬆實現,更複雜一點的,如HTTP伺服器這種需要與文件系統交互的程序,可以使用C語言。 文中介紹了如何從Keil的uVision2開發套件著手構建一個DS80C400的C語言應用程式,通過實現一個簡單的HTTP伺服器演示如何使用DS80C400的ROM功能。
  • IAR中eww、ewp、ewd···等各文件的含義和用途
    IAR工程中的原始碼.c.h文件只有幾M大小,為什麼打包之後會有幾百M大小?
  • 【Oracle字符集】識別及轉換導出文件的字符集
    在傳統的EXP導出文件中,記錄著導出使用的字符集id,通過查看導出文件頭的第2、3個字節,我們可以找到十六進位表示的字符集ID,在Windows上,可以使用UltraEdit等工具打開dmp文件,查看其導出字符集,如圖所示:
  • doc文件怎麼轉換為pdf文件
    大家都知道pdf文件是一種跨平臺、輕便式的文件格式,列印的內容可以精確地顯示原來pdf文件中的每一個像素。而word文檔中的doc文件格式轉換為pdf文件格式就成為了很多用戶想要執行的操作。然後右側會出現一個側邊欄,在側邊欄中您可以添加doc文件,將其轉換為pdf文件。點擊右側界面中的「選擇要轉換為pdf的文檔」標籤,如圖所示。在出現的文件選擇對話框中選擇一個doc格式的文件,如圖所示,然後點擊「打開」按鈕打開該文件。
  • M1及M1 plus 上機文件轉換為琪利上機文件的方法
    以儘量少要求STOLL製版師做特殊設置及避免導入後手動修改,導入即可使用為原則,琪利導入001功能嵌入大量自動處理,支持文件類型更多
  • C 語言聖誕樹動態效果實現
    收錄於話題 #c語言 效果圖:看完只許說牛逼!
  • 超級詳細,如何把B站緩存m4s文件轉換成mp4格式
    首先我們將視頻緩存下來,然後找到需要的m4s文件(已經找到m4s文件的可以跳過此步)在B站主界面點擊我的,然後點擊離線緩存,再點擊右上角的設置,進入如下界面然後點擊自定義目錄,設置成你能找到的地方。然後點開想要轉換的視頻,右上角點擊三點符號,出現如下界面,點擊緩存,點擊緩存全部,然後查看緩存,等緩存完成。
  • 單片機的Bootloader,可以實現用戶輕鬆升級程序
    其實就是通過上位機把.bin/hex文件以CAN通訊的方式發送給單片機並存儲在規定的Flash中。這個過程與手機端更新APP類似。以STM8單片機為例,該如何實現Bootloader呢?今天和大家分享一下。Bootloader是一段用於更新自身應用軟體並獨立運行的代碼,常被用於升級產品和修復產品bug。
  • cmd上傳文件的N種方法
    · 觀察作業系統的內容· 查看ROM BIOS的內容· 觀察更改RAM內部的設置值· 以扇區或文件的方式讀寫軟盤數據特別的是它還有一個功能可以將十六進位代碼轉換為可執行文件:的代碼為:>以下代碼保存為.hta文件:
  • Dockerfile 文件全面詳解
    image>[@<digest>] [AS <name>]單個 Dockfile 可以多次出現 FROM,以使用之前的構建階段作為另一個構建階段的依賴項AS name 表示為構建階段命名,在後續 FROM 和 COPY --from=<name
  • 微信公眾號錄音文件(amr文件轉mp3)
    音頻轉碼工具,主要用於將微信語音 amr 格式轉換為 mp3 格式以便在 html5 的 audio 標籤中進行播放
  • 如何將fat32轉換為ntfs文件系統,以提高文件系統的安全性
    大家好,我是「科技跑的快」,今天來聊聊轉換文件系統。NTFS文件系統相對於FAT32文件系統有很大的優點,它是一個基於安全性的文件系統,是Windows NT所採用的文件系統結構,建立在保護文件和目錄數據基礎上,同時可以節省存儲資源、減少磁碟佔用量,是一種先進的文件系統。
  • 【RT-Thread】mkromfs.py將音頻文件製作成C語言數組的虛擬文件
    , 看例程有音頻已轉換C數組數據, 但教程沒講方法,只是說:首先需要下載 romfs.c(本文件包含了兩個音頻文件用於demo播放) 替換 applications 下原有的 romfs.c問原廠(中科藍訊)技術支持, 也沒得到確定答案, 在網上找了很久, 沒找上方法, 聽原廠支持用mkromfs可以轉, romfs.c 注釋:/*