zoukankan      html  css  js  c++  java
  • hex格式介绍及转bin格式的源程序

    源:hex格式介绍及转bin格式的源程序

      Intel HEX文件是记录文本行的ASCII文本文件,在Intel HEX文件中,每一行是一个HEX记录,由十六进制数组成的机器码或者数据常量。Intel HEX文件经常被用于将程序或数据传输存储到ROM、EPROM,大多数编程器和模拟器使用Intel HEX文件。

      很多编译器的支持生成HEX格式的烧录文件,尤其是Keil c。但是编程器能够下载的往往是BIN格式,因此HEX转BIN是每个编程器都必须支持的功能。

      HEX格式文件以行为单位,每行由“:”(0x3a)开始,以回车键结束(0x0d,0x0a)。行内的数据都是由两个字符表示一个16进制字节,比如”01”就表示数0x01;”0a”,就表示0x0a。对于16位的地址,则高位在前低位在后,比如地址0x010a,在HEX格式文件中就表示为字符串”010a”。下面为HEX文件中的一行:

    :10000000FF0462FF051EFF0A93FF0572FF0A93FFBC

      “:”表示一行的开始。

      “:”后的第1,2个字符“10”表示本行包含的数据的长度,这里就是0x10即16个。

      第3,4,5,6个字符“0000”表示数据存储的起始地址,这里表示从0x0000地址开始存储16个数据,其中高位地址在前,低位地址在后。

      第7,8个字符“00”表示数据的类型。该类型总共有以下几种:

    00 ----数据记录       01 ----文件结束记录 02 ----扩展段地址记录 04 ----扩展线性地址记录

    这里就是0x00即为普通数据记录。

    自后的32个字符就是本行包含的数据,每两个字符表示一个字节数据,总共有16个字节数据跟行首的记录的长度相一致。

    最后两个字符表示校验码。

    每个HEX格式的最后一行都是固定为:

    :00000001FF

      以上的信息其实就足够进行HEX转BIN格式的程序的编写。首先我们只处理数据类型为0x00及0x01的情况。0x02表示对应的存储地址超过了64K,由于我的编程器只针对64K以下的单片机,因此在次不处理,0x04也是如此。

      我的编程思路是从文件中一个一个读出字符,根据“:”判断一行的开始,然后每两个字符转换成一个字节,并解释其对应的意义。然后将数据从该行中剥离出来保存到缓冲区中,并最终输出到文件中。

      具体程序如下,该程序在VC2005下采用控制台项目编译,需要在release下编译,在debug模式中会提示一个dll文件无法找到,这可能是VC自身的错误。

    // hextobin.cpp : 定义控制台应用程序的入口点。

    #include "stdafx.h"
    #include <malloc.h>
    #include <memory.h>
    typedef unsigned char BYTE;
    
    //将两个字符转化为一个字节量
    void CharToByte(char* pChar,BYTE* pByte)
    {
        char h,l;
        h=pChar[0];    //高位
        l=pChar[1];    //低位
        if(l>='0'&&l<='9')
            l=l-'0';
        else if(l>='a' && l<='f')
            l=l-'a'+0xa;
        else if(l>='A' && l<='F')
            l=l-'A'+0xa;
        if(h>='0'&&h<='9')
            h=h-'0';
        else if(h>='a' && h<='f')
            h=h-'a'+0xa;
        else if(h>='A' &&h <='F')
            h=h-'A'+0xa;
        
        *pByte=(BYTE)h*16+l;
    }
    
    int _tmain(int argc, _TCHAR* argv[])
    {
        char fileName[100];
        char data[2];
        BYTE *outBuf;
        FILE *myFile;
        int len;
        int i;
        BYTE adressHigh;
        BYTE adressLow;
        BYTE dataLen;
        BYTE dataType; 
        BYTE byteData;
        int totalLen;
        totalLen = 0;
        len = 0;
        adressHigh = 0;
        adressLow = 0;
        dataLen = 0;
        dataType = 0;
        
        printf("请输入HEX格式文件名:");
        scanf_s("%s",fileName);
        printf("
    ");
        
        if (fopen_s(&myFile,fileName,"r") != 0)
        {
            printf("打开文件%s失败!",fileName);
        }
        
        //将文件长度计算出来用于申请存储数据的缓冲区
        while (!feof(myFile))
        {
            ++len;
            fgetc(myFile);
        }
        
        rewind(myFile);
        //因为是每两个字符表示一个字节,所以最大的数据个数要少于文件字符个数的一半
        outBuf = (BYTE*)malloc(len/2);
        memset(outBuf,0xff,len/2);
    
        while (!feof(myFile))
        {
            //:号表示一行的开始
            if (fgetc(myFile) == ':')
            {
                //一行的头两个字符表示该行包含的数据长度
                data[0] = fgetc(myFile);
                data[1] = fgetc(myFile);
                CharToByte(data,&dataLen);
                //一行的第、个字符表示数据存储起始地址的高位
                data[0] = fgetc(myFile);
                data[1] = fgetc(myFile);
                CharToByte(data,&adressHigh);
                
                //一行的第、个字符表示数据存储起始地址的低位
                data[0] = fgetc(myFile);
                data[1] = fgetc(myFile);
                CharToByte(data,&adressLow);
                
                //一行的第、个字符表示数据类型
                data[0] = fgetc(myFile);
                data[1] = fgetc(myFile);
                CharToByte(data,&dataType);
                
                //当数据类型为时,表示本行包含的是普通数据记录
                if (dataType == 0x00)
                {
                    for (i=0;i<dataLen;i++)
                    {
                        data[0] = fgetc(myFile);
                        data[1] = fgetc(myFile);
                        CharToByte(data,&byteData);
                        outBuf[adressHigh*256+adressLow+i] = byteData;
                    }
                    totalLen += dataLen;
                }
                
                //当数据类型为时,表示到了最后一行
                if (dataType == 0x01)
                {
                    printf("文件结束记录!");
                }
                
                //当数据类型为时,表示本行包含的是扩展段地址记录
                if (dataType == 0x02)
                {
                     printf("不支持扩展段地址记录!");
                     return 0;
                }
                
                //当数据类型为时,表示本行包含的是扩展线性地址记录
                if (dataType == 0x04)
                {
                     printf("不支持扩展线性地址记录!");
                     return 0;
                }
            }
        }
    
        fclose(myFile);
        printf("请输入保存的BIN格式文件名:");
        scanf_s("%s",fileName);
        if (fopen_s(&myFile,fileName,"w") != 0)
        {
            printf("打开文件%s失败!",fileName);
        }
        
        for (i=0;i<totalLen;i++)
        {
            fputc(outBuf[i],myFile);
        }
        
        return 0;
        
    }
  • 相关阅读:
    JavaScript算法学习:获取字符串最后一位方法及判断是否以指定字符串开始或结尾
    前端基础知识学习笔记
    JavaScript算法练习:找出字符串中最长的单词并输出其长度
    HTML+CSS学习笔记
    Mint-ui中loadmore(上拉加载下拉刷新)组件在ios中滑动会触发点击事件的解决方法
    JavaScript中的DOM,BOM详细介绍;
    web前端面试题目整理
    Charles https乱码处理
    String类的三个参数的构造方法
    java.net包中的URL类
  • 原文地址:https://www.cnblogs.com/LittleTiger/p/4384808.html
Copyright © 2011-2022 走看看