程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C >> 關於C >> bin文件轉換為hex文件C語言實現

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

編輯:關於C

對於嵌入式而言,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 
/********************************************************************************
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 
#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下載到單片機上運行試試看。

\

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved