EDN首页   博客首页

最新日志

发表于:2007-11-1 20:31:16
标签:无标签

1

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

最近在写一个编程器,遇到hex格式文件转换为bin格式文件的问题,通过查阅各类资料,对该格式有了了解。所以在此介绍一下hex格式,并将自己的程序思路写下来,供大家探讨一下。

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

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

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

:10000000FF0462FF051EFF0A93FF0572FF0A93FFBC

       “:”表示一行的开始。

       “:”后的第12个字符“10表示本行包含的数据的长度,这里就是0x1016个。

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

       78个字符“00表示数据的类型。该类型总共有以下几种:

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

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

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

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

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

:00000001FF

       以上的信息其实就足够进行HEXBIN格式的程序的编写。首先我们只处理数据类型为0x000x01的情况。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("\n");

     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;

}

 

点击此处查看原文 >>

系统分类: 单片机   |    用户分类:    |    来源: 原创

评论(2) | 阅读(1041)
发表于:2007-11-1 20:29:43
标签:无标签

1

ISD1700独立按键模式

ISD1700系列语音芯片可工作于按键模式下,此时录放音等功能通过按键实现,如同普通的录放机,下面简单介绍下各个按键功能。

一、录音功能

DIP封装的24脚或TSOP封装的17脚,低电平有效。当外接按钮接地时,按下按钮即开始录音,当放开按钮或者存储空间录满时,停止录音。因此需要录多长时间就得按住按钮多长时间。如果需要录制来自麦克风的声音,则FT引脚需要拉高,如果需要录制来自模拟音频Anain的声音,则FT引脚需要拉抵。

二、播放功能

DIP封装的23脚或TSOP封装的16脚,低电平有效。边沿触发即按一下按钮,则播放当前段录制的声音,如果是电平触发,即长按一下按钮,则循环播放录制的所有段的声音。

三、擦除功能

       DIP封装的25脚或TSOP封装的18脚,低电平有效。边沿触发即按一下按钮,则擦除首段录音或者末段录音,这个要看当前段处于哪一段。如果处于首段则擦除首段,处于末段则擦除末段录音,如果处于中间段,则不做擦除动作。如果是电平触发,即长按一下按钮,则进行整片擦除。因此对于需要擦除中间一部分段的录音,则需要使用SPI接口命令来实现。

四、音量控制

       DIP封装的19脚或TSOP封装的12脚,低电平有效。音量分8级,初始时为最高,按一下键则减一级,等减到0时,再按则递增,如此循环。

五、快进功能

       DIP封装的26脚或TSOP封装的19脚,低电平有效。边沿触发即按一下按钮,则当前播放指针会一到下一个地址段,如果当前段为最后一段,则播放指针会循环转移到首地址段。

六、复位功能

       DIP封装的3脚或TSOP封装的24脚,低电平有效。边沿触发即按一下按钮,则ISD芯片复位,所有寄存器回复的初始状态。

七、直通功能

       DIP封装的22脚或TSOP封装的15脚,低电平有效。FT引脚拉低则模拟通道Anain直接接输出口。该功能可以用来测试一下电路的是否正常工作。

       以上按键功能都是ISD1700芯片的基本功能,只要电路正常,一上电就可以实现。因此可以用按键功能来测试自己设计的电路是否正常,如果连按键功能都实现不了,电路肯定是有问题的。

       特别要注意在录音时,如果从Anain口输入音频,FT引脚必须要拉低,悬空的状态下,ISD芯片的输入口还是麦克风。可以简单的将电脑音频连入到Anain口,进行录音操作。

点击此处查看原文 >>

系统分类: 单片机   |    用户分类:    |    来源: 原创

评论(0) | 阅读(328)
发表于:2007-11-1 20:28:55
标签:无标签

1

ISD1700系列语音芯片及应用简介

语音的应用现在已经是非常普遍了,在生活中无处不在。做为语音功能的实现单元,语音芯片目前也出来很多种。对于语音芯片实现的功能,我们可以简单的理解成一个录放机。你可以录音进去自己的声音,并按自己的要求播放,同时可以擦除及长期的保存。

       在各类芯片中,ISD公司的芯片目前算是比较流行,性价比较高的一款产品。在这里简单介绍一下ISD最新推出的一个语音芯片系列ISD1700。该系列芯片有不同的录音时间,及采样频率供你选择。所谓的采样频率其实就是音质,采样频率越高则单位时间内采集的数据越多,播放时音质显然越好,当然要求的存贮空间就越大,因此同个芯片,采样越高,录音时间就越短。

       ISD1700系列支持两中工作模式:按键模式与SPI模式。所谓按键模式,其实就跟普通录放机工作一样,通过按键来操作。它支持录音、放音、删除、快进、音量控制等按键。对于电路上的实现,其实就是相应管脚接个接地按钮。同时比起以前的ISD录音芯片,增加了提示音的功能,即当你按下录音键时,会自动发出一个声音。当然这个声音也可以进行自定义录音。以上的具体操作还会专门写笔记讨论。

       另一种工作模式其实就是用控制器,如单片机等,对ISD1700芯片进行完全的控制。该模式可以实现按键模式下的所有操作,同时还增加了根据地址进行的一系列分段操作。ISD1700芯片将自己的存储声音数据的空间分成若干个段,每个段都有一个自己的地址。我们可以假设一个芯片有2M存储空间,然后每段分个512K,那么就可以分为4段。对与每段我们可以分别赋地址为0x000x010x100x11,然后就可以通过段地址进行段空间的操作。ISD1700的分段操作跟这个类似,通过段地址,我们可以删除某一部分存储空间,也可以播放或录制某一部分存储空间,这样我们就可以将不同段的声音连起来播放,自由组成相应的语句或音乐。对于SPI的具体操作见以后的专题笔记。

       对于语音芯片,我们可以分析一下它的应用范围。

1.最常见的就是公交地铁的报站系统。它应该就用到了前面提到的分段技术。将一些常用语句录成一段,然后将站名录成一段,到时进行组合,就可以出来不同站的到站提示了。

2.电话语音上应该也是使用这种技术。不光是电话语音留言系统,一些自动服务电话也属于此类。

3.对于一些人机交互做得好的设备也有所应用。比如一些大型设备,操作复杂时,采用语音提示功能,对每一步操作都进行指导,这样就会避免很多错误。

应该还有很多应用,我能想到的大概就这些了。总之语音芯片应用还是很广的,学习