bin檔案和hex檔案互轉
-
對於嵌入式而言,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 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 #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 檔案
?1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 #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;
}</stdio.h>
新建main.c檔案,這裡是帶引數的,主要是方便批處理,是另有用途。
?1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 #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
;
}</stdio.h>
就三個原始檔,編譯生成bin2hex.c檔案。
下面描述下用法:
把bin2hex.c檔案拷貝到c盤根目錄下,再拷貝一個需要轉換的bin檔案,這裡我就拷貝了一個hwb.bin檔案。然後點選選單開始->執行->輸入cmd->進入dos視窗->調整當前目錄為c:,這個如果不知道的可以百度一下,無法就是就是命令cd.
最後輸入命令:bin2hex hwb.bin hwb.hex,輸入後,可以看到提示說轉換成功,大家再檢查下是否有一個hex檔案,本文的程式碼支援大於64K,大家把轉換後的hex下載到微控制器上執行試試看。
bin檔案轉換為hex檔案C語言實現
對於嵌入式而言,hex檔案可能大家再熟悉不過了,對,我們大學時學習的51微控制器編寫的程式碼在keil上編譯後就生成了hex檔案。那bin檔案又是什麼意思呢,它又和hex檔案的區別在哪?這也不是本文的重點,下面簡單的描述下: 最通俗的來講,hex是帶地址的,用下載對於嵌入式而言,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_Htypedef 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 /********************************************************************************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格式流資料指標偏移2check = 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格式資料流資料指標偏移2num++;//下一個字元}check = ~check + 1; //反碼+1sprintf(&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檔案,這裡是帶引數的,主要是方便批處理,是另有用途。
就三個原始檔,編譯生成bin2hex.c檔案。
下面描述下用法:
把bin2hex.c檔案拷貝到c盤根目錄下,再拷貝一個需要轉換的bin檔案,這裡我就拷貝了一個hwb.bin檔案。然後點選選單開始->執行->輸入cmd->進入dos視窗->調整當前目錄為c:,這個如果不知道的可以百度一下,無法就是就是命令cd.
最後輸入命令:bin2hex hwb.bin hwb.hex,輸入後,可以看到提示說轉換成功,大家再檢查下是否有一個hex檔案,本文的程式碼支援大於64K,大家把轉換後的hex下載到微控制器上執行試試看。
相關推薦
bin檔案和hex檔案互轉
對於嵌入式而言,hex檔案可能大家再熟悉不過了,對,我們大學時學習的51微控制器編寫的程式碼在keil上編譯後就生成了hex檔案。那bin檔案又是什麼意思呢,它又和hex檔案的區別在哪?這也不是本文的重點,下面簡單的描述下: 最通俗的來講,hex是帶地址的,用下載器下載時
元件使用總結:使用 JAXB 實現 XML檔案和java物件互轉
JAXB JAXB:實現xml和java物件互轉 JAXB是一個業界的標準,實現XML檔案和Java物件的互轉。 JAXB是JDK 的組成部分。我們不需要下載第三方jar包 即可做到輕鬆轉換。 複製程式碼 重要類和介面: ○ JAXBContext類,是應用的入口,用於管理XML/Java繫結資訊。
如何把本地可執行檔案和TXT文字互轉
這個程式實現起來比較簡單,有一點要注意,這個想法只適合比較小的可執行檔案: 下面是bin2hexText.py的原始碼: #!/usr/bin/python #encoding: utf-8 import binascii import sys if __n
stm32 bin檔案和hex檔案的生成和下載
1.hex檔案的生成 在如上圖的設定中勾選上生成建立hex檔案即可,通過串列埠可以將hex檔案下載到晶片中,在下載過程中要讓晶片的boot0為1,boot1為0.在下載完成後晶片斷電將boot0置為0,重新上電啟動開發板。 在通過串列埠進行下載時的flymcu軟體的設定為
檔案與base64 字串互轉
這是專案中用到的檔案與base64字串互轉的工具,記錄於此。 import android.util.Base64; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStre
轉:visual studio中標頭檔案和庫檔案路徑設定
visual studio中標頭檔案和庫檔案路徑設定 2017年12月12日 23:06:34 Jimmy1224 閱讀數:8993 在程式開發中,很多時候需要用到別人開發的工具包,如OpenCV和itk。一般而言,在vs中,很少使用原始檔,大部分是使用對類進行宣告的標頭檔案和封裝了類的連結
C語言中 .h檔案和.c檔案的區別 (轉)
要理解.c檔案與.h檔案有什麼不同之處,首先需要弄明白編譯器的工作過程,一般說來編譯器會做以下幾個過程:1.預處理階段2.詞法與語法分析階段3.編譯階段,首先編譯成純彙編語句,再將之彙編成跟CPU相關的二進位制碼,生成各個目標檔案4.連線階段,將各個目標檔案中的各段程式碼進行絕對地址定位,生成跟特定平臺相關
yaml檔案與java bean互轉
轉載至 https://segmentfault.com/a/1190000007743290 yaml格式 name: demo desc: this is desc items: - name: item desc: item desc
utc時間和local 時間互轉
python utc time local time python 1.utc 時間和 local 時間互轉#utc 轉本地時間def utcFlocaltime(utc_time): utc_time = utc_time + datetime.timedelta(hour
C#將結構體和指針互轉的方法
緩沖 style system RR 程序集 ascii 通知 work Coding 1. 功能及位置 將數據從托管對象封送到非托管內存塊,屬於.NET Framework 類庫 命名空間:System.Runtime.InteropServices
Windows批處理檔案(.bat檔案和.cmd檔案)介紹以及簡單使用
首先說一下cmd檔案和bat檔案的區別,從檔案描述中的區別是,cmd檔案叫做:Windows命令指令碼,bat檔案叫:批處理檔案,兩者都可以使用任意一款文字編輯器進行建立、編輯和修改,只是在cmd中支援的命令要多於bat。 批處理檔案是一種沒有什麼固定格式的檔案,他可以處理一條或者多條命令,
身份證號碼15位和18位互轉
身份證各欄位意義: 工作需要,涉及到身份證號碼15位和18位的轉換,功能實現如下: # -*- coding:utf-8 -*- # 驗證身份證最後一位 ##1、將前面的身份證號碼17位數分別乘以不同的係數。從第一位到第十七位的係數分別為:7-9-10-5-8-4-2-1-6-3-7-9-
Java讀檔案和寫檔案
import java.io.*; public class FileTest { String path = "C:\\Users\\Desktop\\test.txt"; File f = new File(path); private void writeFile() throw
python讀取大檔案和普通檔案
讀取檔案,最常見的方式是: with open('filename', 'r', encoding = 'utf-8') as f: for line in f.readlines(): do_something(line) 但是,當完成這一操作時,readlines()
rest介面訪問webService soap介面 用XStream javabean和xml的互轉
建立javabean ,RequestCommonFPKJ @XStreamAlias("REQUEST_COMMON_FPKJ") public class RequestCommonFPKJ { @XStreamAsAttribute //子元
vue引入外部js檔案和scss檔案
最近在做一個vue專案,已經有一段時間了。之前對在vue專案如何引入外部的js檔案和scss檔案一片空白。網上百度也是魚龍混雜,很少有用的東西。後來自己通過一個線上專案學習到了。所以將自己的實現記錄下來 我將寫在vue檔案的js抽出來,放在一個js檔案裡面。然後在vue引入的形式如下(es6的寫
CPP檔案和C檔案混編和將sqlite3加入自己的c++工程
今天嘗試將使用sqlite3資料庫,直接使用sqlite3的原始碼,得到sqlite3.c和sqlite3.h。 我想將他們加入到我的cpp工程裡面 所以我新建了一個mysqlite3.cpp檔案,在裡面呼叫了sqlite3的函式。 下面來說明我遇到的問題及解決方法 一共有兩種編譯方
java讀取檔案和寫入檔案的方式(位元組流和字元流)
java讀取檔案和寫入檔案的方式 以位元組為單位讀取檔案 一次讀一個位元組 一次讀多個位元組 以字元為單位讀取檔案 一次讀一個字元 一次讀多個字元 以位元組為單位讀取檔案 以位元
新增pch 檔案和 header檔案
一 、新增pch檔案 1 ios->other->PCH file,建立一個pch檔案,新增需要引入的標頭檔案名 2 把該檔案的路徑
EXCEL實現時間戳格式和日期格互轉
EXCEL實現時間戳格式和日期格互轉 reference2. https://blog.csdn.net/ch 時間戳轉成正常日期的公式: D1=(A1+8*3600)/86400+70*365+19 其中A1表示當時的1249488000時間戳數值 其中D1就是所需的日期